Files
azuredatastudio/extensions/sql-database-projects/src/tools/buildHelper.ts
2023-04-05 13:01:51 -07:00

147 lines
5.7 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 path from 'path';
import { promises as fs } from 'fs';
import * as utils from '../common/utils';
import * as sqldbproj from 'sqldbproj';
import * as extractZip from 'extract-zip';
import * as constants from '../common/constants';
import { HttpClient } from '../common/httpClient';
import { DBProjectConfigurationKey } from './netcoreTool';
import { ProjectType } from 'mssql';
const buildDirectory = 'BuildDirectory';
const sdkName = 'Microsoft.Build.Sql';
const microsoftBuildSqlDefaultVersion = '0.1.9-preview'; // default version of Microsoft.Build.Sql nuget to use for building legacy style projects, update in README when updating this
const buildFiles: string[] = [
'Microsoft.Data.SqlClient.dll',
'Microsoft.Data.Tools.Schema.Sql.dll',
'Microsoft.Data.Tools.Schema.Tasks.Sql.dll',
'Microsoft.Data.Tools.Utilities.dll',
'Microsoft.SqlServer.Dac.dll',
'Microsoft.SqlServer.Dac.Extensions.dll',
'Microsoft.SqlServer.TransactSql.ScriptDom.dll',
'Microsoft.SqlServer.Types.dll',
'System.ComponentModel.Composition.dll',
'System.IO.Packaging.dll',
'Microsoft.Data.Tools.Schema.SqlTasks.targets',
'Microsoft.SqlServer.Server.dll'
];
export class BuildHelper {
private extensionDir: string;
private extensionBuildDir: string;
private initialized: boolean = false;
constructor() {
const extName = utils.getAzdataApi() ? sqldbproj.extension.name : sqldbproj.extension.vsCodeName;
this.extensionDir = vscode.extensions.getExtension(extName)?.extensionPath ?? '';
this.extensionBuildDir = path.join(this.extensionDir, buildDirectory);
}
/**
* Create build dlls directory with the dlls and targets needed for building a sqlproj
* @param outputChannel
*/
public async createBuildDirFolder(outputChannel: vscode.OutputChannel): Promise<boolean> {
if (this.initialized) {
return true;
}
if (!await utils.exists(this.extensionBuildDir)) {
await fs.mkdir(this.extensionBuildDir);
}
// check if the settings has a version specified for Microsoft.Build.Sql, otherwise use default
const microsoftBuildSqlVersionConfig = vscode.workspace.getConfiguration(DBProjectConfigurationKey)[constants.microsoftBuildSqlVersionKey];
const sdkVersion = !!microsoftBuildSqlVersionConfig ? microsoftBuildSqlVersionConfig : microsoftBuildSqlDefaultVersion;
const fullSdkName = `${sdkName}.${sdkVersion}`;
// check if this if the nuget needs to be downloaded
const nugetPath = path.join(this.extensionBuildDir, `${fullSdkName}.nupkg`);
if (await utils.exists(nugetPath)) {
// if it does exist, make sure all the necessary files are also in the BuildDirectory
let missingFiles = false;
for (const fileName of buildFiles) {
if (!await (utils.exists(path.join(this.extensionBuildDir, fileName)))) {
missingFiles = true;
break;
}
}
// if all the files are there, no need to continue
if (!missingFiles) {
return true;
}
}
// TODO: check nuget cache locations first to avoid downloading if those exist
// download the Microsoft.Build.Sql sdk nuget
outputChannel.appendLine(constants.downloadingDacFxDlls);
const microsoftBuildSqlUrl = `https://www.nuget.org/api/v2/package/${sdkName}/${sdkVersion}`;
try {
const httpClient = new HttpClient();
outputChannel.appendLine(constants.downloadingFromTo(microsoftBuildSqlUrl, nugetPath));
await httpClient.download(microsoftBuildSqlUrl, nugetPath, outputChannel);
} catch (e) {
void vscode.window.showErrorMessage(constants.errorDownloading(microsoftBuildSqlUrl, utils.getErrorMessage(e)));
return false;
}
// extract the files from the nuget
const extractedFolderPath = path.join(this.extensionDir, buildDirectory, sdkName);
outputChannel.appendLine(constants.extractingDacFxDlls(extractedFolderPath));
try {
await extractZip(nugetPath, { dir: extractedFolderPath });
} catch (e) {
void vscode.window.showErrorMessage(constants.errorExtracting(nugetPath, utils.getErrorMessage(e)));
return false;
}
// copy the dlls and targets file to the BuildDirectory folder
const buildfilesPath = path.join(extractedFolderPath, 'tools', 'netstandard2.1');
for (const fileName of buildFiles) {
if (await (utils.exists(path.join(buildfilesPath, fileName)))) {
await fs.copyFile(path.join(buildfilesPath, fileName), path.join(this.extensionBuildDir, fileName));
}
}
// cleanup extracted folder
await fs.rm(extractedFolderPath, { recursive: true });
this.initialized = true;
return true;
}
public get extensionBuildDirPath(): string {
return this.extensionBuildDir;
}
public constructBuildArguments(projectPath: string, buildDirPath: string, sqlProjStyle: ProjectType): string {
projectPath = utils.getQuotedPath(projectPath);
buildDirPath = utils.getQuotedPath(buildDirPath);
// Right now SystemDacpacsLocation and NETCoreTargetsPath get set to the same thing, but separating them out for if we move
// the system dacpacs somewhere else and also so that the variable name makes more sense if building from the commandline,
// since SDK style projects don't to specify the targets path, just where the system dacpacs are
if (sqlProjStyle === ProjectType.SdkStyle) {
return ` build ${projectPath} /p:NetCoreBuild=true /p:SystemDacpacsLocation=${buildDirPath}`;
} else {
return ` build ${projectPath} /p:NetCoreBuild=true /p:NETCoreTargetsPath=${buildDirPath} /p:SystemDacpacsLocation=${buildDirPath}`;
}
}
}