/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the Source EULA. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ import * as constants from '../common/constants'; import * as mssql from 'mssql'; import * as os from 'os'; import * as path from 'path'; import * as utils from '../common/utils'; import * as UUID from 'vscode-languageclient/lib/utils/uuid'; import * as templates from '../templates/templates'; import * as vscode from 'vscode'; import type * as azdataType from 'azdata'; import * as dataworkspace from 'dataworkspace'; import * as mssqlVscode from 'vscode-mssql'; import { promises as fs } from 'fs'; import { PublishDatabaseDialog } from '../dialogs/publishDatabaseDialog'; import { Project } from '../models/project'; import { SqlDatabaseProjectTreeViewProvider } from './databaseProjectTreeViewProvider'; import { FolderNode, FileNode } from '../models/tree/fileFolderTreeItem'; import { BaseProjectTreeItem } from '../models/tree/baseTreeItem'; import { ImportDataModel } from '../models/api/import'; import { NetCoreTool, DotNetError } from '../tools/netcoreTool'; import { ShellCommandOptions } from '../tools/shellExecutionHelper'; import { BuildHelper } from '../tools/buildHelper'; import { readPublishProfile, promptForSavingProfile, savePublishProfile } from '../models/publishProfile/publishProfile'; import { AddDatabaseReferenceDialog } from '../dialogs/addDatabaseReferenceDialog'; import { ISystemDatabaseReferenceSettings, IDacpacReferenceSettings, IProjectReferenceSettings, INugetPackageReferenceSettings } from '../models/IDatabaseReferenceSettings'; import { DatabaseReferenceTreeItem, SqlProjectReferenceTreeItem } from '../models/tree/databaseReferencesTreeItem'; import { CreateProjectFromDatabaseDialog } from '../dialogs/createProjectFromDatabaseDialog'; import { UpdateProjectFromDatabaseDialog } from '../dialogs/updateProjectFromDatabaseDialog'; import { TelemetryActions, TelemetryReporter, TelemetryViews } from '../common/telemetry'; import { IconPathHelper } from '../common/iconHelper'; import { DashboardData, PublishData, Status } from '../models/dashboardData/dashboardData'; import { getPublishDatabaseSettings, launchPublishTargetOption } from '../dialogs/publishDatabaseQuickpick'; import { launchCreateAzureServerQuickPick } from '../dialogs/deployDatabaseQuickpick'; import { DeployService } from '../models/deploy/deployService'; import { AddItemOptions, EntryType, GenerateProjectFromOpenApiSpecOptions, IDatabaseReferenceProjectEntry, ISqlProject, ItemType, SqlTargetPlatform } from 'sqldbproj'; import { AutorestHelper } from '../tools/autorestHelper'; import { createNewProjectFromDatabaseWithQuickpick } from '../dialogs/createProjectFromDatabaseQuickpick'; import { addDatabaseReferenceQuickpick } from '../dialogs/addDatabaseReferenceQuickpick'; import { ISqlDbDeployProfile } from '../models/deploy/deployProfile'; import { FileProjectEntry, SqlProjectReferenceProjectEntry } from '../models/projectEntry'; import { UpdateProjectAction, UpdateProjectDataModel } from '../models/api/updateProject'; import { AzureSqlClient } from '../models/deploy/azureSqlClient'; import { ConnectionService } from '../models/connections/connectionService'; import { getPublishToDockerSettings } from '../dialogs/publishToDockerQuickpick'; import { SqlCmdVariableTreeItem } from '../models/tree/sqlcmdVariableTreeItem'; import { IPublishToDockerSettings, ISqlProjectPublishSettings } from '../models/deploy/publishSettings'; const maxTableLength = 10; /** * This is a duplicate of the TaskExecutionMode from azdata.d.ts/vscode-mssql.d.ts, which is needed * for using when running in VS Code since we don't have an actual implementation of the enum at runtime * (unlike azdata which is injected by the extension host). Even specifying it as a const enum in the * typings file currently doesn't work as the TypeScript compiler doesn't currently inline const enum * values imported as "import type" https://github.com/microsoft/TypeScript/issues/40344 */ export enum TaskExecutionMode { execute = 0, script = 1, executeAndScript = 2 } export type AddDatabaseReferenceSettings = ISystemDatabaseReferenceSettings | IDacpacReferenceSettings | IProjectReferenceSettings | INugetPackageReferenceSettings; interface FileWatcherStatus { fileWatcher: vscode.FileSystemWatcher; containsCreateTableStatement: boolean; } /** * Controller for managing lifecycle of projects */ export class ProjectsController { private netCoreTool: NetCoreTool; private buildHelper: BuildHelper; private buildInfo: DashboardData[] = []; private publishInfo: PublishData[] = []; public deployService: DeployService; private connectionService: ConnectionService; private azureSqlClient: AzureSqlClient; private autorestHelper: AutorestHelper; private projFileWatchers = new Map(); private fileWatchers = new Map(); constructor(private _outputChannel: vscode.OutputChannel) { this.netCoreTool = new NetCoreTool(this._outputChannel); this.buildHelper = new BuildHelper(); this.azureSqlClient = new AzureSqlClient(); this.deployService = new DeployService(this.azureSqlClient, this._outputChannel); this.connectionService = new ConnectionService(this._outputChannel); this.autorestHelper = new AutorestHelper(this._outputChannel); } //#region Dashboard public getDashboardPublishData(projectFile: string): (string | dataworkspace.IconCellValue)[][] { const infoRows: (string | dataworkspace.IconCellValue)[][] = []; for (let i = this.publishInfo.length - 1; i >= 0; i--) { if (this.publishInfo[i].projectFile === projectFile) { let icon: azdataType.IconPath; let text: string; if (this.publishInfo[i].status === Status.success) { icon = IconPathHelper.success; text = constants.Success; } else if (this.publishInfo[i].status === Status.failed) { icon = IconPathHelper.error; text = constants.Failed; } else { icon = IconPathHelper.inProgress; text = constants.InProgress; } let infoRow: (string | dataworkspace.IconCellValue)[] = [{ text: text, icon: icon }, this.publishInfo[i].startDate, this.publishInfo[i].timeToCompleteAction, this.publishInfo[i].target, this.publishInfo[i].targetServer, this.publishInfo[i].targetDatabase]; infoRows.push(infoRow); } } return infoRows; } public getDashboardBuildData(projectFile: string): (string | dataworkspace.IconCellValue)[][] { const infoRows: (string | dataworkspace.IconCellValue)[][] = []; for (let i = this.buildInfo.length - 1; i >= 0; i--) { if (this.buildInfo[i].projectFile === projectFile) { let icon: azdataType.IconPath; let text: string; if (this.buildInfo[i].status === Status.success) { icon = IconPathHelper.success; text = constants.Success; } else if (this.buildInfo[i].status === Status.failed) { icon = IconPathHelper.error; text = constants.Failed; } else { icon = IconPathHelper.inProgress; text = constants.InProgress; } let infoRow: (string | dataworkspace.IconCellValue)[] = [{ text: text, icon: icon }, this.buildInfo[i].startDate, this.buildInfo[i].timeToCompleteAction, this.buildInfo[i].target]; infoRows.push(infoRow); } } return infoRows; } //#endregion //#region Create new project /** * Creates a new folder with the project name in the specified location, and places the new .sqlproj inside it * @param creationParams */ public async createNewProject(creationParams: NewProjectParams): Promise { TelemetryReporter.createActionEvent(TelemetryViews.ProjectController, TelemetryActions.createNewProject) .withAdditionalProperties({ template: creationParams.projectTypeId, sdkStyle: creationParams.sdkStyle!.toString(), targetPlatform: creationParams.targetPlatform?.toString() ?? '' }) .send(); if (creationParams.projectGuid && !UUID.isUUID(creationParams.projectGuid)) { throw new Error(constants.invalidGuid(creationParams.projectGuid)); } if (creationParams.targetPlatform && !constants.targetPlatformToVersion.get(creationParams.targetPlatform)) { throw new Error(constants.invalidTargetPlatform(creationParams.targetPlatform, Array.from(constants.targetPlatformToVersion.keys()))); } const targetPlatform = creationParams.targetPlatform ? constants.targetPlatformToVersion.get(creationParams.targetPlatform)! : constants.defaultDSP; let newProjFileName = creationParams.newProjName; if (!newProjFileName.toLowerCase().endsWith(constants.sqlprojExtension)) { newProjFileName += constants.sqlprojExtension; } const newProjFilePath = path.join(creationParams.folderUri.fsPath, path.parse(newProjFileName).name, newProjFileName); if (await utils.exists(newProjFilePath)) { throw new Error(constants.projectAlreadyExists(newProjFileName, path.parse(newProjFilePath).dir)); } const sqlProjectsService = await utils.getSqlProjectsService(); if (utils.getAzdataApi()) { const projectStyle = creationParams.sdkStyle ? mssql.ProjectType.SdkStyle : mssql.ProjectType.LegacyStyle; await (sqlProjectsService as mssql.ISqlProjectsService).createProject(newProjFilePath, projectStyle, targetPlatform); } else { const projectStyle = creationParams.sdkStyle ? mssqlVscode.ProjectType.SdkStyle : mssqlVscode.ProjectType.LegacyStyle; await (sqlProjectsService as mssqlVscode.ISqlProjectsService).createProject(newProjFilePath, projectStyle, targetPlatform); } await this.addTemplateFiles(newProjFilePath, creationParams.projectTypeId); return newProjFilePath; } /** * Adds the template files for the provided project type * @param newProjFilePath path to project to add template files to * @param projectTypeId project type id */ private async addTemplateFiles(newProjFilePath: string, projectTypeId: string): Promise { if (projectTypeId === constants.emptySqlDatabaseProjectTypeId || newProjFilePath === '') { return; } if (projectTypeId === constants.edgeSqlDatabaseProjectTypeId) { const project = await Project.openProject(newProjFilePath); await this.addFileToProjectFromTemplate(project, templates.get(ItemType.table), 'DataTable.sql', new Map([['OBJECT_NAME', 'DataTable']])); await this.addFileToProjectFromTemplate(project, templates.get(ItemType.dataSource), 'EdgeHubInputDataSource.sql', new Map([['OBJECT_NAME', 'EdgeHubInputDataSource'], ['LOCATION', 'edgehub://']])); await this.addFileToProjectFromTemplate(project, templates.get(ItemType.dataSource), 'SqlOutputDataSource.sql', new Map([['OBJECT_NAME', 'SqlOutputDataSource'], ['LOCATION', 'sqlserver://tcp:.,1433']])); await this.addFileToProjectFromTemplate(project, templates.get(ItemType.fileFormat), 'StreamFileFormat.sql', new Map([['OBJECT_NAME', 'StreamFileFormat']])); await this.addFileToProjectFromTemplate(project, templates.get(ItemType.externalStream), 'EdgeHubInputStream.sql', new Map([['OBJECT_NAME', 'EdgeHubInputStream'], ['DATA_SOURCE_NAME', 'EdgeHubInputDataSource'], ['LOCATION', 'input'], ['OPTIONS', ',\n\tFILE_FORMAT = StreamFileFormat']])); await this.addFileToProjectFromTemplate(project, templates.get(ItemType.externalStream), 'SqlOutputStream.sql', new Map([['OBJECT_NAME', 'SqlOutputStream'], ['DATA_SOURCE_NAME', 'SqlOutputDataSource'], ['LOCATION', 'TSQLStreaming.dbo.DataTable'], ['OPTIONS', '']])); await this.addFileToProjectFromTemplate(project, templates.get(ItemType.externalStreamingJob), 'EdgeStreamingJob.sql', new Map([['OBJECT_NAME', 'EdgeStreamingJob']])); } } private async addFileToProjectFromTemplate(project: ISqlProject, itemType: templates.ProjectScriptType, relativePath: string, expansionMacros: Map): Promise { const newFileText = templates.macroExpansion(itemType.templateScript, expansionMacros); const absolutePath = path.join(project.projectFolderPath, relativePath) await utils.ensureFileExists(absolutePath, newFileText); switch (itemType.type) { case ItemType.preDeployScript: await project.addPreDeploymentScript(relativePath); break; case ItemType.postDeployScript: await project.addPostDeploymentScript(relativePath); break; case ItemType.publishProfile: await project.addNoneItem(relativePath); break; default: // a normal SQL object script await project.addSqlObjectScript(relativePath); break; } return absolutePath; } //#endregion /** * Builds a project, producing a dacpac * @param treeNode a treeItem in a project's hierarchy, to be used to obtain a Project * @returns path of the built dacpac */ public async buildProject(treeNode: dataworkspace.WorkspaceTreeItem): Promise; /** * Builds a project, producing a dacpac * @param project Project to be built * @returns path of the built dacpac */ public async buildProject(project: Project): Promise; public async buildProject(context: Project | dataworkspace.WorkspaceTreeItem): Promise { const project: Project = await this.getProjectFromContext(context); const startTime = new Date(); const currentBuildTimeInfo = `${startTime.toLocaleDateString()} ${constants.at} ${startTime.toLocaleTimeString()}`; let buildInfoNew = new DashboardData(project.projectFilePath, Status.inProgress, project.getProjectTargetVersion(), currentBuildTimeInfo); this.buildInfo.push(buildInfoNew); if (this.buildInfo.length - 1 === maxTableLength) { this.buildInfo.shift(); // Remove the first element to maintain the length } // get dlls and targets file needed for building for legacy style projects if (project.sqlProjStyle === mssql.ProjectType.LegacyStyle) { const result = await this.buildHelper.createBuildDirFolder(this._outputChannel); if (!result) { void vscode.window.showErrorMessage(constants.errorRetrievingBuildFiles); return ''; } } const options: ShellCommandOptions = { commandTitle: 'Build', workingDirectory: project.projectFolderPath, argument: this.buildHelper.constructBuildArguments(project.projectFilePath, this.buildHelper.extensionBuildDirPath, project.sqlProjStyle) }; try { const crossPlatCompatible: boolean = await Project.checkPromptCrossPlatStatus(project, true /* blocking prompt */); if (!crossPlatCompatible) { // user rejected updating for cross-plat void vscode.window.showErrorMessage(constants.projectNeedsUpdatingForCrossPlat(project.projectFileName)); return '' } } catch (error) { void vscode.window.showErrorMessage(utils.getErrorMessage(error)); return ''; } try { await this.netCoreTool.runDotnetCommand(options); const timeToBuild = new Date().getTime() - startTime.getTime(); const currentBuildIndex = this.buildInfo.findIndex(b => b.startDate === currentBuildTimeInfo); this.buildInfo[currentBuildIndex].status = Status.success; this.buildInfo[currentBuildIndex].timeToCompleteAction = utils.timeConversion(timeToBuild); TelemetryReporter.createActionEvent(TelemetryViews.ProjectController, TelemetryActions.build) .withAdditionalMeasurements({ duration: timeToBuild }) .withAdditionalProperties({ databaseSource: project.getDatabaseSourceValues().join(';') }) .send(); return project.dacpacOutputPath; } catch (err) { const timeToFailureBuild = new Date().getTime() - startTime.getTime(); const currentBuildIndex = this.buildInfo.findIndex(b => b.startDate === currentBuildTimeInfo); this.buildInfo[currentBuildIndex].status = Status.failed; this.buildInfo[currentBuildIndex].timeToCompleteAction = utils.timeConversion(timeToFailureBuild); TelemetryReporter.createErrorEvent2(TelemetryViews.ProjectController, TelemetryActions.build, err) .withAdditionalMeasurements({ duration: timeToFailureBuild }) .withAdditionalProperties({ databaseSource: project.getDatabaseSourceValues().join(';') }) .send(); const message = utils.getErrorMessage(err); if (err instanceof DotNetError) { // DotNetErrors already get shown by the netCoreTool so just show this one in the console console.error(message); } else { void vscode.window.showErrorMessage(constants.projBuildFailed(message)); } return ''; } } //#region Publish /** * Publishes a project to a new Azure server * @param context a treeItem in a project's hierarchy, to be used to obtain a Project or the Project itself * @param deployProfile deploy profile */ public async publishToNewAzureServer(context: Project | dataworkspace.WorkspaceTreeItem, deployProfile: ISqlDbDeployProfile): Promise { try { TelemetryReporter.sendActionEvent(TelemetryViews.ProjectController, TelemetryActions.publishToNewAzureServer); const project: Project = await this.getProjectFromContext(context); if (deployProfile?.deploySettings && deployProfile?.sqlDbSetting) { void utils.showInfoMessageWithOutputChannel(constants.creatingAzureSqlServer(deployProfile?.sqlDbSetting?.serverName), this._outputChannel); const connectionUri = await this.deployService.createNewAzureSqlServer(deployProfile); if (connectionUri) { deployProfile.deploySettings.connectionUri = connectionUri; const publishResult = await this.publishOrScriptProject(project, deployProfile.deploySettings, true); if (publishResult && publishResult.success) { if (deployProfile.sqlDbSetting) { // Connecting to the deployed db to add the profile to connection viewlet await this.connectionService.getConnection(deployProfile.sqlDbSetting, true, deployProfile.sqlDbSetting.dbName); } void vscode.window.showInformationMessage(constants.publishProjectSucceed); } else { void utils.showErrorMessageWithOutputChannel(constants.publishToNewAzureServerFailed, publishResult?.errorMessage || '', this._outputChannel); } } else { void utils.showErrorMessageWithOutputChannel(constants.publishToNewAzureServerFailed, constants.deployProjectFailedMessage, this._outputChannel); } } } catch (error) { void utils.showErrorMessageWithOutputChannel(constants.publishToNewAzureServerFailed, error, this._outputChannel); TelemetryReporter.sendErrorEvent2(TelemetryViews.ProjectController, TelemetryActions.publishToNewAzureServer, error); } } /** * Publishes a project to docker container * @param context a treeItem in a project's hierarchy, to be used to obtain a Project or the Project itself * @param deployProfile */ public async publishToDockerContainer(context: Project | dataworkspace.WorkspaceTreeItem, deployProfile: IPublishToDockerSettings): Promise { const project: Project = await this.getProjectFromContext(context); // Removing the path separator from the image base name to be able to add that in the telemetry. With the separator the name is flagged as user path which is not true // We only need to know the image base parts so it's ok to use a different separator when adding to telemetry const dockerImageNameForTelemetry = deployProfile.dockerSettings.dockerBaseImage.replace(/\//gi, '_'); try { TelemetryReporter.createActionEvent(TelemetryViews.ProjectController, TelemetryActions.publishToContainer) .withAdditionalProperties({ dockerBaseImage: dockerImageNameForTelemetry }) .send(); void utils.showInfoMessageWithOutputChannel(constants.publishingProjectMessage, this._outputChannel); const connectionUri = await this.deployService.deployToContainer(deployProfile, project); if (connectionUri) { deployProfile.sqlProjectPublishSettings.connectionUri = connectionUri; } if (deployProfile.sqlProjectPublishSettings.connectionUri) { const publishResult = await this.publishOrScriptProject(project, deployProfile.sqlProjectPublishSettings, true); if (publishResult && publishResult.success) { await this.connectionService.getConnection(deployProfile.dockerSettings, true, deployProfile.dockerSettings.dbName); void vscode.window.showInformationMessage(constants.publishProjectSucceed); } else { void utils.showErrorMessageWithOutputChannel(constants.publishToContainerFailed, publishResult?.errorMessage || '', this._outputChannel); } } else { void utils.showErrorMessageWithOutputChannel(constants.publishToContainerFailed, constants.deployProjectFailedMessage, this._outputChannel); } } catch (error) { void utils.showErrorMessageWithOutputChannel(constants.publishToContainerFailed, error, this._outputChannel); TelemetryReporter.createErrorEvent2(TelemetryViews.ProjectController, TelemetryActions.publishToContainer, error) .withAdditionalProperties({ dockerBaseImage: dockerImageNameForTelemetry }) .send(); } return; } /** * Builds and publishes a project * @param treeNode a treeItem in a project's hierarchy, to be used to obtain a Project */ public async publishProject(treeNode: dataworkspace.WorkspaceTreeItem): Promise; /** * Builds and publishes a project * @param project Project to be built and published */ public async publishProject(project: Project): Promise; public async publishProject(context: Project | dataworkspace.WorkspaceTreeItem): Promise { const project: Project = await this.getProjectFromContext(context); if (utils.getAzdataApi()) { let publishDatabaseDialog = this.getPublishDialog(project); publishDatabaseDialog.publish = async (proj, prof) => this.publishOrScriptProject(proj, prof, true); publishDatabaseDialog.publishToContainer = async (proj, prof) => this.publishToDockerContainer(proj, prof); publishDatabaseDialog.generateScript = async (proj, prof) => this.publishOrScriptProject(proj, prof, false); publishDatabaseDialog.readPublishProfile = async (profileUri) => readPublishProfile(profileUri); publishDatabaseDialog.savePublishProfile = async (profilePath, databaseName, connectionString, sqlCommandVariableValues, deploymentOptions) => savePublishProfile(profilePath, databaseName, connectionString, sqlCommandVariableValues, deploymentOptions); publishDatabaseDialog.openDialog(); return publishDatabaseDialog.waitForClose(); } else { return this.publishDatabase(project); } } public getPublishDialog(project: Project): PublishDatabaseDialog { return new PublishDatabaseDialog(project); } /** * Create flow for Publishing a database using only VS Code-native APIs such as QuickPick */ private async publishDatabase(project: Project): Promise { const publishTarget = await launchPublishTargetOption(project); // Return when user hits escape if (!publishTarget) { return undefined; } if (publishTarget === constants.PublishTargetType.docker) { const publishToDockerSettings = await getPublishToDockerSettings(project); void promptForSavingProfile(project, publishToDockerSettings); // not awaiting this call, because saving profile should not stop the actual publish workflow if (!publishToDockerSettings) { // User cancelled return; } await this.publishToDockerContainer(project, publishToDockerSettings); } else if (publishTarget === constants.PublishTargetType.newAzureServer) { try { const settings = await launchCreateAzureServerQuickPick(project, this.azureSqlClient); void promptForSavingProfile(project, settings); // not awaiting this call, because saving profile should not stop the actual publish workflow if (settings?.deploySettings && settings?.sqlDbSetting) { await this.publishToNewAzureServer(project, settings); } } catch (error) { void utils.showErrorMessageWithOutputChannel(constants.publishToNewAzureServerFailed, error, this._outputChannel); } } else { let settings: ISqlProjectPublishSettings | undefined = await getPublishDatabaseSettings(project); void promptForSavingProfile(project, settings); // not awaiting this call, because saving profile should not stop the actual publish workflow if (settings) { // 5. Select action to take const action = await vscode.window.showQuickPick( [constants.generateScriptButtonText, constants.publish], { title: constants.chooseAction, ignoreFocusOut: true }); if (!action) { return; } await this.publishOrScriptProject(project, settings, action === constants.publish); } } } /** * Builds and either deploys or generates a deployment script for the specified project. * @param project The project to deploy * @param settings The settings used to configure the deployment * @param publish Whether to publish the deployment or just generate a script * @returns The DacFx result of the deployment */ public async publishOrScriptProject(project: Project, settings: ISqlProjectPublishSettings, publish: boolean): Promise { const telemetryProps: Record = {}; const telemetryMeasures: Record = {}; const buildStartTime = new Date().getTime(); const dacpacPath = await this.buildProject(project); const buildEndTime = new Date().getTime(); telemetryMeasures.buildDuration = buildEndTime - buildStartTime; telemetryProps.buildSucceeded = (dacpacPath !== '').toString(); telemetryProps.databaseSource = project.getDatabaseSourceValues().join(';'); if (!dacpacPath) { TelemetryReporter.createErrorEvent2(TelemetryViews.ProjectController, TelemetryActions.publishProject) .withAdditionalProperties(telemetryProps) .withAdditionalMeasurements(telemetryMeasures) .send(); return undefined; // buildProject() handles displaying the error } // copy dacpac to temp location before publishing const tempPath = path.join(os.tmpdir(), `${path.parse(dacpacPath).name}_${new Date().getTime()}${constants.sqlprojExtension}`); await fs.copyFile(dacpacPath, tempPath); const dacFxService = await utils.getDacFxService(); let result: mssql.DacFxResult; telemetryProps.profileUsed = (settings.publishProfileUri !== undefined ? true : false).toString(); const currentDate = new Date(); const actionStartTime = currentDate.getTime(); const currentPublishTimeInfo = `${currentDate.toLocaleDateString()} ${constants.at} ${currentDate.toLocaleTimeString()}`; let publishInfoNew = new PublishData(project.projectFilePath, Status.inProgress, project.getProjectTargetVersion(), currentPublishTimeInfo, settings.databaseName, settings.serverName); this.publishInfo.push(publishInfoNew); if (this.publishInfo.length - 1 === maxTableLength) { this.publishInfo.shift(); // Remove the first element to maintain the length } try { const azdataApi = utils.getAzdataApi(); if (publish) { telemetryProps.publishAction = 'deploy'; if (azdataApi) { result = await (dacFxService as mssql.IDacFxService).deployDacpac(tempPath, settings.databaseName, true, settings.connectionUri, azdataApi.TaskExecutionMode.execute, settings.sqlCmdVariables, settings.deploymentOptions as mssql.DeploymentOptions); } else { // Have to cast to unknown first to get around compiler error since the mssqlVscode doesn't exist as an actual module at runtime result = await (dacFxService as mssqlVscode.IDacFxService).deployDacpac(tempPath, settings.databaseName, true, settings.connectionUri, TaskExecutionMode.execute as unknown as mssqlVscode.TaskExecutionMode, settings.sqlCmdVariables, settings.deploymentOptions as mssqlVscode.DeploymentOptions); } } else { telemetryProps.publishAction = 'generateScript'; if (azdataApi) { result = await (dacFxService as mssql.IDacFxService).generateDeployScript(tempPath, settings.databaseName, settings.connectionUri, azdataApi.TaskExecutionMode.script, settings.sqlCmdVariables, settings.deploymentOptions as mssql.DeploymentOptions); } else { // Have to cast to unknown first to get around compiler error since the mssqlVscode doesn't exist as an actual module at runtime result = await (dacFxService as mssqlVscode.IDacFxService).generateDeployScript(tempPath, settings.databaseName, settings.connectionUri, TaskExecutionMode.script as unknown as mssqlVscode.TaskExecutionMode, settings.sqlCmdVariables, settings.deploymentOptions as mssqlVscode.DeploymentOptions); } } } catch (err) { const actionEndTime = new Date().getTime(); const timeToFailurePublish = actionEndTime - actionStartTime; telemetryProps.actionDuration = timeToFailurePublish.toString(); telemetryProps.totalDuration = (actionEndTime - buildStartTime).toString(); TelemetryReporter.createErrorEvent2(TelemetryViews.ProjectController, TelemetryActions.publishProject, err) .withAdditionalProperties(telemetryProps) .send(); const currentPublishIndex = this.publishInfo.findIndex(d => d.startDate === currentPublishTimeInfo); this.publishInfo[currentPublishIndex].status = Status.failed; this.publishInfo[currentPublishIndex].timeToCompleteAction = utils.timeConversion(timeToFailurePublish); throw err; } const actionEndTime = new Date().getTime(); const timeToPublish = actionEndTime - actionStartTime; telemetryProps.actionDuration = timeToPublish.toString(); telemetryProps.totalDuration = (actionEndTime - buildStartTime).toString(); telemetryProps.sqlcmdVariablesCount = project.sqlCmdVariables.size.toString(); telemetryProps.projectTargetPLatform = project.getProjectTargetVersion(); const currentPublishIndex = this.publishInfo.findIndex(d => d.startDate === currentPublishTimeInfo); this.publishInfo[currentPublishIndex].status = result.success ? Status.success : Status.failed; this.publishInfo[currentPublishIndex].timeToCompleteAction = utils.timeConversion(timeToPublish); TelemetryReporter.createActionEvent(TelemetryViews.ProjectController, TelemetryActions.publishProject) .withAdditionalProperties(telemetryProps) .send(); return result; } //#endregion /** * Launches the schema compare extension with the source and target * @param source source for schema compare. Either a connection or project node * @param targetParam target for schema compare */ public async schemaCompare(source: dataworkspace.WorkspaceTreeItem | azdataType.IConnectionProfile, targetParam: any = undefined): Promise { try { // check if schema compare extension is installed if (vscode.extensions.getExtension(constants.schemaCompareExtensionId)) { let sourceParam; if (source as dataworkspace.WorkspaceTreeItem) { sourceParam = (await this.getProjectFromContext(source as dataworkspace.WorkspaceTreeItem)).projectFilePath; } else { sourceParam = source as azdataType.IConnectionProfile; } try { TelemetryReporter.sendActionEvent(TelemetryViews.ProjectController, TelemetryActions.projectSchemaCompareCommandInvoked); await vscode.commands.executeCommand(constants.schemaCompareStartCommand, sourceParam, targetParam, undefined); } catch (e) { throw new Error(constants.buildFailedCannotStartSchemaCompare); } } else { throw new Error(constants.schemaCompareNotInstalled); } } catch (err) { const props: Record = {}; const message = utils.getErrorMessage(err); if (message === constants.buildFailedCannotStartSchemaCompare || message === constants.schemaCompareNotInstalled) { props.errorMessage = message; } TelemetryReporter.createErrorEvent2(TelemetryViews.ProjectController, TelemetryActions.projectSchemaCompareCommandInvoked, err) .withAdditionalProperties(props) .send(); void vscode.window.showErrorMessage(utils.getErrorMessage(err)); } } //#region Add/Exclude/Delete Item public async addFolderPrompt(treeNode: dataworkspace.WorkspaceTreeItem): Promise { const project = await this.getProjectFromContext(treeNode); const projectRelativeUri = vscode.Uri.file(path.basename((treeNode.element as BaseProjectTreeItem).projectFileUri.fsPath, constants.sqlprojExtension)); const relativePathToParent = this.getRelativePath(projectRelativeUri, treeNode.element); const absolutePathToParent = path.join(project.projectFolderPath, relativePathToParent); const newFolderName = await this.promptForNewObjectName(new templates.ProjectScriptType(ItemType.folder, constants.folderFriendlyName, ''), project, absolutePathToParent); if (!newFolderName) { return; // user cancelled } const relativeFolderPath = path.join(relativePathToParent, newFolderName); try { // check if folder already exists or is a reserved folder const absoluteFolderPath = path.join(absolutePathToParent, newFolderName); const folderExists = await utils.exists(absoluteFolderPath); if (folderExists || this.isReservedFolder(absoluteFolderPath, project.projectFolderPath)) { throw new Error(constants.folderAlreadyExists(path.parse(absoluteFolderPath).name)); } await project.addFolder(relativeFolderPath); this.refreshProjectsTree(treeNode); } catch (err) { void vscode.window.showErrorMessage(utils.getErrorMessage(err)); } } private async promptForNewObjectName(itemType: templates.ProjectScriptType, _project: ISqlProject, folderPath: string, fileExtension?: string, defaultName?: string): Promise { const suggestedName = utils.sanitizeStringForFilename(defaultName ?? itemType.friendlyName.replace(/\s+/g, '')); let counter: number = 0; do { counter++; } while (counter < Number.MAX_SAFE_INTEGER && await utils.exists(path.join(folderPath, `${suggestedName}${counter}${(fileExtension ?? '')}`))); const itemObjectName = await vscode.window.showInputBox({ prompt: constants.newObjectNamePrompt(itemType.friendlyName), value: `${suggestedName}${counter}`, validateInput: (value) => { return utils.isValidBasenameErrorMessage(value); }, ignoreFocusOut: true, }); return itemObjectName; } public isReservedFolder(absoluteFolderPath: string, projectFolderPath: string): boolean { const sameName = constants.reservedProjectFolders.find(f => f === path.parse(absoluteFolderPath).name) !== undefined; const sameLocation = path.parse(absoluteFolderPath).dir === projectFolderPath; return sameName && sameLocation; } public async addItemPromptFromNode(treeNode: dataworkspace.WorkspaceTreeItem, itemTypeName?: string): Promise { const projectRelativeUri = vscode.Uri.file(path.basename((treeNode.element as BaseProjectTreeItem).projectFileUri.fsPath, constants.sqlprojExtension)); await this.addItemPrompt(await this.getProjectFromContext(treeNode), this.getRelativePath(projectRelativeUri, treeNode.element), { itemType: itemTypeName }, treeNode.treeDataProvider as SqlDatabaseProjectTreeViewProvider); } public async addItemPrompt(project: ISqlProject, relativePath: string, options?: AddItemOptions, treeDataProvider?: SqlDatabaseProjectTreeViewProvider): Promise { let itemTypeName = options?.itemType; if (!itemTypeName) { const items: vscode.QuickPickItem[] = []; for (const itemType of templates.projectScriptTypes()) { items.push({ label: itemType.friendlyName }); } itemTypeName = (await vscode.window.showQuickPick(items, { canPickMany: false }))?.label; if (!itemTypeName) { return; // user cancelled } } const itemType = templates.get(itemTypeName); const absolutePathToParent = path.join(project.projectFolderPath, relativePath); const isItemTypePublishProfile = itemTypeName === constants.publishProfileFriendlyName || itemTypeName === ItemType.publishProfile; const fileExtension = isItemTypePublishProfile ? constants.publishProfileExtension : constants.sqlFileExtension; const defaultName = isItemTypePublishProfile ? `${project.projectFileName}_` : options?.defaultName; let itemObjectName = await this.promptForNewObjectName(itemType, project, absolutePathToParent, fileExtension, defaultName); itemObjectName = itemObjectName?.trim(); if (!itemObjectName) { return; // user cancelled } const relativeFilePath = path.join(relativePath, itemObjectName + fileExtension); const telemetryProps: Record = { itemType: itemType.type }; const telemetryMeasurements: Record = {}; if (itemType.type === ItemType.preDeployScript) { telemetryMeasurements.numPredeployScripts = project.preDeployScripts.length; } else if (itemType.type === ItemType.postDeployScript) { telemetryMeasurements.numPostdeployScripts = project.postDeployScripts.length; } try { const absolutePath = await this.addFileToProjectFromTemplate(project, itemType, relativeFilePath, new Map([['OBJECT_NAME', itemObjectName], ['PROJECT_NAME', project.projectFileName]])); TelemetryReporter.createActionEvent(TelemetryViews.ProjectTree, TelemetryActions.addItemFromTree) .withAdditionalProperties(telemetryProps) .withAdditionalMeasurements(telemetryMeasurements) .send(); await vscode.commands.executeCommand(constants.vscodeOpenCommand, vscode.Uri.file(absolutePath)); treeDataProvider?.notifyTreeDataChanged(); } catch (err) { void vscode.window.showErrorMessage(utils.getErrorMessage(err)); TelemetryReporter.createErrorEvent2(TelemetryViews.ProjectTree, TelemetryActions.addItemFromTree, err) .withAdditionalProperties(telemetryProps) .withAdditionalMeasurements(telemetryMeasurements) .send(); } } public async addExistingItemPrompt(treeNode: dataworkspace.WorkspaceTreeItem): Promise { const project = await this.getProjectFromContext(treeNode); const uris = await vscode.window.showOpenDialog({ canSelectFiles: true, canSelectFolders: false, canSelectMany: false, openLabel: constants.selectString, title: constants.selectFileString }); if (!uris) { return; // user cancelled } try { TelemetryReporter.sendActionEvent(TelemetryViews.ProjectTree, TelemetryActions.addExistingItem); await project.addExistingItem(uris[0].fsPath); this.refreshProjectsTree(treeNode); } catch (err) { void vscode.window.showErrorMessage(utils.getErrorMessage(err)); TelemetryReporter.sendErrorEvent2(TelemetryViews.ProjectTree, TelemetryActions.addExistingItem, err); } } public async exclude(context: dataworkspace.WorkspaceTreeItem): Promise { const node = context.element as BaseProjectTreeItem; const project = await this.getProjectFromContext(node); if (node.entryKey) { TelemetryReporter.sendActionEvent(TelemetryViews.ProjectTree, TelemetryActions.excludeFromProject); switch (node.type) { case constants.DatabaseProjectItemType.sqlObjectScript: case constants.DatabaseProjectItemType.table: case constants.DatabaseProjectItemType.externalStreamingJob: await project.excludeSqlObjectScript(node.entryKey); break; case constants.DatabaseProjectItemType.folder: await project.excludeFolder(node.entryKey); break; case constants.DatabaseProjectItemType.preDeploymentScript: await project.excludePreDeploymentScript(node.entryKey); break; case constants.DatabaseProjectItemType.postDeploymentScript: await project.excludePostDeploymentScript(node.entryKey); break; case constants.DatabaseProjectItemType.noneFile: case constants.DatabaseProjectItemType.publishProfile: await project.excludeNoneItem(node.entryKey); break; default: throw new Error(constants.unhandledExcludeType(node.type)); } } else { TelemetryReporter.sendErrorEvent2(TelemetryViews.ProjectTree, TelemetryActions.excludeFromProject); void vscode.window.showErrorMessage(constants.unableToPerformAction(constants.excludeAction, node.relativeProjectUri.path)); } this.refreshProjectsTree(context); } public async delete(context: dataworkspace.WorkspaceTreeItem): Promise { const node = context.element as BaseProjectTreeItem; const project = await this.getProjectFromContext(node); let confirmationPrompt; if (node instanceof DatabaseReferenceTreeItem) { confirmationPrompt = constants.deleteReferenceConfirmation(node.friendlyName); } else if (node instanceof SqlCmdVariableTreeItem) { confirmationPrompt = constants.deleteSqlCmdVariableConfirmation(node.friendlyName); } else if (node instanceof FolderNode) { confirmationPrompt = constants.deleteConfirmationContents(node.friendlyName); } else { confirmationPrompt = constants.deleteConfirmation(node.friendlyName); } const response = await vscode.window.showWarningMessage(confirmationPrompt, { modal: true }, constants.yesString); if (response !== constants.yesString) { return; } try { if (node instanceof DatabaseReferenceTreeItem) { const databaseReference = this.getDatabaseReference(project, node); if (databaseReference) { await project.deleteDatabaseReferenceByEntry(databaseReference); } } else if (node instanceof SqlCmdVariableTreeItem) { await project.deleteSqlCmdVariable(node.friendlyName); } else if (node instanceof FolderNode) { await project.deleteFolder(node.entryKey); } else if (node instanceof FileNode) { switch (node.type) { case constants.DatabaseProjectItemType.sqlObjectScript: case constants.DatabaseProjectItemType.table: case constants.DatabaseProjectItemType.externalStreamingJob: await project.deleteSqlObjectScript(node.entryKey); break; case constants.DatabaseProjectItemType.preDeploymentScript: await project.deletePreDeploymentScript(node.entryKey); break; case constants.DatabaseProjectItemType.postDeploymentScript: await project.deletePostDeploymentScript(node.entryKey); break; case constants.DatabaseProjectItemType.noneFile: case constants.DatabaseProjectItemType.publishProfile: await project.deleteNoneItem(node.entryKey); break; default: throw new Error(constants.unhandledDeleteType(node.type)); } } TelemetryReporter.createActionEvent(TelemetryViews.ProjectTree, TelemetryActions.deleteObjectFromProject) .withAdditionalProperties({ objectType: node.constructor.name }) .send(); this.refreshProjectsTree(context); } catch (err) { TelemetryReporter.createErrorEvent2(TelemetryViews.ProjectTree, TelemetryActions.deleteObjectFromProject) .withAdditionalProperties({ objectType: node.constructor.name }) .send(); void vscode.window.showErrorMessage(constants.unableToPerformAction(constants.deleteAction, node.relativeProjectUri.path, utils.getErrorMessage(err))); } } public async rename(context: dataworkspace.WorkspaceTreeItem): Promise { const node = context.element as BaseProjectTreeItem; const project = await this.getProjectFromContext(node); const originalAbsolutePath = utils.getPlatformSafeFileEntryPath(node.projectFileUri.fsPath); const originalName = path.basename(node.friendlyName); const originalExt = path.extname(node.friendlyName); // need to use quickpick because input box isn't supported in treeviews // https://github.com/microsoft/vscode/issues/117502 and https://github.com/microsoft/vscode/issues/97190 const newFileName = await vscode.window.showInputBox( { title: constants.enterNewName, value: originalName, valueSelection: [0, path.basename(originalName, originalExt).length], ignoreFocusOut: true, validateInput: async (newName) => { return await this.fileAlreadyExists(newName, originalAbsolutePath) ? constants.fileAlreadyExists(newName) : undefined; } }); if (!newFileName) { return; } const newFilePath = path.join(path.dirname(utils.getPlatformSafeFileEntryPath(node.relativeProjectUri.fsPath!)), newFileName); const result = await project.move(node, newFilePath); if (result?.success) { TelemetryReporter.sendActionEvent(TelemetryViews.ProjectTree, TelemetryActions.rename); } else { TelemetryReporter.sendErrorEvent2(TelemetryViews.ProjectTree, TelemetryActions.rename); void vscode.window.showErrorMessage(constants.errorRenamingFile(node.entryKey!, newFilePath, result?.errorMessage)); } this.refreshProjectsTree(context); } private fileAlreadyExists(newFileName: string, previousFilePath: string): Promise { return utils.exists(path.join(path.dirname(previousFilePath), newFileName)); } /** * Opens a quickpick to edit the value of the SQLCMD variable launched from * @param context */ public async editSqlCmdVariable(context: dataworkspace.WorkspaceTreeItem): Promise { const node = context.element as SqlCmdVariableTreeItem; const project = await this.getProjectFromContext(node); const variableName = node.friendlyName; const originalValue = project.sqlCmdVariables.get(variableName); const newValue = await vscode.window.showInputBox( { title: constants.enterNewValueForVar(variableName), value: originalValue, ignoreFocusOut: true }); if (!newValue) { return; } await project.updateSqlCmdVariable(variableName, newValue) this.refreshProjectsTree(context); } /** * Opens a quickpick to add a new SQLCMD variable to the project * @param context */ public async addSqlCmdVariable(context: dataworkspace.WorkspaceTreeItem): Promise { const project = await this.getProjectFromContext(context); const variableName = await vscode.window.showInputBox( { title: constants.enterNewSqlCmdVariableName, ignoreFocusOut: true, validateInput: (value) => { return project.sqlCmdVariables.has(value) ? constants.sqlcmdVariableAlreadyExists : undefined; } }); if (!variableName) { return; } const defaultValue = await vscode.window.showInputBox( { title: constants.enterNewSqlCmdVariableDefaultValue(variableName), ignoreFocusOut: true }); if (!defaultValue) { return; } await project.addSqlCmdVariable(variableName, defaultValue); this.refreshProjectsTree(context); } private getDatabaseReference(project: Project, context: BaseProjectTreeItem): IDatabaseReferenceProjectEntry | undefined { const databaseReference = context as DatabaseReferenceTreeItem; if (databaseReference) { return project.databaseReferences.find(r => r.referenceName === databaseReference.treeItem.label); } return undefined; } //#endregion /** * Opens the folder containing the project * @param context a treeItem in a project's hierarchy, to be used to obtain a Project */ public async openContainingFolder(context: dataworkspace.WorkspaceTreeItem): Promise { const project = await this.getProjectFromContext(context); await vscode.commands.executeCommand(constants.revealFileInOsCommand, vscode.Uri.file(project.projectFilePath)); } /** * Open the project indicated by `context` in the workspace * @param context a SqlProjectReferenceTreeItem in the project's tree */ public async openReferencedSqlProject(context: dataworkspace.WorkspaceTreeItem): Promise { const node = context.element as BaseProjectTreeItem; const project = await this.getProjectFromContext(node); if (!(node instanceof SqlProjectReferenceTreeItem)) { return; } const absolutePath = path.normalize(path.join(project.projectFolderPath, node.reference.fsUri.fsPath)); await this.openProjectInWorkspace(absolutePath); } /** * Opens the .sqlproj file for the given project. Upon update of file, prompts user to * reload their project. * @param context a treeItem in a project's hierarchy, to be used to obtain a Project */ public async editProjectFile(context: dataworkspace.WorkspaceTreeItem): Promise { const project = await this.getProjectFromContext(context); try { await vscode.commands.executeCommand(constants.vscodeOpenCommand, vscode.Uri.file(project.projectFilePath)); TelemetryReporter.sendActionEvent(TelemetryViews.ProjectTree, TelemetryActions.editProjectFile); const projFileWatcher: vscode.FileSystemWatcher = vscode.workspace.createFileSystemWatcher(project.projectFilePath); this.projFileWatchers.set(project.projectFilePath, projFileWatcher); projFileWatcher.onDidChange(async () => { const result = await vscode.window.showInformationMessage(constants.reloadProject, constants.yesString, constants.noString); if (result === constants.yesString) { return this.reloadProject(context); } }); // stop watching for changes to the sqlproj after it's closed const closeSqlproj = vscode.workspace.onDidCloseTextDocument((d) => { if (this.projFileWatchers.has(d.uri.fsPath)) { this.projFileWatchers.get(d.uri.fsPath)!.dispose(); this.projFileWatchers.delete(d.uri.fsPath); closeSqlproj.dispose(); } }); } catch (err) { void vscode.window.showErrorMessage(utils.getErrorMessage(err)); } } /** * Opens a file in the editor and adds a file watcher to check if a create table statement has been added * @param fileSystemUri uri of file * @param node node of file in the tree */ public async openFileWithWatcher(fileSystemUri: vscode.Uri, node: FileNode): Promise { await vscode.commands.executeCommand(constants.vscodeOpenCommand, fileSystemUri); const project = await Project.openProject(node.projectFileUri.fsPath); const projectTargetVersion = project.getProjectTargetVersion(); const initiallyContainsCreateTableStatement = await utils.fileContainsCreateTableStatement(fileSystemUri.fsPath, projectTargetVersion); const fileWatcher: vscode.FileSystemWatcher = vscode.workspace.createFileSystemWatcher(fileSystemUri.fsPath); this.fileWatchers.set(fileSystemUri.fsPath, { fileWatcher: fileWatcher, containsCreateTableStatement: initiallyContainsCreateTableStatement }); fileWatcher.onDidChange(async (uri: vscode.Uri) => { const afterContainsCreateTableStatement = await utils.fileContainsCreateTableStatement(fileSystemUri.fsPath, projectTargetVersion); const previousStatus = this.fileWatchers.get(uri.fsPath)?.containsCreateTableStatement; // if the contains create table statement status is different, reload the project so that the "Open in Designer" menu option // on the file node is there if a create table statement has been added or removed if it's been removed if (previousStatus !== afterContainsCreateTableStatement) { utils.getDataWorkspaceExtensionApi().refreshProjectsTree(); this.fileWatchers.get(uri.fsPath)!.containsCreateTableStatement = afterContainsCreateTableStatement; } }); // stop watching for changes to the file after it's closed const closeSqlproj = vscode.workspace.onDidCloseTextDocument((d) => { if (this.fileWatchers.has(d.uri.fsPath)) { this.fileWatchers.get(d.uri.fsPath)?.fileWatcher.dispose(); this.fileWatchers.delete(d.uri.fsPath); closeSqlproj.dispose(); } }); } /** * Reloads the given project. Throws an error if given project is not a valid open project. * @param context */ public async reloadProject(context: dataworkspace.WorkspaceTreeItem): Promise { const project = await this.getProjectFromContext(context); if (project) { // won't open any newly referenced projects, but otherwise matches the behavior of reopening the project await project.readProjFile(); this.refreshProjectsTree(context); } else { throw new Error(constants.invalidProjectReload); } } /** * Changes the project's DSP to the selected target platform * @param context a treeItem in a project's hierarchy, to be used to obtain a Project */ public async changeTargetPlatform(context: Project | dataworkspace.WorkspaceTreeItem): Promise { const project = await this.getProjectFromContext(context); const selectedTargetPlatform = (await vscode.window.showQuickPick((Array.from(constants.targetPlatformToVersion.keys())).map(version => { return { label: version }; }), { canPickMany: false, placeHolder: constants.selectTargetPlatform(constants.getTargetPlatformFromVersion(project.getProjectTargetVersion())) }))?.label; if (selectedTargetPlatform) { await project.changeTargetPlatform(constants.targetPlatformToVersion.get(selectedTargetPlatform)!); void vscode.window.showInformationMessage(constants.currentTargetPlatform(project.projectFileName, constants.getTargetPlatformFromVersion(project.getProjectTargetVersion()))); } } //#region database references /** * Adds a database reference to the project * @param context a treeItem in a project's hierarchy, to be used to obtain a Project */ public async addDatabaseReference(context: Project | dataworkspace.WorkspaceTreeItem): Promise { const project = await this.getProjectFromContext(context); if (utils.getAzdataApi()) { const addDatabaseReferenceDialog = this.getAddDatabaseReferenceDialog(project); addDatabaseReferenceDialog.addReference = async (proj, settings) => await this.addDatabaseReferenceCallback(proj, settings, context as dataworkspace.WorkspaceTreeItem); await addDatabaseReferenceDialog.openDialog(); return addDatabaseReferenceDialog; } else { const settings = await addDatabaseReferenceQuickpick(project); if (settings) { await this.addDatabaseReferenceCallback(project, settings, context as dataworkspace.WorkspaceTreeItem); } return undefined; } } public getAddDatabaseReferenceDialog(project: Project): AddDatabaseReferenceDialog { return new AddDatabaseReferenceDialog(project); } /** * Adds a database reference to a project, after selections have been made in the dialog * @param project project to which to add the database reference * @param settings settings for the database reference * @param context a treeItem in a project's hierarchy, to be used to obtain a Project */ public async addDatabaseReferenceCallback(project: Project, settings: AddDatabaseReferenceSettings, context: dataworkspace.WorkspaceTreeItem): Promise { try { if ((settings).projectName !== undefined) { // get project path and guid const projectReferenceSettings = settings as IProjectReferenceSettings; const workspaceProjects = await utils.getSqlProjectsInWorkspace(); const referencedProject = await Project.openProject(workspaceProjects.filter(p => path.parse(p.fsPath).name === projectReferenceSettings.projectName)[0].fsPath); const relativePath = path.relative(project.projectFolderPath, referencedProject?.projectFilePath!); projectReferenceSettings.projectRelativePath = vscode.Uri.file(relativePath); projectReferenceSettings.projectGuid = referencedProject?.projectGuid!; const projectReferences = referencedProject?.databaseReferences.filter(r => r instanceof SqlProjectReferenceProjectEntry) ?? []; // check for cirular dependency for (let r of projectReferences) { if ((r).projectName === project.projectFileName) { void vscode.window.showErrorMessage(constants.cantAddCircularProjectReference(referencedProject?.projectFileName!)); return; } } await project.addProjectReference(projectReferenceSettings); } else if ((settings).systemDb !== undefined) { await project.addSystemDatabaseReference(settings); } else if ((settings).dacpacFileLocation !== undefined) { // update dacpacFileLocation to relative path to project file const dacpacRefSettings = settings as IDacpacReferenceSettings; dacpacRefSettings.dacpacFileLocation = vscode.Uri.file(path.relative(project.projectFolderPath, dacpacRefSettings.dacpacFileLocation.fsPath)); await project.addDatabaseReference(dacpacRefSettings); } else { await project.addNugetPackageReference(settings); } this.refreshProjectsTree(context); } catch (err) { void vscode.window.showErrorMessage(utils.getErrorMessage(err)); } } //#endregion /** * Validates the contents of an external streaming job's query against the last-built dacpac. * If no dacpac exists at the output path, one will be built first. * @param node a treeItem in a project's hierarchy, to be used to obtain a Project */ public async validateExternalStreamingJob(node: dataworkspace.WorkspaceTreeItem): Promise { const project: Project = await this.getProjectFromContext(node); let dacpacPath: string = project.dacpacOutputPath; const preExistingDacpac = await utils.exists(dacpacPath); const telemetryProps: Record = { preExistingDacpac: preExistingDacpac.toString() }; if (!preExistingDacpac) { dacpacPath = await this.buildProject(project); } const streamingJobDefinition: string = (await fs.readFile(node.element.fileSystemUri.fsPath)).toString(); const dacFxService = await utils.getDacFxService(); const actionStartTime = new Date().getTime(); const result: mssql.ValidateStreamingJobResult = await dacFxService.validateStreamingJob(dacpacPath, streamingJobDefinition); const duration = new Date().getTime() - actionStartTime; telemetryProps.success = result.success.toString(); if (result.success) { void vscode.window.showInformationMessage(constants.externalStreamingJobValidationPassed); } else { void vscode.window.showErrorMessage(result.errorMessage); } TelemetryReporter.createActionEvent(TelemetryViews.ProjectTree, TelemetryActions.runStreamingJobValidation) .withAdditionalProperties(telemetryProps) .withAdditionalMeasurements({ duration: duration }) .send(); return result; } //#region AutoRest public async selectAutorestSpecFile(): Promise { let quickpickSelection = await vscode.window.showQuickPick( [constants.browseEllipsisWithIcon], { title: constants.selectSpecFile, ignoreFocusOut: true }); if (!quickpickSelection) { return; } const filters: { [name: string]: string[] } = {}; filters[constants.specSelectionText] = constants.openApiSpecFileExtensions; let uris = await vscode.window.showOpenDialog({ canSelectFiles: true, canSelectFolders: false, canSelectMany: false, openLabel: constants.selectString, filters: filters, title: constants.selectSpecFile }); if (!uris) { return; } return uris[0].fsPath; } /** * @returns \{ newProjectFolder: 'C:\Source\MyProject', * outputFolder: 'C:\Source', * projectName: 'MyProject'} */ public async selectAutorestProjectLocation(projectName: string, defaultOutputLocation: vscode.Uri | undefined): Promise<{ newProjectFolder: string, outputFolder: string, projectName: string } | undefined> { let newProjectFolder = defaultOutputLocation ? path.join(defaultOutputLocation.fsPath, projectName) : ''; let outputFolder = defaultOutputLocation?.fsPath || ''; while (true) { let quickPickTitle = ''; if (newProjectFolder && await utils.exists(newProjectFolder)) { // Folder already exists at target location, prompt for new location quickPickTitle = constants.folderAlreadyExistsChooseNewLocation(newProjectFolder); } else if (!newProjectFolder) { // No target location yet quickPickTitle = constants.selectProjectLocation; } else { // Folder doesn't exist at target location so we're done break; } const quickpickSelection = await vscode.window.showQuickPick([constants.browseEllipsisWithIcon], { title: quickPickTitle, ignoreFocusOut: true }); if (!quickpickSelection) { return; } const folders = await vscode.window.showOpenDialog({ canSelectFiles: false, canSelectFolders: true, canSelectMany: false, openLabel: constants.selectString, defaultUri: defaultOutputLocation ?? vscode.workspace.workspaceFolders?.[0]?.uri, title: constants.selectProjectLocation }); if (!folders) { return; } outputFolder = folders[0].fsPath; newProjectFolder = path.join(outputFolder, projectName); } return { newProjectFolder, outputFolder, projectName }; } public async generateAutorestFiles(specPath: string, newProjectFolder: string): Promise { await fs.mkdir(newProjectFolder, { recursive: true }); return vscode.window.withProgress( { location: vscode.ProgressLocation.Notification, title: constants.generatingProjectFromAutorest(path.basename(specPath)), cancellable: false }, async (_progress, _token) => { return this.autorestHelper.generateAutorestFiles(specPath, newProjectFolder); }); } /** * Adds the provided project in the workspace, opening it in the projects viewlet * @param projectFilePath */ public async openProjectInWorkspace(projectFilePath: string): Promise { const workspaceApi = utils.getDataWorkspaceExtensionApi(); await workspaceApi.validateWorkspace(); await workspaceApi.addProjectsToWorkspace([vscode.Uri.file(projectFilePath)]); workspaceApi.showProjectsView(); } public async promptForAutorestProjectName(defaultName?: string): Promise { let name: string | undefined = await vscode.window.showInputBox({ ignoreFocusOut: true, prompt: constants.autorestProjectName, value: defaultName, validateInput: (value) => { return utils.isValidBasenameErrorMessage(value); } }); if (name === undefined) { return; // cancelled by user } name = name.trim(); return name; } /** * Prompts the user with vscode quickpicks to select an OpenApi or Swagger spec to generate sql project from * @param options optional options to pass in instead of using quickpicks to prompt * @returns created sql project */ public async generateProjectFromOpenApiSpec(options?: GenerateProjectFromOpenApiSpecOptions): Promise { try { TelemetryReporter.sendActionEvent(TelemetryViews.ProjectController, TelemetryActions.generateProjectFromOpenApiSpec); // 1. select spec file const specPath: string | undefined = options?.openApiSpecFile?.fsPath || await this.selectAutorestSpecFile(); if (!specPath) { return; } // 2. prompt for project name const projectName = await this.promptForAutorestProjectName(options?.defaultProjectName || path.basename(specPath, path.extname(specPath))); if (!projectName) { return; } // 3. select location, make new folder const projectInfo = await this.selectAutorestProjectLocation(projectName!, options?.defaultOutputLocation); if (!projectInfo) { return; } // 4. run AutoRest to generate .sql files const result = await this.generateAutorestFiles(specPath, projectInfo.newProjectFolder); if (!result) { // user canceled operation when choosing how to run autorest return; } const scriptList: vscode.Uri[] | undefined = await this.getSqlFileList(projectInfo.newProjectFolder); if (!scriptList || scriptList.length === 0) { void vscode.window.showInformationMessage(constants.noSqlFilesGenerated); this._outputChannel.show(); return; } // 5. create new SQL project const newProjFilePath = await this.createNewProject({ newProjName: projectInfo.projectName, folderUri: vscode.Uri.file(projectInfo.outputFolder), projectTypeId: constants.emptySqlDatabaseProjectTypeId, sdkStyle: !!options?.isSDKStyle }); const project = await Project.openProject(newProjFilePath); // 6. add generated files to SQL project const uriList = scriptList.filter(f => !f.fsPath.endsWith(constants.autorestPostDeploymentScriptName)) const relativePaths = uriList.map(f => path.relative(project.projectFolderPath, f.fsPath)); await project.addSqlObjectScripts(relativePaths); // Add generated file structure to the project const postDeploymentScript: vscode.Uri | undefined = this.findPostDeploymentScript(scriptList); if (postDeploymentScript) { await project.addPostDeploymentScript(path.relative(project.projectFolderPath, postDeploymentScript.fsPath)); } if (options?.doNotOpenInWorkspace !== true) { // 7. add project to workspace and open await this.openProjectInWorkspace(newProjFilePath); } return project; } catch (err) { void vscode.window.showErrorMessage(constants.generatingProjectFailed(utils.getErrorMessage(err))); TelemetryReporter.sendErrorEvent2(TelemetryViews.ProjectController, TelemetryActions.generateProjectFromOpenApiSpec, err); this._outputChannel.show(); return; } } private findPostDeploymentScript(files: vscode.Uri[]): vscode.Uri | undefined { // Locate the post-deployment script generated by autorest, if one exists. // It's only generated if enums are present in spec, b/c the enum values need to be inserted into the generated table. // Because autorest is executed via command rather than API, we can't easily "receive" the name of the script, // so we're stuck just matching on a file name. const results = files.filter(f => f.fsPath.endsWith(constants.autorestPostDeploymentScriptName)); switch (results.length) { case 0: return undefined; case 1: return results[0]; default: throw new Error(constants.multipleMostDeploymentScripts(results.length)); } } private async getSqlFileList(folder: string): Promise { if (!(await utils.exists(folder))) { return undefined; } const entries = await fs.readdir(folder, { withFileTypes: true }); const folders = entries.filter(dir => dir.isDirectory()).map(dir => path.join(folder, dir.name)); const files = entries.filter(file => !file.isDirectory() && path.extname(file.name) === constants.sqlFileExtension).map(file => vscode.Uri.file(path.join(folder, file.name))); for (const folder of folders) { files.push(...(await this.getSqlFileList(folder) ?? [])); } return files; } //#endregion //#region Helper methods private async getProjectFromContext(context: Project | BaseProjectTreeItem | dataworkspace.WorkspaceTreeItem): Promise { if ('element' in context) { context = context.element; } if (context instanceof Project) { return context; } if (context instanceof BaseProjectTreeItem) { return Project.openProject(context.projectFileUri.fsPath); } else { throw new Error(constants.unexpectedProjectContext(JSON.stringify(context))); } } private getRelativePath(rootProjectUri: vscode.Uri, treeNode: BaseProjectTreeItem): string { return treeNode instanceof FolderNode ? utils.trimUri(rootProjectUri, treeNode.relativeProjectUri) : ''; } private getConnectionProfileFromContext(context: azdataType.IConnectionProfile | mssqlVscode.ITreeNodeInfo | undefined): azdataType.IConnectionProfile | mssqlVscode.IConnectionInfo | undefined { if (!context) { return undefined; } // depending on where import new project is launched from, the connection profile could be passed as just // the profile or it could be wrapped in another object return (context)?.connectionProfile ?? (context as mssqlVscode.ITreeNodeInfo).connectionInfo ?? context; } private refreshProjectsTree(workspaceTreeItem: dataworkspace.WorkspaceTreeItem): void { (workspaceTreeItem.treeDataProvider as SqlDatabaseProjectTreeViewProvider).notifyTreeDataChanged(); } //#endregion //#region Create project from database /** * Creates a new SQL database project from the existing database, * prompting the user for a name, file path location and extract target */ public async createProjectFromDatabase(context: azdataType.IConnectionProfile | mssqlVscode.ITreeNodeInfo | undefined): Promise { const profile = this.getConnectionProfileFromContext(context); if (utils.getAzdataApi()) { let createProjectFromDatabaseDialog = this.getCreateProjectFromDatabaseDialog(profile as azdataType.IConnectionProfile); createProjectFromDatabaseDialog.createProjectFromDatabaseCallback = async (model, connectionId) => await this.createProjectFromDatabaseCallback(model, connectionId); await createProjectFromDatabaseDialog.openDialog(); return createProjectFromDatabaseDialog; } else { if (context) { // The profile we get from VS Code is for the overall server connection and isn't updated based on the database node // the command was launched from like it is in ADS. So get the actual database name from the MSSQL extension and // update the connection info here. const treeNodeContext = context as mssqlVscode.ITreeNodeInfo; const databaseName = (await utils.getVscodeMssqlApi()).getDatabaseNameFromTreeNode(treeNodeContext); (profile as mssqlVscode.IConnectionInfo).database = databaseName; } const model = await createNewProjectFromDatabaseWithQuickpick(profile as mssqlVscode.IConnectionInfo); if (model) { await this.createProjectFromDatabaseCallback(model, profile as mssqlVscode.IConnectionInfo); } return undefined; } } public getCreateProjectFromDatabaseDialog(profile: azdataType.IConnectionProfile | undefined): CreateProjectFromDatabaseDialog { return new CreateProjectFromDatabaseDialog(profile); } public async createProjectFromDatabaseCallback(model: ImportDataModel, connectionInfo?: string | mssqlVscode.IConnectionInfo) { try { const newProjFolderUri = model.filePath; let targetPlatform: SqlTargetPlatform | undefined; let serverInfo; if (connectionInfo) { if (typeof connectionInfo === 'string') { serverInfo = await utils.getAzdataApi()!.connection.getServerInfo(connectionInfo); } else { serverInfo = (await utils.getVscodeMssqlApi()).getServerInfo(connectionInfo); } } if (serverInfo) { targetPlatform = await utils.getTargetPlatformFromServerVersion(serverInfo); } const newProjFilePath = await this.createNewProject({ newProjName: model.projName, folderUri: vscode.Uri.file(newProjFolderUri), projectTypeId: model.sdkStyle ? constants.emptySqlDatabaseSdkProjectTypeId : constants.emptySqlDatabaseProjectTypeId, sdkStyle: model.sdkStyle, targetPlatform: targetPlatform }); model.filePath = path.dirname(newProjFilePath); this.setFilePath(model); const project = await Project.openProject(newProjFilePath); const startTime = new Date(); await this.createProjectFromDatabaseApiCall(model); // Call ExtractAPI in DacFx Service const timeToExtract = new Date().getTime() - startTime.getTime(); TelemetryReporter.createActionEvent(TelemetryViews.ProjectController, TelemetryActions.createProjectFromDatabase) .withAdditionalMeasurements({ durationMs: timeToExtract }) .send(); const scriptList: vscode.Uri[] = model.extractTarget === mssql.ExtractTarget.file ? [vscode.Uri.file(model.filePath)] : await this.generateScriptList(model.filePath); // Create a list of all the files to be added to project const relativePaths = scriptList.map(f => path.relative(project.projectFolderPath, f.fsPath)); if (!model.sdkStyle) { await project.addSqlObjectScripts(relativePaths); // Add generated file structure to the project } // add project to workspace const workspaceApi = utils.getDataWorkspaceExtensionApi(); workspaceApi.showProjectsView(); await workspaceApi.addProjectsToWorkspace([vscode.Uri.file(newProjFilePath)]); } catch (err) { void vscode.window.showErrorMessage(utils.getErrorMessage(err)); TelemetryReporter.sendErrorEvent2(TelemetryViews.ProjectController, TelemetryActions.createProjectFromDatabase, err); } } public async createProjectFromDatabaseApiCall(model: ImportDataModel): Promise { const service = await utils.getDacFxService(); const azdataApi = utils.getAzdataApi(); if (azdataApi) { await (service as mssql.IDacFxService).createProjectFromDatabase(model.database, model.filePath, model.projName, model.version, model.connectionUri, model.extractTarget as mssql.ExtractTarget, azdataApi.TaskExecutionMode.execute, model.includePermissions); } else { await (service as mssqlVscode.IDacFxService).createProjectFromDatabase(model.database, model.filePath, model.projName, model.version, model.connectionUri, model.extractTarget as mssqlVscode.ExtractTarget, TaskExecutionMode.execute as unknown as mssqlVscode.TaskExecutionMode, model.includePermissions); } // TODO: Check for success; throw error } public setFilePath(model: ImportDataModel) { if (model.extractTarget === mssql.ExtractTarget.file) { model.filePath = path.join(model.filePath, `${model.projName}.sql`); // File extractTarget specifies the exact file rather than the containing folder } } /** * Generate a flat list of all scripts under a folder. * @param absolutePath absolute path to folder to generate the list of files from * @returns array of uris of files under the provided folder */ public async generateScriptList(absolutePath: string): Promise { let fileList: vscode.Uri[] = []; if (!await utils.exists(absolutePath)) { if (await utils.exists(absolutePath + constants.sqlFileExtension)) { absolutePath += constants.sqlFileExtension; } else { void vscode.window.showErrorMessage(constants.cannotResolvePath(absolutePath)); return fileList; } } const files = [absolutePath]; do { const filepath = files.pop(); if (filepath) { const stat = await fs.stat(filepath); if (stat.isDirectory()) { (await fs .readdir(filepath)) .forEach((f: string) => files.push(path.join(filepath, f))); } else if (stat.isFile() && path.extname(filepath) === constants.sqlFileExtension) { fileList.push(vscode.Uri.file(filepath)); } } } while (files.length !== 0); return fileList; } //#endregion //#region Update project from database /** * Display dialog for user to configure existing SQL Project with the changes/differences from a database */ public async updateProjectFromDatabase(context: azdataType.IConnectionProfile | mssqlVscode.ITreeNodeInfo | dataworkspace.WorkspaceTreeItem): Promise { let connection: azdataType.IConnectionProfile | mssqlVscode.IConnectionInfo | undefined; let project: Project | undefined; try { if ('connectionProfile' in context) { connection = this.getConnectionProfileFromContext(context as azdataType.IConnectionProfile | mssqlVscode.ITreeNodeInfo); } } catch { } try { if ('treeDataProvider' in context) { project = await this.getProjectFromContext(context as dataworkspace.WorkspaceTreeItem); } } catch { } const workspaceProjects = await utils.getSqlProjectsInWorkspace(); const updateProjectFromDatabaseDialog = this.getUpdateProjectFromDatabaseDialog(connection, project, workspaceProjects); updateProjectFromDatabaseDialog.updateProjectFromDatabaseCallback = async (model) => await this.updateProjectFromDatabaseCallback(model); await updateProjectFromDatabaseDialog.openDialog(); return updateProjectFromDatabaseDialog; } public getUpdateProjectFromDatabaseDialog(connection: azdataType.IConnectionProfile | mssqlVscode.IConnectionInfo | undefined, project: Project | undefined, workspaceProjects: vscode.Uri[]): UpdateProjectFromDatabaseDialog { return new UpdateProjectFromDatabaseDialog(connection, project, workspaceProjects); } public async updateProjectFromDatabaseCallback(model: UpdateProjectDataModel) { try { const startTime = new Date(); await this.updateProjectFromDatabaseApiCall(model); const timeToUpdate = new Date().getTime() - startTime.getTime(); TelemetryReporter.createActionEvent(TelemetryViews.ProjectController, TelemetryActions.updateProjectFromDatabase) .withAdditionalMeasurements({ durationMs: timeToUpdate }) .send(); } catch (err) { void vscode.window.showErrorMessage(utils.getErrorMessage(err)); TelemetryReporter.sendErrorEvent2(TelemetryViews.ProjectController, TelemetryActions.updateProjectFromDatabase, err); } } /** * Uses the DacFx service to update an existing SQL Project with the changes/differences from a database */ public async updateProjectFromDatabaseApiCall(model: UpdateProjectDataModel): Promise { if (model.action === UpdateProjectAction.Compare) { await vscode.commands.executeCommand(constants.schemaCompareRunComparisonCommand, model.sourceEndpointInfo, model.targetEndpointInfo, true, undefined); } else if (model.action === UpdateProjectAction.Update) { await vscode.window.showWarningMessage(constants.applyConfirmation, { modal: true }, constants.yesString).then(async (result) => { if (result === constants.yesString) { await vscode.window.withProgress( { location: vscode.ProgressLocation.Notification, title: constants.updatingProjectFromDatabase(path.basename(model.targetEndpointInfo.projectFilePath), model.sourceEndpointInfo.databaseName), cancellable: false }, async (_progress, _token) => { return this.schemaCompareAndUpdateProject(model.sourceEndpointInfo, model.targetEndpointInfo); }); void vscode.commands.executeCommand(constants.refreshDataWorkspaceCommand); utils.getDataWorkspaceExtensionApi().showProjectsView(); } }); } else { throw new Error(`Unknown UpdateProjectAction: ${model.action}`); } return; } /** * Performs a schema compare of the source and target and updates the project with the results * @param source source for schema comparison * @param target target sql project for schema comparison to update */ private async schemaCompareAndUpdateProject(source: mssql.SchemaCompareEndpointInfo, target: mssql.SchemaCompareEndpointInfo): Promise { // Run schema comparison const ext = vscode.extensions.getExtension(mssql.extension.name)!; const service = (await ext.activate() as mssql.IExtension).schemaCompare; const deploymentOptions = await service.schemaCompareGetDefaultOptions(); const operationId = UUID.generateUuid(); target.targetScripts = await this.getProjectScriptFiles(target.projectFilePath); target.dataSchemaProvider = await this.getProjectDatabaseSchemaProvider(target.projectFilePath); TelemetryReporter.sendActionEvent(TelemetryViews.ProjectController, TelemetryActions.SchemaComparisonStarted); // Perform schema comparison. Results are cached in SqlToolsService under the operationId const comparisonResult: mssql.SchemaCompareResult = await service.schemaCompare( operationId, source, target, utils.getAzdataApi()!.TaskExecutionMode.execute, deploymentOptions.defaultDeploymentOptions ); if (!comparisonResult || !comparisonResult.success) { TelemetryReporter.createErrorEvent2(TelemetryViews.ProjectController, 'SchemaComparisonFailed') .withAdditionalProperties({ operationId: comparisonResult.operationId }).send(); await vscode.window.showErrorMessage(constants.compareErrorMessage(comparisonResult?.errorMessage)); return; } TelemetryReporter.createActionEvent(TelemetryViews.ProjectController, TelemetryActions.SchemaComparisonFinished) .withAdditionalProperties({ 'endTime': Date.now().toString(), 'operationId': comparisonResult.operationId }).send(); if (comparisonResult.areEqual) { void vscode.window.showInformationMessage(constants.equalComparison); return; } // Publish the changes (retrieved from the cache by operationId) const publishResult = await this.schemaComparePublishProjectChanges(operationId, target.projectFilePath, target.extractTarget); if (publishResult.success) { void vscode.window.showInformationMessage(constants.applySuccess); } else { void vscode.window.showErrorMessage(constants.applyError(publishResult.errorMessage)); } } public async getProjectScriptFiles(projectFilePath: string): Promise { const project = await Project.openProject(projectFilePath); return project.sqlObjectScripts .filter(f => f.fsUri.fsPath.endsWith(constants.sqlFileExtension)) .map(f => f.fsUri.fsPath); } public async getProjectDatabaseSchemaProvider(projectFilePath: string): Promise { const project = await Project.openProject(projectFilePath); return project.getProjectTargetVersion(); } /** * Updates the provided project with the results of the schema compare * @param operationId id of the schema comparison to update the project with * @param projectFilePath path to sql project to update * @param folderStructure folder structure to use when updating the target project * @returns */ public async schemaComparePublishProjectChanges(operationId: string, projectFilePath: string, folderStructure: mssql.ExtractTarget): Promise { const ext = vscode.extensions.getExtension(mssql.extension.name)!; const service = (await ext.activate() as mssql.IExtension).schemaCompare; const projectPath = path.dirname(projectFilePath); const result: mssql.SchemaComparePublishProjectResult = await service.schemaComparePublishProjectChanges(operationId, projectPath, folderStructure, utils.getAzdataApi()!.TaskExecutionMode.execute); if (!result.errorMessage) { const project = await Project.openProject(projectFilePath); let toAdd: vscode.Uri[] = []; result.addedFiles.forEach((f: any) => toAdd.push(vscode.Uri.file(f))); const relativePaths = toAdd.map(f => path.relative(project.projectFolderPath, f.fsPath)); await project.addSqlObjectScripts(relativePaths); let toRemove: vscode.Uri[] = []; result.deletedFiles.forEach((f: any) => toRemove.push(vscode.Uri.file(f))); let toRemoveEntries: FileProjectEntry[] = []; toRemove.forEach(f => toRemoveEntries.push(new FileProjectEntry(f, f.fsPath.replace(projectPath + '\\', ''), EntryType.File))); toRemoveEntries.forEach(async f => await project.excludeSqlObjectScript(f.fsUri.fsPath)); await this.buildProject(project); } return result; } //#endregion /** * Move a file or folder in the project tree * @param projectUri URI of the project * @param source * @param target */ public async moveFile(projectUri: vscode.Uri, source: any, target: dataworkspace.WorkspaceTreeItem): Promise { const sourceFileNode = source as FileNode | FolderNode; const project = await this.getProjectFromContext(sourceFileNode); // only moving files and folders are supported if (!sourceFileNode || !(sourceFileNode instanceof FileNode || sourceFileNode instanceof FolderNode)) { void vscode.window.showErrorMessage(constants.onlyMoveFilesFoldersSupported); return; } // Moving files/folders to the SQLCMD variables and Database references folders isn't allowed if (!target.element.fileSystemUri) { return; } // TODO: handle moving between different projects if (projectUri.fsPath !== target.element.projectFileUri.fsPath) { void vscode.window.showErrorMessage(constants.movingFilesBetweenProjectsNotSupported); return; } // Calculate the new file path let folderPath; // target is the root of project, which is the .sqlproj if (target.element.projectFileUri.fsPath === target.element.fileSystemUri.fsPath) { folderPath = path.basename(path.dirname(target.element.projectFileUri.fsPath!)); } else { // target is another file or folder folderPath = target.element.relativeProjectUri.fsPath.endsWith(constants.sqlFileExtension) ? path.dirname(target.element.relativeProjectUri.fsPath) : target.element.relativeProjectUri.fsPath; } const newPath = path.join(folderPath!, sourceFileNode.friendlyName); // don't do anything if the path is the same if (newPath === sourceFileNode.relativeProjectUri.fsPath) { return; } const result = await vscode.window.showWarningMessage(constants.moveConfirmationPrompt(path.basename(sourceFileNode.fileSystemUri.fsPath), path.basename(folderPath)), { modal: true }, constants.move) if (result !== constants.move) { return; } // Move the file/folder const moveResult = await project.move(sourceFileNode, newPath); if (moveResult?.success) { TelemetryReporter.sendActionEvent(TelemetryViews.ProjectTree, TelemetryActions.move); } else { TelemetryReporter.sendErrorEvent2(TelemetryViews.ProjectTree, TelemetryActions.move); void vscode.window.showErrorMessage(constants.errorMovingFile(sourceFileNode.fileSystemUri.fsPath, newPath, utils.getErrorMessage(moveResult?.errorMessage))); } } } export interface NewProjectParams { newProjName: string; folderUri: vscode.Uri; projectTypeId: string; sdkStyle: boolean; projectGuid?: string; targetPlatform?: SqlTargetPlatform; }