mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-03-17 03:21:40 -04:00
231 lines
9.1 KiB
TypeScript
231 lines
9.1 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import * as path from 'path';
|
|
import * as azdata from 'azdata';
|
|
import * as vscode from 'vscode';
|
|
import * as os from 'os';
|
|
import * as nls from 'vscode-nls';
|
|
const localize = nls.loadMessageBundle();
|
|
|
|
import * as constants from '../common/constants';
|
|
import * as localizedConstants from '../common/localizedConstants';
|
|
import JupyterServerInstallation from './jupyterServerInstallation';
|
|
import { IServerInstance } from './common';
|
|
import * as utils from '../common/utils';
|
|
import { IPrompter, QuestionTypes, IQuestion } from '../prompts/question';
|
|
|
|
import { AppContext } from '../common/appContext';
|
|
import { ApiWrapper } from '../common/apiWrapper';
|
|
import { LocalJupyterServerManager } from './jupyterServerManager';
|
|
import { NotebookCompletionItemProvider } from '../intellisense/completionItemProvider';
|
|
import { JupyterNotebookProvider } from './jupyterNotebookProvider';
|
|
import { ConfigurePythonDialog } from '../dialog/configurePythonDialog';
|
|
import CodeAdapter from '../prompts/adapter';
|
|
import { ManagePackagesDialog } from '../dialog/managePackages/managePackagesDialog';
|
|
|
|
let untitledCounter = 0;
|
|
|
|
export class JupyterController implements vscode.Disposable {
|
|
private _jupyterInstallation: JupyterServerInstallation;
|
|
private _notebookInstances: IServerInstance[] = [];
|
|
|
|
private outputChannel: vscode.OutputChannel;
|
|
private prompter: IPrompter;
|
|
|
|
constructor(private appContext: AppContext) {
|
|
this.prompter = new CodeAdapter();
|
|
this.outputChannel = this.appContext.apiWrapper.createOutputChannel(constants.extensionOutputChannel);
|
|
}
|
|
|
|
private get apiWrapper(): ApiWrapper {
|
|
return this.appContext.apiWrapper;
|
|
}
|
|
|
|
public get extensionContext(): vscode.ExtensionContext {
|
|
return this.appContext && this.appContext.extensionContext;
|
|
}
|
|
|
|
public dispose(): void {
|
|
this.deactivate();
|
|
}
|
|
|
|
// PUBLIC METHODS //////////////////////////////////////////////////////
|
|
public async activate(): Promise<boolean> {
|
|
this._jupyterInstallation = new JupyterServerInstallation(
|
|
this.extensionContext.extensionPath,
|
|
this.outputChannel,
|
|
this.apiWrapper);
|
|
|
|
// Add command/task handlers
|
|
this.apiWrapper.registerTaskHandler(constants.jupyterOpenNotebookTask, (profile: azdata.IConnectionProfile) => {
|
|
return this.handleOpenNotebookTask(profile);
|
|
});
|
|
this.apiWrapper.registerTaskHandler(constants.jupyterNewNotebookTask, (profile: azdata.IConnectionProfile) => {
|
|
return this.saveProfileAndCreateNotebook(profile);
|
|
});
|
|
this.apiWrapper.registerCommand(constants.jupyterNewNotebookCommand, (explorerContext: azdata.ObjectExplorerContext) => {
|
|
return this.saveProfileAndCreateNotebook(explorerContext ? explorerContext.connectionProfile : undefined);
|
|
});
|
|
this.apiWrapper.registerCommand(constants.jupyterAnalyzeCommand, (explorerContext: azdata.ObjectExplorerContext) => {
|
|
return this.saveProfileAndAnalyzeNotebook(explorerContext);
|
|
});
|
|
|
|
this.apiWrapper.registerCommand(constants.jupyterReinstallDependenciesCommand, () => { return this.handleDependenciesReinstallation(); });
|
|
this.apiWrapper.registerCommand(constants.jupyterManagePackages, () => { return this.doManagePackages(); });
|
|
this.apiWrapper.registerCommand(constants.jupyterConfigurePython, () => { return this.doConfigurePython(this._jupyterInstallation); });
|
|
|
|
let supportedFileFilter: vscode.DocumentFilter[] = [
|
|
{ scheme: 'untitled', language: '*' }
|
|
];
|
|
let notebookProvider = this.registerNotebookProvider();
|
|
this.extensionContext.subscriptions.push(this.apiWrapper.registerCompletionItemProvider(supportedFileFilter, new NotebookCompletionItemProvider(notebookProvider)));
|
|
|
|
return true;
|
|
}
|
|
|
|
private registerNotebookProvider(): JupyterNotebookProvider {
|
|
let notebookProvider = new JupyterNotebookProvider((documentUri: vscode.Uri) => new LocalJupyterServerManager({
|
|
documentPath: documentUri.fsPath,
|
|
jupyterInstallation: this._jupyterInstallation,
|
|
extensionContext: this.extensionContext,
|
|
apiWrapper: this.apiWrapper
|
|
}));
|
|
azdata.nb.registerNotebookProvider(notebookProvider);
|
|
return notebookProvider;
|
|
}
|
|
|
|
private saveProfileAndCreateNotebook(profile: azdata.IConnectionProfile): Promise<void> {
|
|
return this.handleNewNotebookTask(undefined, profile);
|
|
}
|
|
|
|
private saveProfileAndAnalyzeNotebook(oeContext: azdata.ObjectExplorerContext): Promise<void> {
|
|
return this.handleNewNotebookTask(oeContext, oeContext.connectionProfile);
|
|
}
|
|
|
|
public deactivate(): void {
|
|
// Shutdown any open notebooks
|
|
this._notebookInstances.forEach(instance => { instance.stop(); });
|
|
}
|
|
|
|
// EVENT HANDLERS //////////////////////////////////////////////////////
|
|
public async getDefaultConnection(): Promise<azdata.connection.ConnectionProfile> {
|
|
return await this.apiWrapper.getCurrentConnection();
|
|
}
|
|
|
|
private async handleOpenNotebookTask(profile: azdata.IConnectionProfile): Promise<void> {
|
|
let notebookFileTypeName = localize('notebookFileType', 'Notebooks');
|
|
let filter: { [key: string]: Array<string> } = {};
|
|
filter[notebookFileTypeName] = ['ipynb'];
|
|
let uris = await this.apiWrapper.showOpenDialog({
|
|
filters: filter,
|
|
canSelectFiles: true,
|
|
canSelectMany: false
|
|
});
|
|
if (uris && uris.length > 0) {
|
|
let fileUri = uris[0];
|
|
// Verify this is a .ipynb file since this isn't actually filtered on Mac/Linux
|
|
if (path.extname(fileUri.fsPath) !== '.ipynb') {
|
|
// in the future might want additional supported types
|
|
this.apiWrapper.showErrorMessage(localize('unsupportedFileType', 'Only .ipynb Notebooks are supported'));
|
|
} else {
|
|
await azdata.nb.showNotebookDocument(fileUri, {
|
|
connectionProfile: profile,
|
|
providerId: constants.jupyterNotebookProviderId,
|
|
preview: false
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
private async handleNewNotebookTask(oeContext?: azdata.ObjectExplorerContext, profile?: azdata.IConnectionProfile): Promise<void> {
|
|
// Ensure we get a unique ID for the notebook. For now we're using a different prefix to the built-in untitled files
|
|
// to handle this. We should look into improving this in the future
|
|
let untitledUri = vscode.Uri.parse(`untitled:Notebook-${untitledCounter++}`);
|
|
let editor = await azdata.nb.showNotebookDocument(untitledUri, {
|
|
connectionProfile: profile,
|
|
providerId: constants.jupyterNotebookProviderId,
|
|
preview: false,
|
|
defaultKernel: {
|
|
name: 'pyspark3kernel',
|
|
display_name: 'PySpark3',
|
|
language: 'python'
|
|
}
|
|
});
|
|
if (oeContext && oeContext.nodeInfo && oeContext.nodeInfo.nodePath) {
|
|
// Get the file path after '/HDFS'
|
|
let hdfsPath: string = oeContext.nodeInfo.nodePath.substring(oeContext.nodeInfo.nodePath.indexOf('/HDFS') + '/HDFS'.length);
|
|
if (hdfsPath.length > 0) {
|
|
let analyzeCommand = '#' + localizedConstants.msgSampleCodeDataFrame + os.EOL + 'df = (spark.read.option(\"inferSchema\", \"true\")'
|
|
+ os.EOL + '.option(\"header\", \"true\")' + os.EOL + '.csv(\'{0}\'))' + os.EOL + 'df.show(10)';
|
|
// TODO re-enable insert into document once APIs are finalized.
|
|
// editor.document.cells[0].source = [analyzeCommand.replace('{0}', hdfsPath)];
|
|
editor.edit(editBuilder => {
|
|
editBuilder.replace(0, {
|
|
cell_type: 'code',
|
|
source: analyzeCommand.replace('{0}', hdfsPath)
|
|
});
|
|
});
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
private async handleDependenciesReinstallation(): Promise<void> {
|
|
try {
|
|
let doReinstall = await this.confirmReinstall();
|
|
if (doReinstall) {
|
|
await this._jupyterInstallation.startInstallProcess(true);
|
|
}
|
|
} catch (err) {
|
|
this.apiWrapper.showErrorMessage(utils.getErrorMessage(err));
|
|
}
|
|
}
|
|
|
|
//Confirmation message dialog
|
|
private async confirmReinstall(): Promise<boolean> {
|
|
return await this.prompter.promptSingle<boolean>(<IQuestion>{
|
|
type: QuestionTypes.confirm,
|
|
message: localize('confirmReinstall', 'Are you sure you want to reinstall?'),
|
|
default: true
|
|
});
|
|
}
|
|
|
|
public doManagePackages(): void {
|
|
try {
|
|
let packagesDialog = new ManagePackagesDialog(this._jupyterInstallation);
|
|
packagesDialog.showDialog();
|
|
} catch (error) {
|
|
let message = utils.getErrorMessage(error);
|
|
this.apiWrapper.showErrorMessage(message);
|
|
}
|
|
}
|
|
|
|
public doConfigurePython(jupyterInstaller: JupyterServerInstallation): void {
|
|
let pythonDialog = new ConfigurePythonDialog(this.apiWrapper, jupyterInstaller);
|
|
pythonDialog.showDialog().catch((err: any) => {
|
|
this.apiWrapper.showErrorMessage(utils.getErrorMessage(err));
|
|
});
|
|
}
|
|
|
|
public getTextToSendToTerminal(shellType: any): string {
|
|
if (utils.getOSPlatform() === utils.Platform.Windows && typeof shellType === 'string') {
|
|
if (shellType.endsWith('powershell.exe')) {
|
|
return localizedConstants.msgManagePackagesPowershell;
|
|
} else if (shellType.endsWith('cmd.exe')) {
|
|
return localizedConstants.msgManagePackagesCmd;
|
|
} else {
|
|
return localizedConstants.msgManagePackagesBash;
|
|
}
|
|
} else {
|
|
return localizedConstants.msgManagePackagesBash;
|
|
}
|
|
}
|
|
|
|
public get jupyterInstallation() {
|
|
return this._jupyterInstallation;
|
|
}
|
|
}
|