Skip to content

Commit

Permalink
Merge pull request #317 from couchbaselabs/DA#242-DDL-Export
Browse files Browse the repository at this point in the history
Da#242 ddl export
  • Loading branch information
AayushTyagi1 authored Oct 19, 2023
2 parents 52cc29a + 9e4cd37 commit 3e82869
Show file tree
Hide file tree
Showing 7 changed files with 594 additions and 8 deletions.
19 changes: 18 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -578,14 +578,19 @@
},
{
"title": "Refresh Cluster Overview",
"command":"vscode-couchbase.refreshClusterOverview",
"command": "vscode-couchbase.refreshClusterOverview",
"icon": "$(refresh)",
"category": "Couchbase"
},
{
"title": "Check and Create Primary Index",
"command": "vscode-couchbase.checkAndCreatePrimaryIndex",
"category": "Couchbase"
},
{
"command": "vscode-couchbase.tools.DDLExport",
"title": "DDL Export",
"category": "Couchbase"
}
],
"keybindings": [
Expand Down Expand Up @@ -755,6 +760,10 @@
{
"command": "vscode-couchbase.tools.dataExport",
"when": "false"
},
{
"command": "vscode-couchbase.tools.DDLExport",
"when": "false"
}
],
"view/item/context": [
Expand Down Expand Up @@ -791,6 +800,10 @@
"submenu": "vscode-couchbase.toolsMenu",
"when": "view == couchbase && viewItem == active_connection"
},
{
"submenu": "vscode-couchbase.toolsMenu",
"when": "view == couchbase && viewItem == active_connection"
},
{
"command": "vscode-couchbase.useClusterConnection",
"when": "view == couchbase && viewItem == connection"
Expand Down Expand Up @@ -904,6 +917,10 @@
{
"command": "vscode-couchbase.tools.dataExport",
"group": "navigation"
},
{
"command": "vscode-couchbase.tools.DDLExport",
"group": "navigation"
}
]
},
Expand Down
1 change: 1 addition & 0 deletions src/commands/extensionCommands/commands.ts
Original file line number Diff line number Diff line change
Expand Up @@ -55,4 +55,5 @@ export namespace Commands {
export const refreshClusterOverview: string = "vscode-couchbase.refreshClusterOverview";
export const checkAndCreatePrimaryIndex: string = "vscode-couchbase.checkAndCreatePrimaryIndex";
export const dataExport: string = "vscode-couchbase.tools.dataExport";
export const ddlExport: string = "vscode-couchbase.tools.DDLExport";
}
218 changes: 218 additions & 0 deletions src/commands/tools/ddlExport/ddlExport.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,218 @@
import * as vscode from "vscode";
import * as path from 'path';
import { IConnection } from "../../../types/IConnection";
import { getActiveConnection } from "../../../util/connections";
import { logger } from "../../../logger/logger";
import { getLoader } from "../../../webViews/loader.webview";
import { ddlExportWebview } from "../../../webViews/tools/ddlExport.webview";
import { QueryIndex, ScopeSpec } from "couchbase";
import { getCurrentDateTime } from "../../../util/util";
import * as fs from "fs";
import { getIndexDefinition } from "../../../util/indexUtils";

const validateFormData = (formData: any): string => {
const errors = [];

if (!formData.bucket) {
errors.push("Please select a bucket");
} else {
if (formData.scopes.length === 0) {
errors.push("Select one or more scopes");
}
}

if (!formData.fileDestination.trim()) {
errors.push("Please inform the file destination folder");
}

if (errors.length > 0) {
return errors.join("<br>");
}

return "";
};

class DDLExport {
private static async exportScope(bucket: string, scope: ScopeSpec, includeIndexes: boolean): Promise<string> {
const ddlExportData: string[] = [];
const scopePrefix = "`" + bucket + "`.`" + scope.name + "`";

if ("_default" !== scope.name) {
ddlExportData.push("CREATE SCOPE " + scopePrefix + ";");
}
ddlExportData.push("\n");
const connection = getActiveConnection();
if (!connection) {
return "";
}

for (const collectionSpec of scope.collections) {
ddlExportData.push("\n");
ddlExportData.push("/* DDL for collection " + scope.name + "." + collectionSpec.name + " */");
ddlExportData.push("\n");
if ("_default" !== collectionSpec.name) {
ddlExportData.push("CREATE COLLECTION " + scopePrefix + ".`" + collectionSpec.name + "`; \n");
}
if (includeIndexes) {
const result = await listIndexes(connection, bucket, scope.name, collectionSpec.name);
ddlExportData.push(result?.map(index => getIndexDefinition(index)).join("; \n")!);
if (result!.length > 1) {
ddlExportData.push("; \n");
}
}
ddlExportData.push("\n");
}

ddlExportData.push("\n\n");

return ddlExportData.join("");
}

public async exportScope(bucket: string, scopes: string[], filePath: string, includeIndexes: boolean): Promise<void> {
try {
const connection = getActiveConnection();
if (!connection) {
return;
}

const ifAllScopes = scopes.includes("All Scopes");
const ddlExportData: string[] = [];
const allScopes = await connection?.cluster?.bucket(bucket).collections().getAllScopes()!;
const selectedScopes: ScopeSpec[] = ifAllScopes ? allScopes : allScopes.filter(a => scopes.includes(a.name));

for (const scope of selectedScopes) {
ddlExportData.push("/*************************/ \n");
ddlExportData.push("/** Scope " + scope.name + " */ \n");
ddlExportData.push("/*************************/ \n");
ddlExportData.push(await DDLExport.exportScope(bucket, scope, includeIndexes));
ddlExportData.push("\n");
}

let fileName = bucket + "_" + (ifAllScopes ? "_all_" : DDLExport.getScopesFileName(scopes)) + getCurrentDateTime() + ".sqlpp";
if (!filePath.endsWith("/")) {
fileName = "/" + fileName;
}
const fullPath = filePath + fileName;
fs.writeFile(fullPath, ddlExportData.join(''), function (err) {
if (err) { throw err; }
vscode.window.showInformationMessage("DDL Export: File exported successfully");
});

} catch (e) {
logger.error("An error occurred while writing to export the DDL: " + e);
}
};

public static getScopesFileName(scopes: string[]): string {
if (scopes.length === 1) {
return scopes[0] + "_";
} else {
return scopes.length + "_scopes_";
}
}
};

export const listIndexes = async (connection: IConnection, bucket: string, scope: string, collection: string): Promise<QueryIndex[] | undefined> => {
if (connection) {
try {
const result = await connection?.cluster?.queryIndexes().getAllIndexes(bucket, { scopeName: scope, collectionName: collection });
return result;
} catch {
return [];
}
}

};

export const ddlExport = async () => {
const connection = getActiveConnection();
if (!connection) {
return;
}
const currentPanel = vscode.window.createWebviewPanel(
"ddlExport",
"DDL Export",
vscode.ViewColumn.One,
{
enableScripts: true,
enableForms: true,
}
);
currentPanel.iconPath = {
dark: vscode.Uri.file(
path.join(__filename, "..", "..", "images", "dark", "export_dark.svg")
),
light: vscode.Uri.file(
path.join(__filename, "..", "..", "images", "light", "export_light.svg")
),
};
currentPanel.webview.html = getLoader("DDL Export");

// Get all buckets
const buckets = await connection.cluster?.buckets().getAllBuckets();
if (buckets === undefined) {
vscode.window.showErrorMessage("Buckets not found");
return;
}

const bucketNameArr: string[] = buckets.map(b => b.name);
try {
currentPanel.webview.html = await ddlExportWebview(bucketNameArr);
currentPanel.webview.onDidReceiveMessage(async (message) => {
switch (message.command) {
case "vscode-couchbase.tools.DDLExport.runExport":
const formData = message.data;
const validationError = validateFormData(formData);
if (validationError === "") {
const ddl = new DDLExport();
ddl.exportScope(formData.bucket,
formData.scopes, formData.fileDestination, formData.includeIndexes);
} else {
currentPanel.webview.postMessage({
command: "vscode-couchbase.tools.DDLExport.formValidationError",
error: validationError,
});
}
break;
case "vscode-couchbase.tools.DDLExport.getScopes":
const scopes = await connection.cluster
?.bucket(message.bucketId)
.collections()
.getAllScopes();
if (scopes === undefined) {
vscode.window.showErrorMessage("Scopes are undefined");
break;
}

currentPanel.webview.postMessage({
command: "vscode-couchbase.tools.DDLExport.scopesInfo",
scopes: scopes,
});
break;
case "vscode-couchbase.tools.DDLExport.getFolder":
const options: vscode.OpenDialogOptions = {
canSelectMany: false,
openLabel: "Choose Destination Folder",
canSelectFiles: false,
canSelectFolders: true,
};

vscode.window.showOpenDialog(options).then((fileUri) => {
if (fileUri && fileUri[0]) {
currentPanel.webview.postMessage({
command: "vscode-couchbase.tools.DDLExport.folderInfo",
folder: fileUri[0].fsPath,
});
}
});
break;
}
});
} catch (err) {
logger.error(`Failed to open data export webview`);
logger.debug(err);
vscode.window.showErrorMessage(
"Failed to open data export webview: " + err
);
}
};
7 changes: 7 additions & 0 deletions src/extension.ts
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,7 @@ import { clearDocumentFilter } from "./commands/documents/clearDocumentFilter";
import { getClusterOverviewData } from "./util/OverviewClusterUtils/getOverviewClusterData";
import { checkAndCreatePrimaryIndex } from "./commands/indexes/checkAndCreatePrimaryIndex";
import { dataExport } from "./pages/Tools/DataExport/dataExport";
import { ddlExport } from "./commands/tools/ddlExport/ddlExport";

export function activate(context: vscode.ExtensionContext) {
Global.setState(context.globalState);
Expand Down Expand Up @@ -573,6 +574,12 @@ export function activate(context: vscode.ExtensionContext) {
}
)
);

context.subscriptions.push(
vscode.commands.registerCommand(Commands.ddlExport, async () => {
ddlExport();
})
);
}

// this method is called when your extension is deactivated
Expand Down
12 changes: 5 additions & 7 deletions src/pages/Tools/DataExport/dataExport.ts
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ const validateFormData = (formData: any): string => {
errors.push("Please inform the file destination folder");
}

if(!formData.threads.trim() || parseInt(formData.threads)<1){
if (!formData.threads.trim() || parseInt(formData.threads) < 1) {
errors.push("threads cannot be undefined or less than 1");
}

Expand Down Expand Up @@ -109,10 +109,10 @@ export const dataExport = async () => {
});
currentPanel.iconPath = {
dark: vscode.Uri.file(
path.join(__filename, "..", "..", "images","dark","export_dark.svg")
path.join(__filename, "..", "..", "images", "dark", "export_dark.svg")
),
light: vscode.Uri.file(
path.join(__filename, "..", "..", "images","light","export_light.svg")
path.join(__filename, "..", "..", "images", "light", "export_light.svg")
),
};
currentPanel.webview.html = getLoader("Data Export");
Expand All @@ -123,10 +123,8 @@ export const dataExport = async () => {
vscode.window.showErrorMessage("Buckets not found");
return;
}
const bucketNameArr: string[] = [];
for (let bucket of buckets) {
bucketNameArr.push(bucket.name);
}

const bucketNameArr: string[] = buckets.map(b => b.name);

try {
currentPanel.webview.html = await dataExportWebview(bucketNameArr);
Expand Down
25 changes: 25 additions & 0 deletions src/webViews/tools/dataExport.webview.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,31 @@ export const dataExportWebview = async (buckets: string[]): Promise<string> => {
margin-bottom: 5px;
}
input[type="checkbox"] {
appearance: none;
-webkit-appearance: none;
width: 15px;
height: 15px;
border-radius: 2px;
outline: none;
border: 1px solid #999;
background-color: white;
}
input[type="checkbox"]:checked {
background-color: #ea2328;
}
input[type="checkbox"]:checked::before {
content: '\\2714';
display: flex;
justify-content: center;
align-items: center;
font-size: 12px;
color: white;
height: 100%;
}
form {
max-width: 500px;
margin: 0 auto;
Expand Down
Loading

0 comments on commit 3e82869

Please sign in to comment.