UNPKG

24.7 kBJavaScriptView Raw
1"use strict";
2var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
3 function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
4 return new (P || (P = Promise))(function (resolve, reject) {
5 function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
6 function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
7 function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
8 step((generator = generator.apply(thisArg, _arguments || [])).next());
9 });
10};
11Object.defineProperty(exports, "__esModule", { value: true });
12const convert_1 = require("../convert");
13const languageclient_1 = require("../languageclient");
14const fs_1 = require("fs");
15const rimraf = require("rimraf");
16/** Public: Adapts workspace/applyEdit commands to editors. */
17class ApplyEditAdapter {
18 /** Public: Attach to a {LanguageClientConnection} to receive edit events. */
19 static attach(connection) {
20 connection.onApplyEdit((m) => ApplyEditAdapter.onApplyEdit(m));
21 }
22 /** Tries to apply edits and reverts if anything goes wrong. Returns the checkpoint, so the caller can revert changes if needed. */
23 static applyEdits(buffer, edits) {
24 const checkpoint = buffer.createCheckpoint();
25 try {
26 // Sort edits in reverse order to prevent edit conflicts.
27 edits.sort((edit1, edit2) => -edit1.oldRange.compare(edit2.oldRange));
28 edits.reduce((previous, current) => {
29 validateEdit(buffer, current, previous);
30 buffer.setTextInRange(current.oldRange, current.newText);
31 return current;
32 }, null);
33 buffer.groupChangesSinceCheckpoint(checkpoint);
34 return checkpoint;
35 }
36 catch (err) {
37 buffer.revertToCheckpoint(checkpoint);
38 throw err;
39 }
40 }
41 static onApplyEdit(params) {
42 return __awaiter(this, void 0, void 0, function* () {
43 return ApplyEditAdapter.apply(params.edit);
44 });
45 }
46 static apply(workspaceEdit) {
47 return __awaiter(this, void 0, void 0, function* () {
48 normalize(workspaceEdit);
49 // Keep checkpoints from all successful buffer edits
50 const checkpoints = [];
51 const promises = (workspaceEdit.documentChanges || []).map((edit) => __awaiter(this, void 0, void 0, function* () {
52 if (!languageclient_1.TextDocumentEdit.is(edit)) {
53 return ApplyEditAdapter.handleResourceOperation(edit).catch((err) => {
54 throw Error(`Error during ${edit.kind} resource operation: ${err.message}`);
55 });
56 }
57 const path = convert_1.default.uriToPath(edit.textDocument.uri);
58 const editor = (yield atom.workspace.open(path, {
59 searchAllPanes: true,
60 // Open new editors in the background.
61 activatePane: false,
62 activateItem: false,
63 }));
64 const buffer = editor.getBuffer();
65 const edits = convert_1.default.convertLsTextEdits(edit.edits);
66 const checkpoint = ApplyEditAdapter.applyEdits(buffer, edits);
67 checkpoints.push({ buffer, checkpoint });
68 }));
69 // Apply all edits or fail and revert everything
70 const applied = yield Promise.all(promises)
71 .then(() => true)
72 .catch((err) => {
73 atom.notifications.addError("workspace/applyEdits failed", {
74 description: "Failed to apply edits.",
75 detail: err.message,
76 });
77 checkpoints.forEach(({ buffer, checkpoint }) => {
78 buffer.revertToCheckpoint(checkpoint);
79 });
80 return false;
81 });
82 return { applied };
83 });
84 }
85 static handleResourceOperation(edit) {
86 var _a, _b, _c, _d, _e, _f, _g;
87 return __awaiter(this, void 0, void 0, function* () {
88 if (languageclient_1.DeleteFile.is(edit)) {
89 const path = convert_1.default.uriToPath(edit.uri);
90 const stats = yield fs_1.promises.lstat(path).catch(() => false);
91 const ignoreIfNotExists = (_a = edit.options) === null || _a === void 0 ? void 0 : _a.ignoreIfNotExists;
92 if (!stats) {
93 if (ignoreIfNotExists !== false) {
94 return;
95 }
96 throw Error(`Target doesn't exist.`);
97 }
98 if (stats.isDirectory()) {
99 if ((_b = edit.options) === null || _b === void 0 ? void 0 : _b.recursive) {
100 return new Promise((resolve, reject) => {
101 rimraf(path, { glob: false }, (err) => {
102 if (err) {
103 reject(err);
104 }
105 resolve();
106 });
107 });
108 }
109 return fs_1.promises.rmdir(path, { recursive: (_c = edit.options) === null || _c === void 0 ? void 0 : _c.recursive });
110 }
111 return fs_1.promises.unlink(path);
112 }
113 if (languageclient_1.RenameFile.is(edit)) {
114 const oldPath = convert_1.default.uriToPath(edit.oldUri);
115 const newPath = convert_1.default.uriToPath(edit.newUri);
116 const exists = yield fs_1.promises
117 .access(newPath)
118 .then(() => true)
119 .catch(() => false);
120 const ignoreIfExists = (_d = edit.options) === null || _d === void 0 ? void 0 : _d.ignoreIfExists;
121 const overwrite = (_e = edit.options) === null || _e === void 0 ? void 0 : _e.overwrite;
122 if (exists && ignoreIfExists && !overwrite) {
123 return;
124 }
125 if (exists && !ignoreIfExists && !overwrite) {
126 throw Error(`Target exists.`);
127 }
128 return fs_1.promises.rename(oldPath, newPath);
129 }
130 if (languageclient_1.CreateFile.is(edit)) {
131 const path = convert_1.default.uriToPath(edit.uri);
132 const exists = yield fs_1.promises
133 .access(path)
134 .then(() => true)
135 .catch(() => false);
136 const ignoreIfExists = (_f = edit.options) === null || _f === void 0 ? void 0 : _f.ignoreIfExists;
137 const overwrite = (_g = edit.options) === null || _g === void 0 ? void 0 : _g.overwrite;
138 if (exists && ignoreIfExists && !overwrite) {
139 return;
140 }
141 return fs_1.promises.writeFile(path, "");
142 }
143 });
144 }
145}
146exports.default = ApplyEditAdapter;
147function normalize(workspaceEdit) {
148 const documentChanges = workspaceEdit.documentChanges || [];
149 if (!("documentChanges" in workspaceEdit) && "changes" in workspaceEdit) {
150 Object.keys(workspaceEdit.changes || []).forEach((uri) => {
151 documentChanges.push({
152 textDocument: {
153 version: null,
154 uri,
155 },
156 edits: workspaceEdit.changes[uri],
157 });
158 });
159 }
160 workspaceEdit.documentChanges = documentChanges;
161}
162function validateEdit(buffer, edit, prevEdit) {
163 const path = buffer.getPath() || "";
164 if (prevEdit && edit.oldRange.end.compare(prevEdit.oldRange.start) > 0) {
165 throw Error(`Found overlapping edit ranges in ${path}`);
166 }
167 const startRow = edit.oldRange.start.row;
168 const startCol = edit.oldRange.start.column;
169 const lineLength = buffer.lineLengthForRow(startRow);
170 if (lineLength == null || startCol > lineLength) {
171 throw Error(`Out of range edit on ${path}:${startRow + 1}:${startCol + 1}`);
172 }
173}
174//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"apply-edit-adapter.js","sourceRoot":"","sources":["../../../lib/adapters/apply-edit-adapter.ts"],"names":[],"mappings":"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","sourcesContent":["import type * as atomIde from \"atom-ide-base\"\nimport Convert from \"../convert\"\nimport {\n  LanguageClientConnection,\n  ApplyWorkspaceEditParams,\n  ApplyWorkspaceEditResponse,\n  WorkspaceEdit,\n  TextDocumentEdit,\n  CreateFile,\n  RenameFile,\n  DeleteFile,\n  DocumentUri,\n} from \"../languageclient\"\nimport { TextBuffer, TextEditor } from \"atom\"\nimport { promises as fsp, Stats } from \"fs\"\nimport * as rimraf from \"rimraf\"\n\n/** Public: Adapts workspace/applyEdit commands to editors. */\nexport default class ApplyEditAdapter {\n  /** Public: Attach to a {LanguageClientConnection} to receive edit events. */\n  public static attach(connection: LanguageClientConnection): void {\n    connection.onApplyEdit((m) => ApplyEditAdapter.onApplyEdit(m))\n  }\n\n  /** Tries to apply edits and reverts if anything goes wrong. Returns the checkpoint, so the caller can revert changes if needed. */\n  public static applyEdits(buffer: TextBuffer, edits: atomIde.TextEdit[]): number {\n    const checkpoint = buffer.createCheckpoint()\n    try {\n      // Sort edits in reverse order to prevent edit conflicts.\n      edits.sort((edit1, edit2) => -edit1.oldRange.compare(edit2.oldRange))\n      edits.reduce((previous: atomIde.TextEdit | null, current) => {\n        validateEdit(buffer, current, previous)\n        buffer.setTextInRange(current.oldRange, current.newText)\n        return current\n      }, null)\n      buffer.groupChangesSinceCheckpoint(checkpoint)\n      return checkpoint\n    } catch (err) {\n      buffer.revertToCheckpoint(checkpoint)\n      throw err\n    }\n  }\n\n  public static async onApplyEdit(params: ApplyWorkspaceEditParams): Promise<ApplyWorkspaceEditResponse> {\n    return ApplyEditAdapter.apply(params.edit)\n  }\n\n  public static async apply(workspaceEdit: WorkspaceEdit): Promise<ApplyWorkspaceEditResponse> {\n    normalize(workspaceEdit)\n\n    // Keep checkpoints from all successful buffer edits\n    const checkpoints: Array<{ buffer: TextBuffer; checkpoint: number }> = []\n\n    const promises = (workspaceEdit.documentChanges || []).map(async (edit): Promise<void> => {\n      if (!TextDocumentEdit.is(edit)) {\n        return ApplyEditAdapter.handleResourceOperation(edit).catch((err) => {\n          throw Error(`Error during ${edit.kind} resource operation: ${err.message}`)\n        })\n      }\n      const path = Convert.uriToPath(edit.textDocument.uri)\n      const editor = (await atom.workspace.open(path, {\n        searchAllPanes: true,\n        // Open new editors in the background.\n        activatePane: false,\n        activateItem: false,\n      })) as TextEditor\n      const buffer = editor.getBuffer()\n      const edits = Convert.convertLsTextEdits(edit.edits)\n      const checkpoint = ApplyEditAdapter.applyEdits(buffer, edits)\n      checkpoints.push({ buffer, checkpoint })\n    })\n\n    // Apply all edits or fail and revert everything\n    const applied = await Promise.all(promises)\n      .then(() => true)\n      .catch((err) => {\n        atom.notifications.addError(\"workspace/applyEdits failed\", {\n          description: \"Failed to apply edits.\",\n          detail: err.message,\n        })\n        checkpoints.forEach(({ buffer, checkpoint }) => {\n          buffer.revertToCheckpoint(checkpoint)\n        })\n        return false\n      })\n\n    return { applied }\n  }\n\n  private static async handleResourceOperation(edit: CreateFile | RenameFile | DeleteFile): Promise<void> {\n    if (DeleteFile.is(edit)) {\n      const path = Convert.uriToPath(edit.uri)\n      const stats: boolean | Stats = await fsp.lstat(path).catch(() => false)\n      const ignoreIfNotExists = edit.options?.ignoreIfNotExists\n\n      if (!stats) {\n        if (ignoreIfNotExists !== false) {\n          return\n        }\n        throw Error(`Target doesn't exist.`)\n      }\n\n      if (stats.isDirectory()) {\n        if (edit.options?.recursive) {\n          return new Promise((resolve, reject) => {\n            rimraf(path, { glob: false }, (err) => {\n              if (err) {\n                reject(err)\n              }\n              resolve()\n            })\n          })\n        }\n        return fsp.rmdir(path, { recursive: edit.options?.recursive })\n      }\n\n      return fsp.unlink(path)\n    }\n    if (RenameFile.is(edit)) {\n      const oldPath = Convert.uriToPath(edit.oldUri)\n      const newPath = Convert.uriToPath(edit.newUri)\n      const exists = await fsp\n        .access(newPath)\n        .then(() => true)\n        .catch(() => false)\n      const ignoreIfExists = edit.options?.ignoreIfExists\n      const overwrite = edit.options?.overwrite\n\n      if (exists && ignoreIfExists && !overwrite) {\n        return\n      }\n\n      if (exists && !ignoreIfExists && !overwrite) {\n        throw Error(`Target exists.`)\n      }\n\n      return fsp.rename(oldPath, newPath)\n    }\n    if (CreateFile.is(edit)) {\n      const path = Convert.uriToPath(edit.uri)\n      const exists = await fsp\n        .access(path)\n        .then(() => true)\n        .catch(() => false)\n      const ignoreIfExists = edit.options?.ignoreIfExists\n      const overwrite = edit.options?.overwrite\n\n      if (exists && ignoreIfExists && !overwrite) {\n        return\n      }\n\n      return fsp.writeFile(path, \"\")\n    }\n  }\n}\n\nfunction normalize(workspaceEdit: WorkspaceEdit): void {\n  const documentChanges = workspaceEdit.documentChanges || []\n\n  if (!(\"documentChanges\" in workspaceEdit) && \"changes\" in workspaceEdit) {\n    Object.keys(workspaceEdit.changes || []).forEach((uri: DocumentUri) => {\n      documentChanges.push({\n        textDocument: {\n          version: null,\n          uri,\n        },\n        edits: workspaceEdit.changes![uri],\n      })\n    })\n  }\n\n  workspaceEdit.documentChanges = documentChanges\n}\n\nfunction validateEdit(buffer: TextBuffer, edit: atomIde.TextEdit, prevEdit: atomIde.TextEdit | null): void {\n  const path = buffer.getPath() || \"\"\n  if (prevEdit && edit.oldRange.end.compare(prevEdit.oldRange.start) > 0) {\n    throw Error(`Found overlapping edit ranges in ${path}`)\n  }\n  const startRow = edit.oldRange.start.row\n  const startCol = edit.oldRange.start.column\n  const lineLength = buffer.lineLengthForRow(startRow)\n  if (lineLength == null || startCol > lineLength) {\n    throw Error(`Out of range edit on ${path}:${startRow + 1}:${startCol + 1}`)\n  }\n}\n"]}
\No newline at end of file