mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-28 09:35:38 -05:00
* first set of changes to experiment the registration of cms related apis * Adding cms service entry to workbench * Adding basic functionality for add remove reg servers and group * Returning relative path as part of RegServerResult as string * initial extension * cleaned building with connecting to server * get list of registered servers * progress with registered servers tree * cms base node with server selection * removed unused services * replaced azure stuff with cms * removed cmsResourceService * list servers progress * Removing the cms apis from core. Having mssql extension expose them for cms extension * create server working fine * initial expansion and nodes * Propogating the backend name changes to apis * initial cms extension working * cached connection needs change in api * connect without dashboard in proposed * Fixing some missing sqlops references * add registered server bug found * added refresh context menu option * added payload * server description not disabled after reject connection * added more context actions and action icons * added empty resource and error when same name server is added * fixed connection issues with cms and normal connections * added initial tests * added cms icons * removed azure readme * test script revert * fix build tests * added more cms tests * fixed test script * fixed silent error when expanding servers * added more cms tests * removed cmsdialog from api * cms dialog without object * fixed theming issues * initial connection dialog done * can make connections * PM asks for strings and icons * removed search * removed unused code and fixed 1 test * fix connection management tests * changed icons * format file * fixed hygiene * initial cr comments * refactored cms connection dialog * fixed bug when switching dialogs * localized connection provider options * fixed cms provider name * code review comments * localized options in cms and mssql * localized more options
305 lines
12 KiB
TypeScript
305 lines
12 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 vscode from 'vscode';
|
|
import * as azdata from 'azdata';
|
|
import * as path from 'path';
|
|
import * as os from 'os';
|
|
import * as nls from 'vscode-nls';
|
|
const localize = nls.loadMessageBundle();
|
|
|
|
import { SqlOpsDataClient, ClientOptions } from 'dataprotocol-client';
|
|
import { IConfig, ServerProvider, Events } from 'service-downloader';
|
|
import { ServerOptions, TransportKind } from 'vscode-languageclient';
|
|
|
|
import * as Constants from './constants';
|
|
import ContextProvider from './contextProvider';
|
|
import { CredentialStore } from './credentialstore/credentialstore';
|
|
import { AzureResourceProvider } from './resourceProvider/resourceProvider';
|
|
import * as Utils from './utils';
|
|
import { Telemetry, LanguageClientErrorHandler } from './telemetry';
|
|
import { TelemetryFeature, AgentServicesFeature, DacFxServicesFeature, SchemaCompareServicesFeature } from './features';
|
|
import { AppContext } from './appContext';
|
|
import { ApiWrapper } from './apiWrapper';
|
|
import { UploadFilesCommand, MkDirCommand, SaveFileCommand, PreviewFileCommand, CopyPathCommand, DeleteFilesCommand } from './objectExplorerNodeProvider/hdfsCommands';
|
|
import { IPrompter } from './prompts/question';
|
|
import CodeAdapter from './prompts/adapter';
|
|
import { MssqlExtensionApi, MssqlObjectExplorerBrowser } from './api/mssqlapis';
|
|
import { OpenSparkJobSubmissionDialogCommand, OpenSparkJobSubmissionDialogFromFileCommand, OpenSparkJobSubmissionDialogTask } from './sparkFeature/dialog/dialogCommands';
|
|
import { OpenSparkYarnHistoryTask } from './sparkFeature/historyTask';
|
|
import { MssqlObjectExplorerNodeProvider, mssqlOutputChannel } from './objectExplorerNodeProvider/objectExplorerNodeProvider';
|
|
import { CmsService } from './cms/cmsService';
|
|
import { registerSearchServerCommand } from './objectExplorerNodeProvider/command';
|
|
|
|
const baseConfig = require('./config.json');
|
|
const outputChannel = vscode.window.createOutputChannel(Constants.serviceName);
|
|
const statusView = vscode.window.createStatusBarItem(vscode.StatusBarAlignment.Left);
|
|
const msgSampleCodeDataFrame = localize('msgSampleCodeDataFrame', 'This sample code loads the file into a data frame and shows the first 10 results.');
|
|
|
|
|
|
export async function activate(context: vscode.ExtensionContext): Promise<MssqlExtensionApi> {
|
|
// lets make sure we support this platform first
|
|
let supported = await Utils.verifyPlatform();
|
|
|
|
if (!supported) {
|
|
vscode.window.showErrorMessage('Unsupported platform');
|
|
return undefined;
|
|
}
|
|
|
|
let config: IConfig = JSON.parse(JSON.stringify(baseConfig));
|
|
config.installDirectory = path.join(__dirname, config.installDirectory);
|
|
config.proxy = vscode.workspace.getConfiguration('http').get('proxy');
|
|
config.strictSSL = vscode.workspace.getConfiguration('http').get('proxyStrictSSL') || true;
|
|
|
|
const credentialsStore = new CredentialStore(config);
|
|
const resourceProvider = new AzureResourceProvider(config);
|
|
let languageClient: SqlOpsDataClient;
|
|
let cmsService: CmsService;
|
|
|
|
const serverdownloader = new ServerProvider(config);
|
|
|
|
serverdownloader.eventEmitter.onAny(generateHandleServerProviderEvent());
|
|
|
|
let clientOptions: ClientOptions = {
|
|
documentSelector: ['sql'],
|
|
synchronize: {
|
|
configurationSection: Constants.extensionConfigSectionName
|
|
},
|
|
providerId: Constants.providerId,
|
|
errorHandler: new LanguageClientErrorHandler(),
|
|
features: [
|
|
// we only want to add new features
|
|
...SqlOpsDataClient.defaultFeatures,
|
|
TelemetryFeature,
|
|
AgentServicesFeature,
|
|
DacFxServicesFeature,
|
|
SchemaCompareServicesFeature
|
|
],
|
|
outputChannel: new CustomOutputChannel()
|
|
};
|
|
|
|
let prompter: IPrompter = new CodeAdapter();
|
|
let appContext = new AppContext(context, new ApiWrapper());
|
|
|
|
const installationStart = Date.now();
|
|
let serverPromise = serverdownloader.getOrDownloadServer().then(e => {
|
|
const installationComplete = Date.now();
|
|
let serverOptions = generateServerOptions(e);
|
|
languageClient = new SqlOpsDataClient(Constants.serviceName, serverOptions, clientOptions);
|
|
const processStart = Date.now();
|
|
languageClient.onReady().then(() => {
|
|
const processEnd = Date.now();
|
|
statusView.text = 'Service Started';
|
|
setTimeout(() => {
|
|
statusView.hide();
|
|
}, 1500);
|
|
Telemetry.sendTelemetryEvent('startup/LanguageClientStarted', {
|
|
installationTime: String(installationComplete - installationStart),
|
|
processStartupTime: String(processEnd - processStart),
|
|
totalTime: String(processEnd - installationStart),
|
|
beginningTimestamp: String(installationStart)
|
|
});
|
|
});
|
|
statusView.show();
|
|
statusView.text = 'Starting service';
|
|
languageClient.start();
|
|
credentialsStore.start();
|
|
resourceProvider.start();
|
|
|
|
let nodeProvider = new MssqlObjectExplorerNodeProvider(prompter, appContext);
|
|
azdata.dataprotocol.registerObjectExplorerNodeProvider(nodeProvider);
|
|
|
|
cmsService = new CmsService(appContext, languageClient);
|
|
|
|
activateSparkFeatures(appContext);
|
|
activateNotebookTask(appContext);
|
|
}, e => {
|
|
Telemetry.sendTelemetryEvent('ServiceInitializingFailed');
|
|
vscode.window.showErrorMessage('Failed to start Sql tools service');
|
|
});
|
|
registerSearchServerCommand(appContext);
|
|
let contextProvider = new ContextProvider();
|
|
context.subscriptions.push(contextProvider);
|
|
context.subscriptions.push(credentialsStore);
|
|
context.subscriptions.push(resourceProvider);
|
|
context.subscriptions.push(new UploadFilesCommand(prompter, appContext));
|
|
context.subscriptions.push(new MkDirCommand(prompter, appContext));
|
|
context.subscriptions.push(new SaveFileCommand(prompter, appContext));
|
|
context.subscriptions.push(new PreviewFileCommand(prompter, appContext));
|
|
context.subscriptions.push(new CopyPathCommand(appContext));
|
|
context.subscriptions.push(new DeleteFilesCommand(prompter, appContext));
|
|
context.subscriptions.push({ dispose: () => languageClient.stop() });
|
|
|
|
let api: MssqlExtensionApi = {
|
|
getMssqlObjectExplorerBrowser(): MssqlObjectExplorerBrowser {
|
|
return {
|
|
getNode: (context: azdata.ObjectExplorerContext) => {
|
|
let oeProvider = appContext.getService<MssqlObjectExplorerNodeProvider>(Constants.ObjectExplorerService);
|
|
return <any>oeProvider.findSqlClusterNodeByContext(context);
|
|
}
|
|
};
|
|
},
|
|
getCmsServiceProvider(): Promise<CmsService> {
|
|
return serverPromise.then(() => cmsService);
|
|
}
|
|
};
|
|
return api;
|
|
}
|
|
|
|
function activateSparkFeatures(appContext: AppContext): void {
|
|
let extensionContext = appContext.extensionContext;
|
|
let apiWrapper = appContext.apiWrapper;
|
|
let outputChannel: vscode.OutputChannel = mssqlOutputChannel;
|
|
extensionContext.subscriptions.push(new OpenSparkJobSubmissionDialogCommand(appContext, outputChannel));
|
|
extensionContext.subscriptions.push(new OpenSparkJobSubmissionDialogFromFileCommand(appContext, outputChannel));
|
|
apiWrapper.registerTaskHandler(Constants.mssqlClusterLivySubmitSparkJobTask, (profile: azdata.IConnectionProfile) => {
|
|
new OpenSparkJobSubmissionDialogTask(appContext, outputChannel).execute(profile);
|
|
});
|
|
apiWrapper.registerTaskHandler(Constants.mssqlClusterLivyOpenSparkHistory, (profile: azdata.IConnectionProfile) => {
|
|
new OpenSparkYarnHistoryTask(appContext).execute(profile, true);
|
|
});
|
|
apiWrapper.registerTaskHandler(Constants.mssqlClusterLivyOpenYarnHistory, (profile: azdata.IConnectionProfile) => {
|
|
new OpenSparkYarnHistoryTask(appContext).execute(profile, false);
|
|
});
|
|
}
|
|
|
|
function activateNotebookTask(appContext: AppContext): void {
|
|
let apiWrapper = appContext.apiWrapper;
|
|
apiWrapper.registerTaskHandler(Constants.mssqlClusterNewNotebookTask, (profile: azdata.IConnectionProfile) => {
|
|
return saveProfileAndCreateNotebook(profile);
|
|
});
|
|
apiWrapper.registerTaskHandler(Constants.mssqlClusterOpenNotebookTask, (profile: azdata.IConnectionProfile) => {
|
|
return handleOpenNotebookTask(profile);
|
|
});
|
|
}
|
|
|
|
function saveProfileAndCreateNotebook(profile: azdata.IConnectionProfile): Promise<void> {
|
|
return handleNewNotebookTask(undefined, profile);
|
|
}
|
|
|
|
function findNextUntitledEditorName(): string {
|
|
let nextVal = 0;
|
|
// Note: this will go forever if it's coded wrong, or you have inifinite Untitled notebooks!
|
|
while (true) {
|
|
let title = `Notebook-${nextVal}`;
|
|
let hasNotebookDoc = azdata.nb.notebookDocuments.findIndex(doc => doc.isUntitled && doc.fileName === title) > -1;
|
|
if (!hasNotebookDoc) {
|
|
return title;
|
|
}
|
|
nextVal++;
|
|
}
|
|
}
|
|
|
|
async function 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 title = findNextUntitledEditorName();
|
|
let untitledUri = vscode.Uri.parse(`untitled:${title}`);
|
|
let editor = await azdata.nb.showNotebookDocument(untitledUri, {
|
|
connectionProfile: profile,
|
|
preview: false
|
|
});
|
|
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 = '#' + msgSampleCodeDataFrame + os.EOL + 'df = (spark.read.option("inferSchema", "true")'
|
|
+ os.EOL + '.option("header", "true")' + os.EOL + '.csv("{0}"))' + os.EOL + 'df.show(10)';
|
|
editor.edit(editBuilder => {
|
|
editBuilder.replace(0, {
|
|
cell_type: 'code',
|
|
source: analyzeCommand.replace('{0}', hdfsPath)
|
|
});
|
|
});
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
async function handleOpenNotebookTask(profile: azdata.IConnectionProfile): Promise<void> {
|
|
let notebookFileTypeName = localize('notebookFileType', 'Notebooks');
|
|
let filter = {};
|
|
filter[notebookFileTypeName] = 'ipynb';
|
|
let uris = await vscode.window.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
|
|
vscode.window.showErrorMessage(localize('unsupportedFileType', 'Only .ipynb Notebooks are supported'));
|
|
} else {
|
|
await azdata.nb.showNotebookDocument(fileUri, {
|
|
connectionProfile: profile,
|
|
preview: false
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
function generateServerOptions(executablePath: string): ServerOptions {
|
|
let launchArgs = Utils.getCommonLaunchArgsAndCleanupOldLogFiles('sqltools', executablePath);
|
|
return { command: executablePath, args: launchArgs, transport: TransportKind.stdio };
|
|
}
|
|
|
|
function generateHandleServerProviderEvent() {
|
|
let dots = 0;
|
|
return (e: string, ...args: any[]) => {
|
|
outputChannel.show();
|
|
statusView.show();
|
|
switch (e) {
|
|
case Events.INSTALL_START:
|
|
outputChannel.appendLine(`Installing ${Constants.serviceName} service to ${args[0]}`);
|
|
statusView.text = 'Installing Service';
|
|
break;
|
|
case Events.INSTALL_END:
|
|
outputChannel.appendLine('Installed');
|
|
break;
|
|
case Events.DOWNLOAD_START:
|
|
outputChannel.appendLine(`Downloading ${args[0]}`);
|
|
outputChannel.append(`(${Math.ceil(args[1] / 1024)} KB)`);
|
|
statusView.text = 'Downloading Service';
|
|
break;
|
|
case Events.DOWNLOAD_PROGRESS:
|
|
let newDots = Math.ceil(args[0] / 5);
|
|
if (newDots > dots) {
|
|
outputChannel.append('.'.repeat(newDots - dots));
|
|
dots = newDots;
|
|
}
|
|
break;
|
|
case Events.DOWNLOAD_END:
|
|
outputChannel.appendLine('Done!');
|
|
break;
|
|
}
|
|
};
|
|
}
|
|
|
|
// this method is called when your extension is deactivated
|
|
export function deactivate(): void {
|
|
}
|
|
|
|
class CustomOutputChannel implements vscode.OutputChannel {
|
|
name: string;
|
|
append(value: string): void {
|
|
console.log(value);
|
|
}
|
|
appendLine(value: string): void {
|
|
console.log(value);
|
|
}
|
|
clear(): void {
|
|
}
|
|
show(preserveFocus?: boolean): void;
|
|
show(column?: vscode.ViewColumn, preserveFocus?: boolean): void;
|
|
show(column?: any, preserveFocus?: any) {
|
|
}
|
|
hide(): void {
|
|
}
|
|
dispose(): void {
|
|
}
|
|
}
|