/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the Source EULA. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ // keytar depends on a native module shipped in vscode, so this is // how we load it import * as keytarType from 'keytar'; import * as vscode from 'vscode'; import Logger from './logger'; import * as nls from 'vscode-nls'; const localize = nls.loadMessageBundle(); function getKeytar(): Keytar | undefined { try { return require('keytar'); } catch (err) { console.log(err); } return undefined; } export type Keytar = { getPassword: typeof keytarType['getPassword']; setPassword: typeof keytarType['setPassword']; deletePassword: typeof keytarType['deletePassword']; }; const SERVICE_ID = `${vscode.env.uriScheme}-github.login`; const ACCOUNT_ID = 'account'; export class Keychain { private keytar: Keytar; constructor() { const keytar = getKeytar(); if (!keytar) { throw new Error('System keychain unavailable'); } this.keytar = keytar; } async setToken(token: string): Promise { try { return await this.keytar.setPassword(SERVICE_ID, ACCOUNT_ID, token); } catch (e) { // Ignore Logger.error(`Setting token failed: ${e}`); const troubleshooting = localize('troubleshooting', "Troubleshooting Guide"); const result = await vscode.window.showErrorMessage(localize('keychainWriteError', "Writing login information to the keychain failed with error '{0}'.", e.message), troubleshooting); if (result === troubleshooting) { vscode.env.openExternal(vscode.Uri.parse('https://code.visualstudio.com/docs/editor/settings-sync#_troubleshooting-keychain-issues')); } } } async getToken(): Promise { try { return await this.keytar.getPassword(SERVICE_ID, ACCOUNT_ID); } catch (e) { // Ignore Logger.error(`Getting token failed: ${e}`); return Promise.resolve(undefined); } } async deleteToken(): Promise { try { return await this.keytar.deletePassword(SERVICE_ID, ACCOUNT_ID); } catch (e) { // Ignore Logger.error(`Deleting token failed: ${e}`); return Promise.resolve(undefined); } } } export const keychain = new Keychain();