Files
vscode/src/vs/workbench/api/test/browser/extHostAuthentication.test.ts
Tyler James Leonhardt 6c8d8a2654 Have forceNewSession behave like createIfNone if no sessions are present (#148819)
* Have forceNewSession behave like createIfNone if no sessions are present

* include note in vscode.d.ts about forceNewSesssion behavior
2022-05-05 13:30:14 -07:00

446 lines
13 KiB
TypeScript

/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
import * as assert from 'assert';
import { DisposableStore } from 'vs/base/common/lifecycle';
import { IDialogService } from 'vs/platform/dialogs/common/dialogs';
import { TestDialogService } from 'vs/platform/dialogs/test/common/testDialogService';
import { TestInstantiationService } from 'vs/platform/instantiation/test/common/instantiationServiceMock';
import { INotificationService } from 'vs/platform/notification/common/notification';
import { TestNotificationService } from 'vs/platform/notification/test/common/testNotificationService';
import { IQuickInputHideEvent, IQuickInputService, IQuickPickDidAcceptEvent } from 'vs/platform/quickinput/common/quickInput';
import { IStorageService } from 'vs/platform/storage/common/storage';
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
import { NullTelemetryService } from 'vs/platform/telemetry/common/telemetryUtils';
import { MainThreadAuthentication } from 'vs/workbench/api/browser/mainThreadAuthentication';
import { ExtHostContext, MainContext } from 'vs/workbench/api/common/extHost.protocol';
import { ExtHostAuthentication } from 'vs/workbench/api/common/extHostAuthentication';
import { IActivityService } from 'vs/workbench/services/activity/common/activity';
import { AuthenticationService } from 'vs/workbench/services/authentication/browser/authenticationService';
import { IAuthenticationService } from 'vs/workbench/services/authentication/common/authentication';
import { IExtensionService, nullExtensionDescription as extensionDescription } from 'vs/workbench/services/extensions/common/extensions';
import { IRemoteAgentService } from 'vs/workbench/services/remote/common/remoteAgentService';
import { TestRPCProtocol } from 'vs/workbench/api/test/common/testRPCProtocol';
import { TestQuickInputService, TestRemoteAgentService } from 'vs/workbench/test/browser/workbenchTestServices';
import { TestActivityService, TestExtensionService, TestStorageService } from 'vs/workbench/test/common/workbenchTestServices';
import type { AuthenticationProvider, AuthenticationSession } from 'vscode';
class AuthQuickPick {
private listener: ((e: IQuickPickDidAcceptEvent) => any) | undefined;
public items = [];
public get selectedItems(): string[] {
return this.items;
}
onDidAccept(listener: (e: IQuickPickDidAcceptEvent) => any) {
this.listener = listener;
}
onDidHide(listener: (e: IQuickInputHideEvent) => any) {
}
dispose() {
}
show() {
this.listener!({
inBackground: false
});
}
}
class AuthTestQuickInputService extends TestQuickInputService {
override createQuickPick() {
return <any>new AuthQuickPick();
}
}
class TestAuthProvider implements AuthenticationProvider {
private id = 1;
private sessions = new Map<string, AuthenticationSession>();
onDidChangeSessions = () => { return { dispose() { } }; };
async getSessions(scopes?: readonly string[]): Promise<AuthenticationSession[]> {
if (!scopes) {
return [...this.sessions.values()];
}
if (scopes[0] === 'return multiple') {
return [...this.sessions.values()];
}
const sessions = this.sessions.get(scopes.join(' '));
return sessions ? [sessions] : [];
}
async createSession(scopes: readonly string[]): Promise<AuthenticationSession> {
const scopesStr = scopes.join(' ');
const session = {
scopes,
id: `${this.id}`,
account: {
label: `${this.id}`,
id: `${this.id}`,
},
accessToken: Math.random() + '',
};
this.sessions.set(scopesStr, session);
this.id++;
return session;
}
async removeSession(sessionId: string): Promise<void> {
this.sessions.delete(sessionId);
}
}
suite('ExtHostAuthentication', () => {
let disposables: DisposableStore;
let extHostAuthentication: ExtHostAuthentication;
let instantiationService: TestInstantiationService;
suiteSetup(async () => {
instantiationService = new TestInstantiationService();
instantiationService.stub(IDialogService, new TestDialogService());
instantiationService.stub(IStorageService, new TestStorageService());
instantiationService.stub(IQuickInputService, new AuthTestQuickInputService());
instantiationService.stub(IExtensionService, new TestExtensionService());
instantiationService.stub(IActivityService, new TestActivityService());
instantiationService.stub(IRemoteAgentService, new TestRemoteAgentService());
instantiationService.stub(INotificationService, new TestNotificationService());
instantiationService.stub(ITelemetryService, NullTelemetryService);
const rpcProtocol = new TestRPCProtocol();
instantiationService.stub(IAuthenticationService, instantiationService.createInstance(AuthenticationService));
rpcProtocol.set(MainContext.MainThreadAuthentication, instantiationService.createInstance(MainThreadAuthentication, rpcProtocol));
extHostAuthentication = new ExtHostAuthentication(rpcProtocol);
rpcProtocol.set(ExtHostContext.ExtHostAuthentication, extHostAuthentication);
});
setup(async () => {
disposables = new DisposableStore();
disposables.add(extHostAuthentication.registerAuthenticationProvider('test', 'test provider', new TestAuthProvider()));
disposables.add(extHostAuthentication.registerAuthenticationProvider(
'test-multiple',
'test multiple provider',
new TestAuthProvider(),
{ supportsMultipleAccounts: true }));
});
teardown(() => {
disposables.dispose();
});
test('createIfNone - true', async () => {
const scopes = ['foo'];
const session = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{
createIfNone: true
});
assert.strictEqual(session?.id, '1');
assert.strictEqual(session?.scopes[0], 'foo');
});
test('createIfNone - false', async () => {
const scopes = ['foo'];
const nosession = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{});
assert.strictEqual(nosession, undefined);
// Now create the session
const session = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{
createIfNone: true
});
assert.strictEqual(session?.id, '1');
assert.strictEqual(session?.scopes[0], 'foo');
const session2 = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{});
assert.strictEqual(session.id, session2?.id);
assert.strictEqual(session.scopes[0], session2?.scopes[0]);
assert.strictEqual(session.accessToken, session2?.accessToken);
});
// should behave the same as createIfNone: false
test('silent - true', async () => {
const scopes = ['foo'];
const nosession = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{
silent: true
});
assert.strictEqual(nosession, undefined);
// Now create the session
const session = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{
createIfNone: true
});
assert.strictEqual(session?.id, '1');
assert.strictEqual(session?.scopes[0], 'foo');
const session2 = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{
silent: true
});
assert.strictEqual(session.id, session2?.id);
assert.strictEqual(session.scopes[0], session2?.scopes[0]);
});
test('forceNewSession - true - existing session', async () => {
const scopes = ['foo'];
const session1 = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{
createIfNone: true
});
// Now create the session
const session2 = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{
forceNewSession: true
});
assert.strictEqual(session2?.id, '2');
assert.strictEqual(session2?.scopes[0], 'foo');
assert.notStrictEqual(session1.accessToken, session2?.accessToken);
});
// Should behave like createIfNone: true
test('forceNewSession - true - no existing session', async () => {
const scopes = ['foo'];
const session = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{
forceNewSession: true
});
assert.strictEqual(session?.id, '1');
assert.strictEqual(session?.scopes[0], 'foo');
});
test('forceNewSession - detail', async () => {
const scopes = ['foo'];
const session1 = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{
createIfNone: true
});
// Now create the session
const session2 = await extHostAuthentication.getSession(
extensionDescription,
'test',
scopes,
{
forceNewSession: { detail: 'bar' }
});
assert.strictEqual(session2?.id, '2');
assert.strictEqual(session2?.scopes[0], 'foo');
assert.notStrictEqual(session1.accessToken, session2?.accessToken);
});
//#region Multi-Account AuthProvider
test('clearSessionPreference - true', async () => {
const scopes = ['foo'];
// Now create the session
const session = await extHostAuthentication.getSession(
extensionDescription,
'test-multiple',
scopes,
{
createIfNone: true
});
assert.strictEqual(session?.id, '1');
assert.strictEqual(session?.scopes[0], scopes[0]);
const scopes2 = ['bar'];
const session2 = await extHostAuthentication.getSession(
extensionDescription,
'test-multiple',
scopes2,
{
createIfNone: true
});
assert.strictEqual(session2?.id, '2');
assert.strictEqual(session2?.scopes[0], scopes2[0]);
const session3 = await extHostAuthentication.getSession(
extensionDescription,
'test-multiple',
['return multiple'],
{
clearSessionPreference: true,
createIfNone: true
});
// clearing session preference causes us to get the first session
// because it would normally show a quick pick for the user to choose
assert.strictEqual(session.id, session3?.id);
assert.strictEqual(session.scopes[0], session3?.scopes[0]);
assert.strictEqual(session.accessToken, session3?.accessToken);
});
test('silently getting session should return a session (if any) regardless of preference - fixes #137819', async () => {
const scopes = ['foo'];
// Now create the session
const session = await extHostAuthentication.getSession(
extensionDescription,
'test-multiple',
scopes,
{
createIfNone: true
});
assert.strictEqual(session?.id, '1');
assert.strictEqual(session?.scopes[0], scopes[0]);
const scopes2 = ['bar'];
const session2 = await extHostAuthentication.getSession(
extensionDescription,
'test-multiple',
scopes2,
{
createIfNone: true
});
assert.strictEqual(session2?.id, '2');
assert.strictEqual(session2?.scopes[0], scopes2[0]);
const shouldBeSession1 = await extHostAuthentication.getSession(
extensionDescription,
'test-multiple',
scopes,
{});
assert.strictEqual(session.id, shouldBeSession1?.id);
assert.strictEqual(session.scopes[0], shouldBeSession1?.scopes[0]);
assert.strictEqual(session.accessToken, shouldBeSession1?.accessToken);
const shouldBeSession2 = await extHostAuthentication.getSession(
extensionDescription,
'test-multiple',
scopes2,
{});
assert.strictEqual(session2.id, shouldBeSession2?.id);
assert.strictEqual(session2.scopes[0], shouldBeSession2?.scopes[0]);
assert.strictEqual(session2.accessToken, shouldBeSession2?.accessToken);
});
//#endregion
//#region error cases
test('createIfNone and forceNewSession', async () => {
try {
await extHostAuthentication.getSession(
extensionDescription,
'test',
['foo'],
{
createIfNone: true,
forceNewSession: true
});
assert.fail('should have thrown an Error.');
} catch (e) {
assert.ok(e);
}
});
test('forceNewSession and silent', async () => {
try {
await extHostAuthentication.getSession(
extensionDescription,
'test',
['foo'],
{
forceNewSession: true,
silent: true
});
assert.fail('should have thrown an Error.');
} catch (e) {
assert.ok(e);
}
});
test('createIfNone and silent', async () => {
try {
await extHostAuthentication.getSession(
extensionDescription,
'test',
['foo'],
{
createIfNone: true,
silent: true
});
assert.fail('should have thrown an Error.');
} catch (e) {
assert.ok(e);
}
});
test('Can get multiple sessions (with different scopes) in one extension', async () => {
let session: AuthenticationSession | undefined = await extHostAuthentication.getSession(
extensionDescription,
'test-multiple',
['foo'],
{
createIfNone: true
});
session = await extHostAuthentication.getSession(
extensionDescription,
'test-multiple',
['bar'],
{
createIfNone: true
});
assert.strictEqual(session?.id, '2');
assert.strictEqual(session?.scopes[0], 'bar');
session = await extHostAuthentication.getSession(
extensionDescription,
'test-multiple',
['foo'],
{
createIfNone: false
});
assert.strictEqual(session?.id, '1');
assert.strictEqual(session?.scopes[0], 'foo');
});
//#endregion
});