From cf2fcc361174432152f59c8c0fe9fdddf81ac0fc Mon Sep 17 00:00:00 2001 From: Alena Khineika Date: Wed, 26 Jul 2023 19:11:10 +0200 Subject: [PATCH] chore: add language server logs VSCODE-447, VSCODE-448, VSCODE-449 (#563) * chore: add language server logs VSCODE-447 * chore: update log message * docs: update comment * feat: grab the active connection if ls recovers from a failure VSCODE-448 * refactor: update output length * docs: update error message * docs: update comments * refactor: remove ls extra initialisation from playground controller * refactor: address PR comments * refactor: rename * test: try to skip the failing test * test: stub trackNewConnection * test: stub trackNewConnection in playground tests * test: more trackNewConnection stubs * feat: add connection id to tree logs * test: skip telemetry test with live connection * docs: link jira ticket * test: clean up --- package-lock.json | 14 +- package.json | 2 +- src/connectionController.ts | 12 +- src/editors/playgroundController.ts | 87 +++++------ src/explorer/explorerController.ts | 6 - src/explorer/explorerTreeController.ts | 6 +- src/explorer/helpExplorer.ts | 5 - src/explorer/playgroundsExplorer.ts | 5 - src/explorer/playgroundsTree.ts | 4 +- src/extension.ts | 30 +++- src/language/languageServerController.ts | 129 +++++++++++----- src/language/mongoDBService.ts | 135 +++++++++++------ src/language/server.ts | 24 ++- src/language/serverCommands.ts | 6 +- src/language/visitor.ts | 6 +- src/logging.ts | 2 +- src/mdbExtensionController.ts | 7 - src/telemetry/telemetryService.ts | 4 +- src/test/suite/connectionController.test.ts | 1 + ...ollectionDocumentsCodeLensProvider.test.ts | 2 +- .../collectionDocumentsProvider.test.ts | 4 + .../editors/playgroundController.test.ts | 45 +----- ...aygroundSelectedCodeActionProvider.test.ts | 10 +- .../suite/explorer/explorerController.test.ts | 4 + .../language/languageServerController.test.ts | 2 +- .../suite/language/mongoDBService.test.ts | 10 +- src/test/suite/mdbExtensionController.test.ts | 8 + src/test/suite/playground.test.ts | 42 +++++- src/test/suite/stubs.ts | 7 +- .../telemetry/connectionTelemetry.test.ts | 4 +- .../suite/views/webviewController.test.ts | 142 ++++++++++++------ 31 files changed, 463 insertions(+), 302 deletions(-) diff --git a/package-lock.json b/package-lock.json index 9b8fb750a..e0fe1d156 100644 --- a/package-lock.json +++ b/package-lock.json @@ -31,7 +31,7 @@ "micromatch": "^4.0.5", "mongodb": "^5.6.0", "mongodb-build-info": "^1.5.0", - "mongodb-cloud-info": "^2.0.1", + "mongodb-cloud-info": "^2.1.0", "mongodb-connection-string-url": "^2.6.0", "mongodb-data-service": "^22.8.0", "mongodb-query-parser": "^2.5.0", @@ -14722,9 +14722,9 @@ } }, "node_modules/mongodb-cloud-info": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/mongodb-cloud-info/-/mongodb-cloud-info-2.0.1.tgz", - "integrity": "sha512-foNUam/t3r2niuN5l9kF9KzNPyiwO9emp8gPN97kxImA+ZumJ37LYTL3XpdCI79CZNUcGtB1WdD9kXGW4CyxXA==", + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mongodb-cloud-info/-/mongodb-cloud-info-2.1.0.tgz", + "integrity": "sha512-IueWuLvkG1xF9Ooxm3blKHVE8x5UL9BYKeCP+VYXNfEzmPruidW5D/5M35Ql5ZedzQhxbZ/RCA55OsuRC7RISw==", "dependencies": { "cross-fetch": "^3.1.6", "gce-ips": "^1.0.2", @@ -34341,9 +34341,9 @@ } }, "mongodb-cloud-info": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/mongodb-cloud-info/-/mongodb-cloud-info-2.0.1.tgz", - "integrity": "sha512-foNUam/t3r2niuN5l9kF9KzNPyiwO9emp8gPN97kxImA+ZumJ37LYTL3XpdCI79CZNUcGtB1WdD9kXGW4CyxXA==", + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mongodb-cloud-info/-/mongodb-cloud-info-2.1.0.tgz", + "integrity": "sha512-IueWuLvkG1xF9Ooxm3blKHVE8x5UL9BYKeCP+VYXNfEzmPruidW5D/5M35Ql5ZedzQhxbZ/RCA55OsuRC7RISw==", "requires": { "cross-fetch": "^3.1.6", "gce-ips": "^1.0.2", diff --git a/package.json b/package.json index 70bf85baa..8805c923e 100644 --- a/package.json +++ b/package.json @@ -986,7 +986,7 @@ "micromatch": "^4.0.5", "mongodb": "^5.6.0", "mongodb-build-info": "^1.5.0", - "mongodb-cloud-info": "^2.0.1", + "mongodb-cloud-info": "^2.1.0", "mongodb-connection-string-url": "^2.6.0", "mongodb-data-service": "^22.8.0", "mongodb-query-parser": "^2.5.0", diff --git a/src/connectionController.ts b/src/connectionController.ts index 0da160909..b3250d269 100644 --- a/src/connectionController.ts +++ b/src/connectionController.ts @@ -110,6 +110,7 @@ export default class ConnectionController { } = Object.create(null); _activeDataService: DataService | null = null; _storageController: StorageController; + _telemetryService: TelemetryService; private readonly _serviceName = 'mdb.vscode.savedConnections'; private _currentConnectionId: null | string = null; @@ -125,7 +126,6 @@ export default class ConnectionController { private _disconnecting = false; private _statusView: StatusView; - private _telemetryService: TelemetryService; // Used by other parts of the extension that respond to changes in the connections. private eventEmitter: EventEmitter = new EventEmitter(); @@ -509,10 +509,18 @@ export default class ConnectionController { this.eventEmitter.emit(DataServiceEventTypes.CONNECTIONS_DID_CHANGE); if (this._activeDataService) { + log.info('Disconnecting from the previous connection...', { + connectionId: this._currentConnectionId, + }); await this.disconnect(); } this._statusView.showMessage('Connecting to MongoDB...'); + log.info('Connecting to MongoDB...', { + connectionInfo: JSON.stringify( + extractSecrets(this._connections[connectionId]).connectionInfo + ), + }); const connectionOptions = this._connections[connectionId].connectionOptions; @@ -551,7 +559,7 @@ export default class ConnectionController { throw connectError; } - log.info('Successfully connected'); + log.info('Successfully connected', { connectionId }); void vscode.window.showInformationMessage('MongoDB connection successful.'); this._activeDataService = dataService; diff --git a/src/editors/playgroundController.ts b/src/editors/playgroundController.ts index 830f479df..2f843d9d0 100644 --- a/src/editors/playgroundController.ts +++ b/src/editors/playgroundController.ts @@ -152,7 +152,7 @@ export default class PlaygroundController { this._connectionController.addEventListener( DataServiceEventTypes.ACTIVE_CONNECTION_CHANGED, () => { - void this._connectToServiceProvider(); + void this._activeConnectionChanged(); } ); @@ -163,14 +163,15 @@ export default class PlaygroundController { this._playgroundResultViewColumn = editor.viewColumn; this._playgroundResultTextDocument = editor?.document; } + const isPlaygroundEditor = isPlayground(editor?.document.uri); void vscode.commands.executeCommand( 'setContext', 'mdb.isPlayground', - isPlayground(editor?.document.uri) + isPlaygroundEditor ); - if (editor?.document.languageId !== 'Log') { + if (isPlaygroundEditor) { this._activeTextEditor = editor; this._activeConnectionCodeLensProvider.setActiveTextEditor( this._activeTextEditor @@ -178,7 +179,10 @@ export default class PlaygroundController { this._playgroundSelectedCodeActionProvider.setActiveTextEditor( this._activeTextEditor ); - log.info('Active editor path', editor?.document.uri?.path); + log.info('Active editor', { + documentPath: editor?.document.uri?.path, + documentLanguageId: editor?.document.languageId, + }); } }; @@ -245,35 +249,24 @@ export default class PlaygroundController { ); } - async _connectToServiceProvider(): Promise { - // Disconnect if already connected. - await this._languageServerController.disconnectFromServiceProvider(); - + async _activeConnectionChanged(): Promise { const dataService = this._connectionController.getActiveDataService(); const connectionId = this._connectionController.getActiveConnectionId(); + let mongoClientOption; - if (!dataService || !connectionId) { - this._activeConnectionCodeLensProvider.refresh(); - - return; - } - - const mongoClientOption = - this._connectionController.getMongoClientConnectionOptions(); - - if (!mongoClientOption) { - this._activeConnectionCodeLensProvider.refresh(); + this._activeConnectionCodeLensProvider.refresh(); - return; + if (dataService && connectionId) { + mongoClientOption = + this._connectionController.getMongoClientConnectionOptions(); } - await this._languageServerController.connectToServiceProvider({ + // The connectionId is null when disconnecting. + await this._languageServerController.activeConnectionChanged({ connectionId, - connectionString: mongoClientOption.url, - connectionOptions: mongoClientOption.options, + connectionString: mongoClientOption?.url, + connectionOptions: mongoClientOption?.options, }); - - this._activeConnectionCodeLensProvider.refresh(); } async _createPlaygroundFileWithContent( @@ -420,36 +413,28 @@ export default class PlaygroundController { this._statusView.showMessage('Getting results...'); + let result: ShellEvaluateResult; try { // Send a request to the language server to execute scripts from a playground. - const result: ShellEvaluateResult = - await this._languageServerController.evaluate({ - codeToEvaluate, - connectionId, - }); - - this._statusView.hideMessage(); - this._telemetryService.trackPlaygroundCodeExecuted( - result, - this._isPartialRun, - result ? false : true - ); - - return result; - } catch (err: any) { - // We re-initialize the language server when we encounter an error. - // This happens when the language server worker runs out of memory, can't be revitalized, and restarts. - if (err?.code === -32097) { - void vscode.window.showErrorMessage( - 'An error occurred when running the playground. This can occur when the playground runner runs out of memory.' - ); + result = await this._languageServerController.evaluate({ + codeToEvaluate, + connectionId, + }); + } catch (error) { + const msg = + 'An internal error has occurred. The playground services have been restored. This can occur when the playground runner runs out of memory.'; + log.error(msg, error); + void vscode.window.showErrorMessage(msg); + } - await this._languageServerController.startLanguageServer(); - void this._connectToServiceProvider(); - } + this._statusView.hideMessage(); + this._telemetryService.trackPlaygroundCodeExecuted( + result, + this._isPartialRun, + result ? false : true + ); - throw err; - } + return result; } _getAllText(): string { diff --git a/src/explorer/explorerController.ts b/src/explorer/explorerController.ts index e7a810c70..5e56aaa78 100644 --- a/src/explorer/explorerController.ts +++ b/src/explorer/explorerController.ts @@ -5,10 +5,6 @@ import ConnectionController, { } from '../connectionController'; import ExplorerTreeController from './explorerTreeController'; -import { createLogger } from '../logging'; - -const log = createLogger('explorer controller'); - export default class ExplorerController { private _connectionController: ConnectionController; private _treeController: ExplorerTreeController; @@ -38,14 +34,12 @@ export default class ExplorerController { }; activateConnectionsTreeView(): void { - log.info('Activating explorer controller...'); // Listen for a change in connections to occur before we create the tree // so that we show the `viewsWelcome` before any connections are added. this._connectionController.addEventListener( DataServiceEventTypes.CONNECTIONS_DID_CHANGE, this.createTreeView ); - log.info('Explorer controller activated'); } deactivate(): void { diff --git a/src/explorer/explorerTreeController.ts b/src/explorer/explorerTreeController.ts index 153374f4f..7db23f3c9 100644 --- a/src/explorer/explorerTreeController.ts +++ b/src/explorer/explorerTreeController.ts @@ -64,7 +64,11 @@ export default class ExplorerTreeController }); treeView.onDidExpandElement(async (event: any): Promise => { - log.info('Tree item was expanded', event.element.label); + log.info('Connection tree item was expanded', { + connectionId: event.element.connectionId, + connectionName: event.element.label, + isExpanded: event.element.isExpanded, + }); if (!event.element.onDidExpand) { return; diff --git a/src/explorer/helpExplorer.ts b/src/explorer/helpExplorer.ts index f40ae0734..8a0e791f7 100644 --- a/src/explorer/helpExplorer.ts +++ b/src/explorer/helpExplorer.ts @@ -1,10 +1,7 @@ import * as vscode from 'vscode'; import HelpTree from './helpTree'; -import { createLogger } from '../logging'; import { TelemetryService } from '../telemetry'; -const log = createLogger('help and info explorer'); - export default class HelpExplorer { _treeController: HelpTree; _treeView?: vscode.TreeView; @@ -15,7 +12,6 @@ export default class HelpExplorer { activateHelpTreeView(telemetryService: TelemetryService): void { if (!this._treeView) { - log.info('Activating help explorer...'); this._treeView = vscode.window.createTreeView('mongoDBHelpExplorer', { treeDataProvider: this._treeController, }); @@ -23,7 +19,6 @@ export default class HelpExplorer { this._treeView, telemetryService ); - log.info('Help explorer activated'); } } diff --git a/src/explorer/playgroundsExplorer.ts b/src/explorer/playgroundsExplorer.ts index aaa4d3972..99e51ff3c 100644 --- a/src/explorer/playgroundsExplorer.ts +++ b/src/explorer/playgroundsExplorer.ts @@ -1,8 +1,5 @@ import * as vscode from 'vscode'; import PlaygroundsTree from './playgroundsTree'; -import { createLogger } from '../logging'; - -const log = createLogger('playgrounds explorer'); export default class PlaygroundsExplorer { private _treeController: PlaygroundsTree; @@ -25,9 +22,7 @@ export default class PlaygroundsExplorer { }; public activatePlaygroundsTreeView(): void { - log.info('Activating playgrounds explorer...'); this.createPlaygroundsTreeView(); - log.info('Playgrounds explorer activated'); } public deactivate(): void { diff --git a/src/explorer/playgroundsTree.ts b/src/explorer/playgroundsTree.ts index caed6becd..b613afe94 100644 --- a/src/explorer/playgroundsTree.ts +++ b/src/explorer/playgroundsTree.ts @@ -50,7 +50,9 @@ export default class PlaygroundsTree }); treeView.onDidExpandElement(async (event: any): Promise => { - log.info('Tree item was expanded', event.element.label); + log.info('Playground tree item was expanded', { + playgroundName: event.element.label, + }); if (!event.element.onDidExpand) { return; diff --git a/src/extension.ts b/src/extension.ts index f0d7b576b..ae57ea4d3 100644 --- a/src/extension.ts +++ b/src/extension.ts @@ -5,6 +5,8 @@ import * as vscode from 'vscode'; import { ext } from './extensionConstants'; import { createKeytar } from './utils/keytar'; import { createLogger } from './logging'; +// eslint-disable-next-line @typescript-eslint/no-var-requires +const { version } = require('../package.json'); const log = createLogger('extension'); @@ -27,15 +29,38 @@ let mdbExtension: MDBExtensionController; export async function activate( context: vscode.ExtensionContext ): Promise { - log.info('Activating extension...'); ext.context = context; + let hasKeytar = false; try { ext.keytarModule = createKeytar(); + hasKeytar = true; } catch (err) { // Couldn't load keytar, proceed without storing & loading connections. } + const defaultConnectionSavingLocation = vscode.workspace + .getConfiguration('mdb.connectionSaving') + .get('defaultConnectionSavingLocation'); + + log.info('Activating extension...', { + id: context.extension.id, + version: version, + mode: vscode.ExtensionMode[context.extensionMode], + kind: vscode.ExtensionKind[context.extension.extensionKind], + extensionPath: context.extensionPath, + logPath: context.logUri.path, + workspaceStoragePath: context.storageUri?.path, + globalStoragePath: context.globalStorageUri.path, + defaultConnectionSavingLocation, + hasKeytar, + buildInfo: { + nodeVersion: process.version, + runtimePlatform: process.platform, + runtimeArch: process.arch, + }, + }); + mdbExtension = new MDBExtensionController(context, { shouldTrackTelemetry: true, }); @@ -43,12 +68,11 @@ export async function activate( // Add our extension to a list of disposables for when we are deactivated. context.subscriptions.push(mdbExtension); - - log.info('Extension activated'); } // Called when our extension is deactivated. export async function deactivate(): Promise { + log.info('Deactivating extension...'); if (mdbExtension) { await mdbExtension.deactivate(); } diff --git a/src/language/languageServerController.ts b/src/language/languageServerController.ts index 64a0dd66c..719760e46 100644 --- a/src/language/languageServerController.ts +++ b/src/language/languageServerController.ts @@ -34,13 +34,15 @@ export default class LanguageServerController { _source?: CancellationTokenSource; _isExecutingInProgress = false; _client: LanguageClient; + _currentConnectionId: string | null = null; + _currentConnectionString?: string; + _currentConnectionOptions?: MongoClientOptions; constructor(context: ExtensionContext) { this._context = context; - // The server is implemented in node. - const serverModule = path.join( - this._context.extensionPath, + const languageServerPath = path.join( + context.extensionPath, 'dist', 'languageServer.js' ); @@ -53,45 +55,52 @@ export default class LanguageServerController { // If the extension is launched in debug mode then the debug server options are used. // Otherwise the run options are used. const serverOptions: ServerOptions = { - run: { module: serverModule, transport: TransportKind.ipc }, + run: { module: languageServerPath, transport: TransportKind.ipc }, debug: { - module: serverModule, + module: languageServerPath, transport: TransportKind.ipc, options: debugOptions, }, }; + const languageServerId = 'mongodbLanguageServer'; + const languageServerName = 'MongoDB Language Server'; + // Define the document patterns to register the language server for. + const documentSelector = [ + { pattern: '**/*.mongodb.js' }, + { pattern: '**/*.mongodb' }, + ]; // Options to control the language client. const clientOptions: LanguageClientOptions = { - // Register the language server for mongodb documents. - documentSelector: [ - { pattern: '**/*.mongodb.js' }, - { pattern: '**/*.mongodb' }, - ], + documentSelector: documentSelector, synchronize: { // Notify the server about file changes in the workspace. fileEvents: workspace.createFileSystemWatcher('**/*'), }, - outputChannel: vscode.window.createOutputChannel( - 'MongoDB Language Server' - ), + outputChannel: vscode.window.createOutputChannel(languageServerName), }; - log.info('Create MongoDB Language Server', { - serverOptions, - clientOptions, + log.info('Creating MongoDB Language Server...', { + extensionPath: context.extensionPath, + languageServer: { + id: languageServerId, + name: languageServerName, + path: languageServerPath, + documentSelector: JSON.stringify(documentSelector), + }, }); // Create the language server client. this._client = new LanguageClient( - 'mongodbLanguageServer', - 'MongoDB Language Server', + languageServerId, + languageServerName, serverOptions, clientOptions ); } async startLanguageServer(): Promise { + log.info('Starting the language server...'); // Start the client. This will also launch the server. await this._client.start(); @@ -101,15 +110,34 @@ export default class LanguageServerController { this._context.subscriptions.push(this._client); } - // Subscribe on notifications from the server when the client is ready. - await this._client.sendRequest( - ServerCommands.SET_EXTENSION_PATH, - this._context.extensionPath - ); + // Subscribe on notifications from the server when the MongoDBService is ready. + // If the connection to server got closed, server will restart, + // but we also need to re-send default configurations + // https://jira.mongodb.org/browse/VSCODE-448 + this._client.onNotification(ServerCommands.MONGODB_SERVICE_CREATED, () => { + const msg = this._currentConnectionId + ? 'MongoDBService restored from an internal error' + : 'MongoDBService initialized'; + log.info( + `${msg}. Sending default settings... ${JSON.stringify({ + extensionPath: this._context.extensionPath, + connectionId: this._currentConnectionId, + hasConnectionString: !!this._currentConnectionString, + hasConnectionOptions: !!this._currentConnectionOptions, + })}` + ); + void this._client.sendRequest(ServerCommands.INITIALIZE_MONGODB_SERVICE, { + extensionPath: this._context.extensionPath, + connectionId: this._currentConnectionId, + connectionString: this._currentConnectionString, + connectionOptions: this._currentConnectionOptions, + }); + }); this._client.onNotification( ServerCommands.SHOW_INFO_MESSAGE, (messsage) => { + log.info('The info message shown to a user', messsage); void vscode.window.showInformationMessage(messsage); } ); @@ -117,13 +145,16 @@ export default class LanguageServerController { this._client.onNotification( ServerCommands.SHOW_ERROR_MESSAGE, (messsage) => { + log.info('The error message shown to a user', messsage); void vscode.window.showErrorMessage(messsage); } ); } deactivate(): Thenable | undefined { + log.info('Deactivating the language server...'); if (!this._client) { + log.info('The LanguageServerController client is not found'); return undefined; } @@ -134,6 +165,10 @@ export default class LanguageServerController { async evaluate( playgroundExecuteParameters: PlaygroundEvaluateParams ): Promise { + log.info('Running a playground...', { + connectionId: playgroundExecuteParameters.connectionId, + inputLength: playgroundExecuteParameters.codeToEvaluate.length, + }); this._isExecutingInProgress = true; // Instantiate a new CancellationTokenSource object @@ -143,7 +178,7 @@ export default class LanguageServerController { // Send a request with a cancellation token // to the language server instance to execute scripts from a playground // and return results to the playground controller when ready. - const result: ShellEvaluateResult = await this._client.sendRequest( + const res: ShellEvaluateResult = await this._client.sendRequest( ServerCommands.EXECUTE_CODE_FROM_PLAYGROUND, playgroundExecuteParameters, this._source.token @@ -151,7 +186,16 @@ export default class LanguageServerController { this._isExecutingInProgress = false; - return result; + log.info('Evaluate response', { + namespace: res?.result?.namespace, + type: res?.result?.type, + outputLength: res?.result?.content + ? JSON.stringify(res.result.content).length + : 0, + language: res?.result?.language, + }); + + return res; } async getExportToLanguageMode( @@ -172,24 +216,34 @@ export default class LanguageServerController { ); } - async connectToServiceProvider(params: { - connectionId: string; - connectionString: string; - connectionOptions: MongoClientOptions; + async activeConnectionChanged({ + connectionId, + connectionString, + connectionOptions, + }: { + connectionId: null | string; + connectionString?: string; + connectionOptions?: MongoClientOptions; }): Promise { - await this._client.sendRequest( - ServerCommands.CONNECT_TO_SERVICE_PROVIDER, - params - ); - } + log.info('Changing MongoDBService active connection...', { connectionId }); - async disconnectFromServiceProvider(): Promise { - await this._client.sendRequest( - ServerCommands.DISCONNECT_TO_SERVICE_PROVIDER + this._currentConnectionId = connectionId; + this._currentConnectionString = connectionString; + this._currentConnectionOptions = connectionOptions; + + const res = await this._client.sendRequest( + ServerCommands.ACTIVE_CONNECTION_CHANGED, + { + connectionId, + connectionString, + connectionOptions, + } ); + log.info('MongoDBService active connection has changed', res); } async resetCache(clear: ClearCompletionsCache): Promise { + log.info('Reseting MongoDBService cache...', clear); await this._client.sendRequest( ServerCommands.CLEAR_CACHED_COMPLETIONS, clear @@ -197,6 +251,7 @@ export default class LanguageServerController { } cancelAll(): void { + log.info('Canceling a playground...'); // Send a request for cancellation. As a result // the associated CancellationToken will be notified of the cancellation, // the onCancellationRequested event will be fired, diff --git a/src/language/mongoDBService.ts b/src/language/mongoDBService.ts index 1f0c9d4bf..a12584978 100644 --- a/src/language/mongoDBService.ts +++ b/src/language/mongoDBService.ts @@ -51,17 +51,17 @@ const SET_WINDOW_FIELDS = '$setWindowFields'; export const languageServerWorkerFileName = 'languageServerWorker.js'; interface ServiceProviderParams { - connectionId: string; - connectionString: string; - connectionOptions: MongoClientOptions; + connectionId: string | null; + connectionString?: string; + connectionOptions?: MongoClientOptions; } export default class MongoDBService { - _extensionPath?: string; + _extensionPath?: string; // The absolute file path of the directory containing the extension. _connection: Connection; - _connectionId?: string; - _connectionString?: string; - _connectionOptions?: MongoClientOptions; + _currentConnectionId: string | null = null; + _currentConnectionString?: string; + _currentConnectionOptions?: MongoClientOptions; _databaseCompletionItems: CompletionItem[] = []; _shellSymbolCompletionItems: { [symbol: string]: CompletionItem[] } = {}; @@ -73,6 +73,8 @@ export default class MongoDBService { _serviceProvider?: CliServiceProvider; constructor(connection: Connection) { + connection.console.log('MongoDBService initializing...'); + this._connection = connection; this._visitor = new Visitor(); @@ -84,42 +86,93 @@ export default class MongoDBService { * The connectionString used by LS to connect to MongoDB. */ get connectionString(): string | undefined { - return this._connectionString; + return this._currentConnectionString; } /** * The connectionOptions used by LS to connect to MongoDB. */ get connectionOptions(): MongoClientOptions | undefined { - return this._connectionOptions; + return this._currentConnectionOptions; } - /** - * The absolute file path of the directory containing the extension. - */ - setExtensionPath(extensionPath: string): void { + initialize({ + extensionPath, + connectionId, + connectionString, + connectionOptions, + }): void { this._extensionPath = extensionPath; + this._currentConnectionId = connectionId; + this._currentConnectionString = connectionString; + this._currentConnectionOptions = connectionOptions; + this._connection.console.log( + `MongoDBService initialized ${JSON.stringify({ + extensionPath, + connectionId, + hasConnectionString: !!connectionString, + hasConnectionOptions: !!connectionOptions, + })}` + ); } /** - * Connect to CliServiceProvider. + * Change CliServiceProvider active connection. */ - async connectToServiceProvider({ + async activeConnectionChanged({ connectionId, connectionString, connectionOptions, - }: ServiceProviderParams): Promise { - // If already connected close the previous connection. - await this.disconnectFromServiceProvider(); - - this._connectionId = connectionId; - this._connectionString = connectionString; - this._connectionOptions = connectionOptions; - this._serviceProvider = await CliServiceProvider.connect( - connectionString, - connectionOptions + }: ServiceProviderParams): Promise<{ + connectionId: string | null; + successfullyConnected: boolean; + connectionErrorMessage?: string; + }> { + this._connection.console.log( + `Changing CliServiceProvider active connection... ${JSON.stringify({ + currentConnectionId: this._currentConnectionId, + newConnectionId: connectionId, + hasConnectionString: !!connectionString, + hasConnectionOptions: !!connectionOptions, + })}` ); + // If already connected close the previous connection. + if ( + this._currentConnectionId && + this._currentConnectionId !== connectionId + ) { + this.clearCachedCompletions({ + databases: true, + collections: true, + fields: true, + }); + await this._closeCurrentConnection(); + } + + this._currentConnectionId = connectionId; + this._currentConnectionString = connectionString; + this._currentConnectionOptions = connectionOptions; + + if (connectionId && (!connectionString || !connectionOptions)) { + this._connection.console.error( + 'Failed to change CliServiceProvider active connection: connectionString and connectionOptions are required' + ); + return { + connectionId, + successfullyConnected: false, + connectionErrorMessage: + 'connectionString and connectionOptions are required', + }; + } + + if (connectionString && connectionOptions) { + this._serviceProvider = await CliServiceProvider.connect( + connectionString, + connectionOptions + ); + } + try { // Get database names for the current connection. const databases = await this._getDatabases(); @@ -130,18 +183,14 @@ export default class MongoDBService { `LS get databases error: ${util.inspect(error)}` ); } - } - /** - * Disconnect from CliServiceProvider. - */ - async disconnectFromServiceProvider(): Promise { - this.clearCachedCompletions({ - databases: true, - collections: true, - fields: true, - }); - await this._clearCurrentConnection(); + this._connection.console.log( + `CliServiceProvider active connection has changed: { connectionId: ${connectionId} }` + ); + return { + successfullyConnected: true, + connectionId, + }; } /** @@ -154,7 +203,7 @@ export default class MongoDBService { this.clearCachedFields(); return new Promise((resolve) => { - if (this._connectionId !== params.connectionId) { + if (this._currentConnectionId !== params.connectionId) { void this._connection.sendNotification( ServerCommands.SHOW_ERROR_MESSAGE, "The playground's active connection does not match the extension's active connection. Please reconnect and try again." @@ -194,9 +243,6 @@ export default class MongoDBService { languageServerWorkerFileName ) ); - this._connection.console.log( - `WORKER thread is created on path: ${this._extensionPath}` - ); worker?.on( 'message', @@ -1060,12 +1106,11 @@ export default class MongoDBService { this._collections = {}; } - async _clearCurrentConnection(): Promise { - this._connectionId = undefined; - this._connectionString = undefined; - this._connectionOptions = undefined; - + async _closeCurrentConnection(): Promise { if (this._serviceProvider) { + this._connection.console.log( + `Disconnecting from a previous connection... { connectionId: ${this._currentConnectionId} }` + ); const serviceProvider = this._serviceProvider; this._serviceProvider = undefined; await serviceProvider.close(true); diff --git a/src/language/server.ts b/src/language/server.ts index 284572cd3..0bef401a4 100644 --- a/src/language/server.ts +++ b/src/language/server.ts @@ -79,6 +79,11 @@ connection.onInitialize((params: InitializeParams) => { }); connection.onInitialized(() => { + void connection.sendNotification( + ServerCommands.MONGODB_SERVICE_CREATED, + 'An instance of MongoDBService is created' + ); + if (hasConfigurationCapability) { // Register for all configuration changes. void connection.client.register( @@ -161,21 +166,14 @@ connection.onRequest( } ); -// Pass the extension path to the MongoDB service. -connection.onRequest(ServerCommands.SET_EXTENSION_PATH, (extensionPath) => { - mongoDBService.setExtensionPath(extensionPath); -}); - -// Connect the MongoDB language service to CliServiceProvider -// using the current connection of the client. -connection.onRequest(ServerCommands.CONNECT_TO_SERVICE_PROVIDER, (params) => { - return mongoDBService.connectToServiceProvider(params); +// Send default configurations to mongoDBService. +connection.onRequest(ServerCommands.INITIALIZE_MONGODB_SERVICE, (settings) => { + mongoDBService.initialize(settings); }); -// Clear connectionString and connectionOptions values -// when there is no active connection. -connection.onRequest(ServerCommands.DISCONNECT_TO_SERVICE_PROVIDER, () => { - return mongoDBService.disconnectFromServiceProvider(); +// Change CliServiceProvider active connection. +connection.onRequest(ServerCommands.ACTIVE_CONNECTION_CHANGED, (params) => { + return mongoDBService.activeConnectionChanged(params); }); // Set fields for tests. diff --git a/src/language/serverCommands.ts b/src/language/serverCommands.ts index 418ea914a..883229dc6 100644 --- a/src/language/serverCommands.ts +++ b/src/language/serverCommands.ts @@ -1,15 +1,15 @@ export enum ServerCommands { - CONNECT_TO_SERVICE_PROVIDER = 'CONNECT_TO_SERVICE_PROVIDER', - DISCONNECT_TO_SERVICE_PROVIDER = 'DISCONNECT_TO_SERVICE_PROVIDER', + ACTIVE_CONNECTION_CHANGED = 'ACTIVE_CONNECTION_CHANGED', EXECUTE_CODE_FROM_PLAYGROUND = 'EXECUTE_CODE_FROM_PLAYGROUND', EXECUTE_RANGE_FROM_PLAYGROUND = 'EXECUTE_RANGE_FROM_PLAYGROUND', - SET_EXTENSION_PATH = 'SET_EXTENSION_PATH', SHOW_ERROR_MESSAGE = 'SHOW_ERROR_MESSAGE', SHOW_INFO_MESSAGE = 'SHOW_INFO_MESSAGE', GET_NAMESPACE_FOR_SELECTION = 'GET_NAMESPACE_FOR_SELECTION', GET_EXPORT_TO_LANGUAGE_MODE = 'GET_EXPORT_TO_LANGUAGE_MODE', UPDATE_CURRENT_SESSION_FIELDS = 'UPDATE_CURRENT_SESSION_FIELDS', CLEAR_CACHED_COMPLETIONS = 'CLEAR_CACHED_COMPLETIONS', + MONGODB_SERVICE_CREATED = 'MONGODB_SERVICE_CREATED', + INITIALIZE_MONGODB_SERVICE = 'INITIALIZE_MONGODB_SERVICE', } export type PlaygroundRunParameters = { diff --git a/src/language/visitor.ts b/src/language/visitor.ts index c49d8994d..75433e7fa 100644 --- a/src/language/visitor.ts +++ b/src/language/visitor.ts @@ -1,7 +1,6 @@ import type * as babel from '@babel/core'; import * as parser from '@babel/parser'; import traverse from '@babel/traverse'; -import * as util from 'util'; const PLACEHOLDER = 'TRIGGER_CHARACTER'; @@ -180,10 +179,7 @@ export class Visitor { sourceType: 'module', }); } catch (error) { - console.error(`parseAST error: ${util.inspect((error as any).message)}`); - console.error( - `parseAST error textFromEditor: ${util.inspect(textFromEditor)}` - ); + /* Silent fail. When a user hasn't finished typing it causes parsing JS errors */ } traverse(ast, { diff --git a/src/logging.ts b/src/logging.ts index f2204edf2..43183c043 100644 --- a/src/logging.ts +++ b/src/logging.ts @@ -4,7 +4,7 @@ import util from 'util'; class Logger implements ILogger { static channel: vscode.OutputChannel = - vscode.window.createOutputChannel('mongodb'); + vscode.window.createOutputChannel('MongoDB Extension'); private name: string; diff --git a/src/mdbExtensionController.ts b/src/mdbExtensionController.ts index 246a70add..591397322 100644 --- a/src/mdbExtensionController.ts +++ b/src/mdbExtensionController.ts @@ -10,7 +10,6 @@ import PlaygroundSelectedCodeActionProvider from './editors/playgroundSelectedCo import PlaygroundDiagnosticsCodeActionProvider from './editors/playgroundDiagnosticsCodeActionProvider'; import ConnectionController from './connectionController'; import ConnectionTreeItem from './explorer/connectionTreeItem'; -import { createLogger } from './logging'; import DatabaseTreeItem from './explorer/databaseTreeItem'; import DocumentListTreeItem from './explorer/documentListTreeItem'; import { DocumentSource } from './documentSource'; @@ -40,8 +39,6 @@ import PlaygroundResultProvider from './editors/playgroundResultProvider'; import WebviewController from './views/webviewController'; import { createIdFactory, generateId } from './utils/objectIdHelper'; -const log = createLogger('commands'); - // This class is the top-level controller for our extension. // Commands which the extensions handles are defined in the function `activate`. export default class MDBExtensionController implements vscode.Disposable { @@ -150,8 +147,6 @@ export default class MDBExtensionController implements vscode.Disposable { } registerCommands = (): void => { - log.info('Registering commands...'); - // Register our extension's commands. These are the event handlers and // control the functionality of our extension. // ------ CONNECTION ------ // @@ -252,8 +247,6 @@ export default class MDBExtensionController implements vscode.Disposable { this.registerEditorCommands(); this.registerTreeViewCommands(); - - log.info('Commands registered'); }; registerCommand = ( diff --git a/src/telemetry/telemetryService.ts b/src/telemetry/telemetryService.ts index e774d6308..42c5fc297 100644 --- a/src/telemetry/telemetryService.ts +++ b/src/telemetry/telemetryService.ts @@ -151,8 +151,6 @@ export default class TelemetryService { if (!this._segmentKey) { return; } - - log.info('Activating segment analytics...'); this._segmentAnalytics = new SegmentAnalytics(this._segmentKey, { // Segment batches messages and flushes asynchronously to the server. // The flushAt is a number of messages to enqueue before flushing. @@ -166,7 +164,7 @@ export default class TelemetryService { const segmentProperties = this.getTelemetryUserIdentity(); this._segmentAnalytics.identify(segmentProperties); - log.info('Segment analytics activated with properties', segmentProperties); + log.info('Segment analytics activated', segmentProperties); } deactivate(): void { diff --git a/src/test/suite/connectionController.test.ts b/src/test/suite/connectionController.test.ts index 2ce61cac4..8f639b3cc 100644 --- a/src/test/suite/connectionController.test.ts +++ b/src/test/suite/connectionController.test.ts @@ -65,6 +65,7 @@ suite('Connection Controller Test Suite', function () { vscode.window, 'showInformationMessage' ); + sandbox.stub(testTelemetryService, 'trackNewConnection'); showErrorMessageStub = sandbox.stub(vscode.window, 'showErrorMessage'); }); diff --git a/src/test/suite/editors/collectionDocumentsCodeLensProvider.test.ts b/src/test/suite/editors/collectionDocumentsCodeLensProvider.test.ts index 046a07bbf..c464ecb38 100644 --- a/src/test/suite/editors/collectionDocumentsCodeLensProvider.test.ts +++ b/src/test/suite/editors/collectionDocumentsCodeLensProvider.test.ts @@ -5,7 +5,7 @@ import CollectionDocumentsCodeLensProvider from '../../../editors/collectionDocu import CollectionDocumentsOperationsStore from '../../../editors/collectionDocumentsOperationsStore'; import { mockVSCodeTextDocument } from '../stubs'; -suite('Collection Documents Provider Test Suite', () => { +suite('Collection CodeLens Provider Test Suite', () => { test('expected provideCodeLenses to return a code lens with positions at the end of the document', () => { const testQueryStore = new CollectionDocumentsOperationsStore(); const testCodeLensProvider = new CollectionDocumentsCodeLensProvider( diff --git a/src/test/suite/editors/collectionDocumentsProvider.test.ts b/src/test/suite/editors/collectionDocumentsProvider.test.ts index a853526b0..a44bd060b 100644 --- a/src/test/suite/editors/collectionDocumentsProvider.test.ts +++ b/src/test/suite/editors/collectionDocumentsProvider.test.ts @@ -68,6 +68,10 @@ suite('Collection Documents Provider Test Suite', () => { statusView: testStatusView, editDocumentCodeLensProvider: testCodeLensProvider, }); + sandbox.stub( + testConnectionController._telemetryService, + 'trackNewConnection' + ); }); afterEach(() => { diff --git a/src/test/suite/editors/playgroundController.test.ts b/src/test/suite/editors/playgroundController.test.ts index 68edf9333..326098298 100644 --- a/src/test/suite/editors/playgroundController.test.ts +++ b/src/test/suite/editors/playgroundController.test.ts @@ -100,6 +100,7 @@ suite('Playground Controller Test Suite', function () { playgroundSelectedCodeActionProvider: testCodeActionProvider, }); showErrorMessageStub = sandbox.stub(vscode.window, 'showErrorMessage'); + sandbox.stub(testTelemetryService, 'trackNewConnection'); }); afterEach(() => { @@ -130,7 +131,7 @@ suite('Playground Controller Test Suite', function () { ); sandbox.replace( testPlaygroundController._languageServerController, - 'connectToServiceProvider', + 'activeConnectionChanged', fakeConnectToServiceProvider ); sandbox.stub(vscode.window, 'showInformationMessage'); @@ -138,7 +139,7 @@ suite('Playground Controller Test Suite', function () { testPlaygroundController._connectionController.setActiveDataService( mockActiveDataService ); - await testPlaygroundController._connectToServiceProvider(); + await testPlaygroundController._activeConnectionChanged(); }); test('it should pass the active connection id to the language server for connecting', () => { @@ -305,7 +306,7 @@ suite('Playground Controller Test Suite', function () { ); showTextDocumentStub = sandbox.stub(vscode.window, 'showTextDocument'); - await testPlaygroundController._connectToServiceProvider(); + await testPlaygroundController._activeConnectionChanged(); }); test('keep a playground in focus after running it', async () => { @@ -333,44 +334,6 @@ suite('Playground Controller Test Suite', function () { }); }); - test('it shows an error message and restarts, and connects the language server when an error occurs in evaluate (out of memory can cause this)', async () => { - const mockConnectionDisposedError = new Error( - 'Pending response rejected since connection got disposed' - ); - (mockConnectionDisposedError).code = -32097; - sinon - .stub(languageServerControllerStub, 'evaluate') - .rejects(mockConnectionDisposedError); - - const stubStartLanguageServer = sinon - .stub(languageServerControllerStub, 'startLanguageServer') - .resolves(); - - const stubConnectToServiceProvider = sinon - .stub(testPlaygroundController, '_connectToServiceProvider') - .resolves(); - - try { - await testPlaygroundController._evaluate('console.log("test");'); - - // It should have thrown in the above evaluation. - expect(true).to.equal(false); - } catch (error) { - expect((error).message).to.equal( - 'Pending response rejected since connection got disposed' - ); - expect((error).code).to.equal(-32097); - } - - expect(showErrorMessageStub.calledOnce).to.equal(true); - expect(showErrorMessageStub.firstCall.args[0]).to.equal( - 'An error occurred when running the playground. This can occur when the playground runner runs out of memory.' - ); - - expect(stubStartLanguageServer.calledOnce).to.equal(true); - expect(stubConnectToServiceProvider.calledOnce).to.equal(true); - }); - test('playground controller loads the active editor on start', () => { sandbox.replaceGetter( vscode.window, diff --git a/src/test/suite/editors/playgroundSelectedCodeActionProvider.test.ts b/src/test/suite/editors/playgroundSelectedCodeActionProvider.test.ts index e3260b534..ea8efa192 100644 --- a/src/test/suite/editors/playgroundSelectedCodeActionProvider.test.ts +++ b/src/test/suite/editors/playgroundSelectedCodeActionProvider.test.ts @@ -37,6 +37,10 @@ suite('Playground Selected CodeAction Provider Test Suite', function () { new LanguageServerController(extensionContextStub) ); sandbox.stub(vscode.window, 'showInformationMessage'); + sandbox.stub( + mdbTestExtension.testExtensionController._telemetryService, + 'trackNewConnection' + ); await mdbTestExtension.testExtensionController._connectionController.addNewConnectionStringAndConnect( TEST_DATABASE_URI @@ -79,7 +83,7 @@ suite('Playground Selected CodeAction Provider Test Suite', function () { .update('confirmRunAll', false); await mdbTestExtension.testExtensionController._languageServerController.startLanguageServer(); - await mdbTestExtension.testExtensionController._playgroundController._connectToServiceProvider(); + await mdbTestExtension.testExtensionController._playgroundController._activeConnectionChanged(); const fakeIsPlayground = sandbox.fake.returns(true); sandbox.replace(testCodeActionProvider, 'isPlayground', fakeIsPlayground); @@ -479,6 +483,10 @@ suite('Playground Selected CodeAction Provider Test Suite', function () { beforeEach(() => { const fakeIsPlayground = sandbox.fake.returns(false); sandbox.replace(testCodeActionProvider, 'isPlayground', fakeIsPlayground); + sandbox.stub( + mdbTestExtension.testExtensionController._telemetryService, + 'trackNewConnection' + ); }); afterEach(() => { diff --git a/src/test/suite/explorer/explorerController.test.ts b/src/test/suite/explorer/explorerController.test.ts index 7fbaa99c8..8b8d936eb 100644 --- a/src/test/suite/explorer/explorerController.test.ts +++ b/src/test/suite/explorer/explorerController.test.ts @@ -30,6 +30,10 @@ suite('Explorer Controller Test Suite', function () { ); sandbox.stub(vscode.window, 'showInformationMessage'); sandbox.stub(vscode.window, 'showErrorMessage'); + sandbox.stub( + mdbTestExtension.testExtensionController._telemetryService, + 'trackNewConnection' + ); }); afterEach(async () => { diff --git a/src/test/suite/language/languageServerController.test.ts b/src/test/suite/language/languageServerController.test.ts index 64eb8da23..0e66629c1 100644 --- a/src/test/suite/language/languageServerController.test.ts +++ b/src/test/suite/language/languageServerController.test.ts @@ -77,7 +77,7 @@ suite('Language Server Controller Test Suite', () => { playgroundSelectedCodeActionProvider: testCodeActionProvider, }); await languageServerControllerStub.startLanguageServer(); - await testPlaygroundController._connectToServiceProvider(); + await testPlaygroundController._activeConnectionChanged(); }); beforeEach(() => { diff --git a/src/test/suite/language/mongoDBService.test.ts b/src/test/suite/language/mongoDBService.test.ts index 1b372bc96..dfa1c0bab 100644 --- a/src/test/suite/language/mongoDBService.test.ts +++ b/src/test/suite/language/mongoDBService.test.ts @@ -57,7 +57,7 @@ suite('MongoDBService Test Suite', () => { before(async () => { testMongoDBService._extensionPath = ''; - await testMongoDBService.connectToServiceProvider(params); + await testMongoDBService.activeConnectionChanged(params); }); test('catches error when evaluate is called and extension path is empty string', async () => { @@ -99,13 +99,13 @@ suite('MongoDBService Test Suite', () => { const testMongoDBService = new MongoDBService(connection); test('connect and disconnect from cli service provider', async () => { - await testMongoDBService.connectToServiceProvider(params); + await testMongoDBService.activeConnectionChanged(params); expect(testMongoDBService.connectionString).to.be.equal( 'mongodb://localhost:27018' ); - await testMongoDBService.disconnectFromServiceProvider(); + await testMongoDBService.activeConnectionChanged({ connectionId: null }); expect(testMongoDBService.connectionString).to.be.undefined; expect(testMongoDBService.connectionOptions).to.be.undefined; @@ -129,7 +129,7 @@ suite('MongoDBService Test Suite', () => { testMongoDBService._getSchemaFields = (): Promise => Promise.resolve([]); - await testMongoDBService.connectToServiceProvider(params); + await testMongoDBService.activeConnectionChanged(params); }); test('provide shell collection methods completion if global scope', async () => { @@ -1806,7 +1806,7 @@ suite('MongoDBService Test Suite', () => { before(async () => { testMongoDBService._extensionPath = mdbTestExtension.extensionContextStub.extensionPath; - await testMongoDBService.connectToServiceProvider(params); + await testMongoDBService.activeConnectionChanged(params); }); test('evaluate should sum numbers', async () => { diff --git a/src/test/suite/mdbExtensionController.test.ts b/src/test/suite/mdbExtensionController.test.ts index 71fc9e14a..1efe137d9 100644 --- a/src/test/suite/mdbExtensionController.test.ts +++ b/src/test/suite/mdbExtensionController.test.ts @@ -126,6 +126,10 @@ suite('MDBExtensionController Test Suite', function () { sandbox.stub(vscode.workspace, 'openTextDocument'); sandbox.stub(vscode.window, 'showTextDocument'); showErrorMessageStub = sandbox.stub(vscode.window, 'showErrorMessage'); + sandbox.stub( + mdbTestExtension.testExtensionController._telemetryService, + 'trackNewConnection' + ); }); afterEach(() => { @@ -181,6 +185,10 @@ suite('MDBExtensionController Test Suite', function () { ); showErrorMessageStub = sandbox.stub(vscode.window, 'showErrorMessage'); sandbox.stub(vscode.window, 'showTextDocument'); + sandbox.stub( + mdbTestExtension.testExtensionController._telemetryService, + 'trackNewConnection' + ); }); afterEach(() => { diff --git a/src/test/suite/playground.test.ts b/src/test/suite/playground.test.ts index e8b329e88..e4a484a43 100644 --- a/src/test/suite/playground.test.ts +++ b/src/test/suite/playground.test.ts @@ -2,6 +2,7 @@ import * as vscode from 'vscode'; import { afterEach, beforeEach } from 'mocha'; import chai from 'chai'; import sinon from 'sinon'; +import type { SinonStub } from 'sinon'; import chaiAsPromised from 'chai-as-promised'; import { mdbTestExtension } from './stubbableMdbExtension'; @@ -22,6 +23,7 @@ suite('Playground', function () { const _disposables: vscode.Disposable[] = []; const sandbox = sinon.createSandbox(); + let showErrorMessageStub: SinonStub; beforeEach(async () => { sandbox.replace( @@ -71,14 +73,18 @@ suite('Playground', function () { 'getMongoClientConnectionOptions', fakeGetMongoClientConnectionOptions ); + showErrorMessageStub = sandbox.stub(vscode.window, 'showErrorMessage'); - await mdbTestExtension.testExtensionController._playgroundController._connectToServiceProvider(); + await mdbTestExtension.testExtensionController._playgroundController._activeConnectionChanged(); await mdbTestExtension.testExtensionController._playgroundController._languageServerController.updateCurrentSessionFields( { namespace: 'mongodbVSCodePlaygroundDB.sales', schemaFields: ['_id', 'name', 'time'], } ); + await vscode.workspace + .getConfiguration('mdb') + .update('confirmRunAll', false); }); afterEach(async () => { @@ -87,7 +93,7 @@ suite('Playground', function () { sandbox.restore(); }); - test('show mongodb completion items before other js completion', async () => { + test('shows mongodb completion items before other js completion', async () => { await vscode.commands.executeCommand('mdb.createPlayground'); const editor = vscode.window.activeTextEditor; @@ -119,4 +125,36 @@ suite('Playground', function () { "use('mongodbVSCodePlaygroundDB'); db.sales.find({ name});" ); }); + + test('restores the language server when the out of memory error occurred', async function () { + this.timeout(20000); + await vscode.commands.executeCommand('mdb.createPlayground'); + + const editor = vscode.window.activeTextEditor; + if (!editor) { + throw new Error('Window active text editor is undefined'); + } + + const testDocumentUri = editor.document.uri; + const edit = new vscode.WorkspaceEdit(); + edit.replace( + testDocumentUri, + getFullRange(editor.document), + "use('test'); const mockDataArray = []; for(let i = 0; i < 50000; i++) { mockDataArray.push(Math.random() * 10000); } const docs = []; for(let i = 0; i < 10000000; i++) { docs.push({ mockData: [...mockDataArray], a: 'test 123', b: Math.ceil(Math.random() * 10000) }); }" + ); + await vscode.workspace.applyEdit(edit); + await vscode.commands.executeCommand('mdb.runPlayground'); + + const onDidChangeDiagnostics = () => + new Promise((resolve) => { + // The diagnostics are set again when the server restarts. + vscode.languages.onDidChangeDiagnostics(resolve); + }); + await onDidChangeDiagnostics(); + + expect(showErrorMessageStub.calledOnce).to.equal(true); + expect(showErrorMessageStub.firstCall.args[0]).to.equal( + 'An internal error has occurred. The playground services have been restored. This can occur when the playground runner runs out of memory.' + ); + }); }); diff --git a/src/test/suite/stubs.ts b/src/test/suite/stubs.ts index b6cf7a890..1c31f5d07 100644 --- a/src/test/suite/stubs.ts +++ b/src/test/suite/stubs.ts @@ -247,6 +247,7 @@ class LanguageServerControllerStub { _source?: CancellationTokenSource; _isExecutingInProgress: boolean; _client: LanguageClient; + _currentConnectionId: string | null = null; constructor( context: ExtensionContextStub, @@ -339,7 +340,7 @@ class LanguageServerControllerStub { return Promise.resolve({ databaseName: null, collectionName: null }); } - connectToServiceProvider(/* params: { + activeConnectionChanged(/* params: { connectionString?: string; connectionOptions?: MongoClientOptions; extensionPath: string; @@ -347,10 +348,6 @@ class LanguageServerControllerStub { return Promise.resolve(); } - disconnectFromServiceProvider(): Promise { - return Promise.resolve(); - } - cancelAll(): void { return; } diff --git a/src/test/suite/telemetry/connectionTelemetry.test.ts b/src/test/suite/telemetry/connectionTelemetry.test.ts index 27c5af31d..95de92e2a 100644 --- a/src/test/suite/telemetry/connectionTelemetry.test.ts +++ b/src/test/suite/telemetry/connectionTelemetry.test.ts @@ -109,7 +109,9 @@ suite('ConnectionTelemetry Controller Test Suite', function () { }); }); - suite('with live connection', function () { + // TODO: Enable test back when Insider is fixed https://jira.mongodb.org/browse/VSCODE-452 + // MS GitHub Issue: https://github.com/microsoft/vscode/issues/188676 + suite.skip('with live connection', function () { this.timeout(20000); let dataServ; diff --git a/src/test/suite/views/webviewController.test.ts b/src/test/suite/views/webviewController.test.ts index 88f311239..3e1999a79 100644 --- a/src/test/suite/views/webviewController.test.ts +++ b/src/test/suite/views/webviewController.test.ts @@ -19,22 +19,31 @@ import WebviewController, { import * as linkHelper from '../../../utils/linkHelper'; suite('Webview Test Suite', () => { + const sandbox = sinon.createSandbox(); + + beforeEach(() => { + sandbox.stub( + mdbTestExtension.testExtensionController._telemetryService, + 'trackNewConnection' + ); + }); + afterEach(() => { - sinon.restore(); + sandbox.restore(); }); test('it creates a web view panel and sets the html content', () => { - const stubOnDidRecieveMessage = sinon.stub(); + const stubOnDidRecieveMessage = sandbox.stub(); const fakeWebview = { html: '', onDidReceiveMessage: stubOnDidRecieveMessage, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel @@ -131,6 +140,9 @@ suite('Webview Test Suite', () => { }); let messageReceivedSet = false; let messageReceived; + + sandbox.stub(testTelemetryService, 'trackNewConnection'); + const fakeWebview = { html: '', postMessage: async (): Promise => { @@ -147,14 +159,14 @@ suite('Webview Test Suite', () => { messageReceived = callback; messageReceivedSet = true; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel @@ -200,6 +212,9 @@ suite('Webview Test Suite', () => { }); let messageReceivedSet = false; let messageReceived; + + sandbox.stub(testTelemetryService, 'trackNewConnection'); + const fakeWebview = { html: '', postMessage: async (message): Promise => { @@ -217,13 +232,13 @@ suite('Webview Test Suite', () => { messageReceived = callback; messageReceivedSet = true; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel @@ -268,6 +283,9 @@ suite('Webview Test Suite', () => { telemetryService: testTelemetryService, }); let messageReceived; + + sandbox.stub(testTelemetryService, 'trackNewConnection'); + const fakeWebview = { html: '', postMessage: async (message): Promise => { @@ -280,13 +298,13 @@ suite('Webview Test Suite', () => { onDidReceiveMessage: (callback): void => { messageReceived = callback; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel @@ -327,6 +345,9 @@ suite('Webview Test Suite', () => { telemetryService: testTelemetryService, }); let messageReceived; + + sandbox.stub(testTelemetryService, 'trackNewConnection'); + const fakeWebview = { html: '', postMessage: (message): void => { @@ -343,12 +364,12 @@ suite('Webview Test Suite', () => { onDidReceiveMessage: (callback): void => { messageReceived = callback; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel @@ -391,11 +412,13 @@ suite('Webview Test Suite', () => { storageController: testStorageController, telemetryService: testTelemetryService, }); - const fakeVSCodeOpenDialog = sinon.fake.resolves({ + const fakeVSCodeOpenDialog = sandbox.fake.resolves({ path: '/somefilepath/test.text', }); - let messageReceived; + + sandbox.stub(testTelemetryService, 'trackNewConnection'); + const fakeWebview = { html: '', postMessage: async (): Promise => { @@ -408,19 +431,19 @@ suite('Webview Test Suite', () => { onDidReceiveMessage: (callback): void => { messageReceived = callback; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel ); - sinon.replace(vscode.window, 'showOpenDialog', fakeVSCodeOpenDialog); + sandbox.replace(vscode.window, 'showOpenDialog', fakeVSCodeOpenDialog); const testWebviewController = new WebviewController({ connectionController: testConnectionController, @@ -452,6 +475,9 @@ suite('Webview Test Suite', () => { telemetryService: testTelemetryService, }); let messageReceived; + + sandbox.stub(testTelemetryService, 'trackNewConnection'); + const fakeWebview = { html: '', postMessage: async (message): Promise => { @@ -469,25 +495,25 @@ suite('Webview Test Suite', () => { onDidReceiveMessage: (callback): void => { messageReceived = callback; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel ); - const fakeVSCodeOpenDialog = sinon.fake.resolves([ + const fakeVSCodeOpenDialog = sandbox.fake.resolves([ { fsPath: '/somefilepath/test.text', }, ]); - sinon.replace(vscode.window, 'showOpenDialog', fakeVSCodeOpenDialog); + sandbox.replace(vscode.window, 'showOpenDialog', fakeVSCodeOpenDialog); const testWebviewController = new WebviewController({ connectionController: testConnectionController, @@ -518,22 +544,29 @@ suite('Webview Test Suite', () => { telemetryService: testTelemetryService, }); let messageReceived; + + sandbox.stub(testTelemetryService, 'trackNewConnection'); + const fakeWebview = { html: '', onDidReceiveMessage: (callback): void => { messageReceived = callback; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeExecuteCommand = sinon.fake.resolves(false); + const fakeVSCodeExecuteCommand = sandbox.fake.resolves(false); - sinon.replace(vscode.commands, 'executeCommand', fakeVSCodeExecuteCommand); + sandbox.replace( + vscode.commands, + 'executeCommand', + fakeVSCodeExecuteCommand + ); - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel @@ -576,6 +609,9 @@ suite('Webview Test Suite', () => { telemetryService: testTelemetryService, }); let messageReceived; + + sandbox.stub(testTelemetryService, 'trackNewConnection'); + const fakeWebview = { html: '', postMessage: (message): void => { @@ -588,13 +624,13 @@ suite('Webview Test Suite', () => { onDidReceiveMessage: (callback): void => { messageReceived = callback; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel @@ -629,6 +665,9 @@ suite('Webview Test Suite', () => { telemetryService: testTelemetryService, }); let messageReceived; + + sandbox.stub(testTelemetryService, 'trackNewConnection'); + const fakeWebview = { html: '', postMessage: async (message): Promise => { @@ -642,13 +681,13 @@ suite('Webview Test Suite', () => { onDidReceiveMessage: (callback): void => { messageReceived = callback; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel @@ -687,27 +726,31 @@ suite('Webview Test Suite', () => { telemetryService: testTelemetryService, }); let messageReceived; + + sandbox.stub(testTelemetryService, 'trackNewConnection'); + const fakeWebview = { html: '', postMessage: (): void => {}, onDidReceiveMessage: (callback): void => { messageReceived = callback; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel ); - const mockRenameConnectionOnConnectionController = sinon.fake.returns(null); + const mockRenameConnectionOnConnectionController = + sandbox.fake.returns(null); - sinon.replace( + sandbox.replace( testConnectionController, 'renameConnection', mockRenameConnectionOnConnectionController @@ -768,13 +811,13 @@ suite('Webview Test Suite', () => { onDidReceiveMessage: (callback): void => { messageReceived = callback; }, - asWebviewUri: sinon.fake.returns(''), + asWebviewUri: sandbox.fake.returns(''), }; - const fakeVSCodeCreateWebviewPanel = sinon.fake.returns({ + const fakeVSCodeCreateWebviewPanel = sandbox.fake.returns({ webview: fakeWebview, }); - sinon.replace( + sandbox.replace( vscode.window, 'createWebviewPanel', fakeVSCodeCreateWebviewPanel @@ -787,11 +830,12 @@ suite('Webview Test Suite', () => { }); testWebviewController.openWebview(mdbTestExtension.extensionContextStub); + sandbox.stub(testTelemetryService, 'trackNewConnection'); }); test('it should handle opening trusted links', () => { - const stubOpenLink = sinon.fake.resolves(null); - sinon.replace(linkHelper, 'openLink', stubOpenLink); + const stubOpenLink = sandbox.fake.resolves(null); + sandbox.replace(linkHelper, 'openLink', stubOpenLink); messageReceived({ command: MESSAGE_TYPES.OPEN_TRUSTED_LINK,