UNPKG

32 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3const convert_1 = require("../convert");
4const languageclient_1 = require("../languageclient");
5/**
6 * Public: Listen to diagnostics messages from the language server and publish them to the user by way of the Linter
7 * Push (Indie) v2 API provided by the Base Linter package.
8 */
9class LinterPushV2Adapter {
10 /**
11 * Public: Create a new {LinterPushV2Adapter} that will listen for diagnostics via the supplied {LanguageClientConnection}.
12 *
13 * @param connection A {LanguageClientConnection} to the language server that will provide diagnostics.
14 */
15 constructor(connection) {
16 /*
17 * A map from file path calculated using the LS diagnostic uri to an array of linter messages {linter.Message[]}
18 */
19 this._diagnosticMap = new Map();
20 /**
21 * A map from file path {linter.Message["location"]["file"]} to a Map of all Message keys to Diagnostics
22 * ${Map<linter.Message["key"], Diagnostic>} It has to be stored separately because a {Message} object cannot hold all
23 * of the information that a {Diagnostic} provides, thus we store the original {Diagnostic} object.
24 */
25 this._lsDiagnosticMap = new Map();
26 this._indies = new Set();
27 connection.onPublishDiagnostics(this.captureDiagnostics.bind(this));
28 }
29 /** Dispose this adapter ensuring any resources are freed and events unhooked. */
30 dispose() {
31 this.detachAll();
32 }
33 /**
34 * Public: Attach this {LinterPushV2Adapter} to a given {V2IndieDelegate} registry.
35 *
36 * @param indie A {V2IndieDelegate} that wants to receive messages.
37 */
38 attach(indie) {
39 this._indies.add(indie);
40 this._diagnosticMap.forEach((value, key) => indie.setMessages(key, value));
41 indie.onDidDestroy(() => {
42 this._indies.delete(indie);
43 });
44 }
45 /** Public: Remove all {V2IndieDelegate} registries attached to this adapter and clear them. */
46 detachAll() {
47 this._indies.forEach((i) => i.clearMessages());
48 this._indies.clear();
49 }
50 /**
51 * Public: Capture the diagnostics sent from a langguage server, convert them to the Linter V2 format and forward them
52 * on to any attached {V2IndieDelegate}s.
53 *
54 * @param params The {PublishDiagnosticsParams} received from the language server that should be captured and
55 * forwarded on to any attached {V2IndieDelegate}s.
56 */
57 captureDiagnostics(params) {
58 const path = convert_1.default.uriToPath(params.uri);
59 const codeMap = new Map();
60 const messages = params.diagnostics.map((d) => {
61 const linterMessage = lsDiagnosticToV2Message(path, d);
62 codeMap.set(getMessageKey(linterMessage), d);
63 return linterMessage;
64 });
65 this._diagnosticMap.set(path, messages);
66 this._lsDiagnosticMap.set(path, codeMap);
67 this._indies.forEach((i) => i.setMessages(path, messages));
68 }
69 /**
70 * Public: Convert a single {Diagnostic} received from a language server into a single {V2Message} expected by the
71 * Linter V2 API.
72 *
73 * @param path A string representing the path of the file the diagnostic belongs to.
74 * @param diagnostics A {Diagnostic} object received from the language server.
75 * @returns A {V2Message} equivalent to the {Diagnostic} object supplied by the language server.
76 */
77 diagnosticToV2Message(path, diagnostic) {
78 return {
79 location: {
80 file: path,
81 position: convert_1.default.lsRangeToAtomRange(diagnostic.range),
82 },
83 excerpt: diagnostic.message,
84 linterName: diagnostic.source,
85 severity: LinterPushV2Adapter.diagnosticSeverityToSeverity(diagnostic.severity || -1),
86 };
87 }
88 /**
89 * Public: get diagnostics for the given linter messages
90 *
91 * @param linterMessages An array of linter {V2Message}
92 * @returns An array of LS {Diagnostic[]}
93 */
94 getLSDiagnosticsForMessages(linterMessages) {
95 return linterMessages
96 .map(this.getLSDiagnosticForMessage)
97 // filter out undefined
98 .filter((diagnostic) => diagnostic !== undefined);
99 }
100 /**
101 * Public: Get the {Diagnostic} that is associated with the given Base Linter v2 {Message}.
102 *
103 * @param message The {Message} object to fetch the {Diagnostic} for.
104 * @returns The associated {Diagnostic}.
105 */
106 getLSDiagnosticForMessage(message) {
107 var _a;
108 return (_a = this._lsDiagnosticMap.get(message.location.file)) === null || _a === void 0 ? void 0 : _a.get(getMessageKey(message));
109 }
110 /**
111 * Public: Convert a diagnostic severity number obtained from the language server into the textual equivalent for a
112 * Linter {V2Message}.
113 *
114 * @param severity A number representing the severity of the diagnostic.
115 * @returns A string of 'error', 'warning' or 'info' depending on the severity.
116 */
117 static diagnosticSeverityToSeverity(severity) {
118 switch (severity) {
119 case languageclient_1.DiagnosticSeverity.Error:
120 return "error";
121 case languageclient_1.DiagnosticSeverity.Warning:
122 return "warning";
123 case languageclient_1.DiagnosticSeverity.Information:
124 case languageclient_1.DiagnosticSeverity.Hint:
125 default:
126 return "info";
127 }
128 }
129}
130exports.default = LinterPushV2Adapter;
131/**
132 * Public: Convert a single {Diagnostic} received from a language server into a single {Message} expected by the Linter V2 API.
133 *
134 * @param path A string representing the path of the file the diagnostic belongs to.
135 * @param diagnostics A {Diagnostic} object received from the language server.
136 * @returns A {Message} equivalent to the {Diagnostic} object supplied by the language server.
137 */
138function lsDiagnosticToV2Message(path, diagnostic) {
139 var _a, _b, _c;
140 return {
141 location: {
142 file: path,
143 position: convert_1.default.lsRangeToAtomRange(diagnostic.range),
144 },
145 reference: relatedInformationToReference(diagnostic.relatedInformation),
146 url: (_a = diagnostic.codeDescription) === null || _a === void 0 ? void 0 : _a.href,
147 icon: iconForLSSeverity((_b = diagnostic.severity) !== null && _b !== void 0 ? _b : languageclient_1.DiagnosticSeverity.Error),
148 excerpt: diagnostic.message,
149 linterName: diagnostic.source,
150 severity: lsSeverityToV2MessageSeverity((_c = diagnostic.severity) !== null && _c !== void 0 ? _c : languageclient_1.DiagnosticSeverity.Error),
151 // BLOCKED: on steelbrain/linter#1722
152 solutions: undefined,
153 };
154}
155/**
156 * Convert a severity level of an LSP {Diagnostic} to that of a Base Linter v2 {Message}. Note: this conversion is lossy
157 * due to the v2 Message not being able to represent hints.
158 *
159 * @param severity A severity level of of an LSP {Diagnostic} to be converted.
160 * @returns A severity level a Base Linter v2 {Message}.
161 */
162function lsSeverityToV2MessageSeverity(severity) {
163 switch (severity) {
164 case languageclient_1.DiagnosticSeverity.Error:
165 return "error";
166 case languageclient_1.DiagnosticSeverity.Warning:
167 return "warning";
168 case languageclient_1.DiagnosticSeverity.Information:
169 case languageclient_1.DiagnosticSeverity.Hint:
170 return "info";
171 default:
172 throw Error(`Unexpected diagnostic severity '${severity}'`);
173 }
174}
175/**
176 * Convert a diagnostic severity number obtained from the language server into an Octicon icon.
177 *
178 * @param severity A number representing the severity of the diagnostic.
179 * @returns An Octicon name.
180 */
181function iconForLSSeverity(severity) {
182 switch (severity) {
183 case languageclient_1.DiagnosticSeverity.Error:
184 return "stop";
185 case languageclient_1.DiagnosticSeverity.Warning:
186 return "warning";
187 case languageclient_1.DiagnosticSeverity.Information:
188 return "info";
189 case languageclient_1.DiagnosticSeverity.Hint:
190 return "light-bulb";
191 default:
192 return undefined;
193 }
194}
195/**
196 * Convert the related information from a diagnostic into a reference point for a Linter {V2Message}.
197 *
198 * @param relatedInfo Several related information objects (only the first is used).
199 * @returns A value that is suitable for using as {V2Message}.reference.
200 */
201function relatedInformationToReference(relatedInfo) {
202 if (relatedInfo === undefined || relatedInfo.length === 0) {
203 return undefined;
204 }
205 const location = relatedInfo[0].location;
206 return {
207 file: convert_1.default.uriToPath(location.uri),
208 position: convert_1.default.lsRangeToAtomRange(location.range).start,
209 };
210}
211/**
212 * Get a unique key for a Linter v2 Message
213 *
214 * @param message A {Message} object
215 * @returns ${string} a unique key
216 */
217function getMessageKey(message) {
218 if (typeof message.key !== "string") {
219 updateMessageKey(message);
220 }
221 return message.key; // updateMessageKey adds message.key string
222}
223/**
224 * Construct an unique key for a Linter v2 Message and store it in `Message.key`
225 *
226 * @param message A {Message} object to serialize.
227 * @returns ${string} a unique key
228 */
229function updateMessageKey(message) {
230 // From https://github.com/steelbrain/linter/blob/fadd462914ef0a8ed5b73a489f662a9393bdbe9f/lib/helpers.ts#L50-L64
231 const { reference, location } = message;
232 const nameStr = `$LINTER:${message.linterName}`;
233 const locationStr = `$LOCATION:${location.file}$${location.position.start.row}$${location.position.start.column}$${location.position.end.row}$${location.position.end.column}`;
234 const referenceStr = reference
235 ? `$REFERENCE:${reference.file}$${reference.position ? `${reference.position.row}$${reference.position.column}` : ""}`
236 : "$REFERENCE:null";
237 const excerptStr = `$EXCERPT:${message.excerpt}`;
238 const severityStr = `$SEVERITY:${message.severity}`;
239 const iconStr = message.icon ? `$ICON:${message.icon}` : "$ICON:null";
240 const urlStr = message.url ? `$URL:${message.url}` : "$URL:null";
241 const descriptionStr = typeof message.description === "string" ? `$DESCRIPTION:${message.description}` : "$DESCRIPTION:null";
242 message.key = `${nameStr}${locationStr}${referenceStr}${excerptStr}${severityStr}${iconStr}${urlStr}${descriptionStr}`;
243}
244//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"linter-push-v2-adapter.js","sourceRoot":"","sources":["../../../lib/adapters/linter-push-v2-adapter.ts"],"names":[],"mappings":"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","sourcesContent":["import * as linter from \"atom/linter\"\nimport Convert from \"../convert\"\nimport {\n  Diagnostic,\n  DiagnosticSeverity,\n  DiagnosticRelatedInformation,\n  LanguageClientConnection,\n  PublishDiagnosticsParams,\n} from \"../languageclient\"\n\n/**\n * Public: Listen to diagnostics messages from the language server and publish them to the user by way of the Linter\n * Push (Indie) v2 API provided by the Base Linter package.\n */\nexport default class LinterPushV2Adapter {\n  /*\n   * A map from file path calculated using the LS diagnostic uri to an array of linter messages {linter.Message[]}\n   */\n  protected _diagnosticMap: Map<string, linter.Message[]> = new Map()\n  /**\n   * A map from file path {linter.Message[\"location\"][\"file\"]} to a Map of all Message keys to Diagnostics\n   * ${Map<linter.Message[\"key\"], Diagnostic>} It has to be stored separately because a {Message} object cannot hold all\n   * of the information that a {Diagnostic} provides, thus we store the original {Diagnostic} object.\n   */\n  protected _lsDiagnosticMap: Map<linter.Message[\"location\"][\"file\"], Map<linter.Message[\"key\"], Diagnostic>> =\n    new Map()\n  protected _indies: Set<linter.IndieDelegate> = new Set()\n\n  /**\n   * Public: Create a new {LinterPushV2Adapter} that will listen for diagnostics via the supplied {LanguageClientConnection}.\n   *\n   * @param connection A {LanguageClientConnection} to the language server that will provide diagnostics.\n   */\n  constructor(connection: LanguageClientConnection) {\n    connection.onPublishDiagnostics(this.captureDiagnostics.bind(this))\n  }\n\n  /** Dispose this adapter ensuring any resources are freed and events unhooked. */\n  public dispose(): void {\n    this.detachAll()\n  }\n\n  /**\n   * Public: Attach this {LinterPushV2Adapter} to a given {V2IndieDelegate} registry.\n   *\n   * @param indie A {V2IndieDelegate} that wants to receive messages.\n   */\n  public attach(indie: linter.IndieDelegate): void {\n    this._indies.add(indie)\n    this._diagnosticMap.forEach((value, key) => indie.setMessages(key, value))\n    indie.onDidDestroy(() => {\n      this._indies.delete(indie)\n    })\n  }\n\n  /** Public: Remove all {V2IndieDelegate} registries attached to this adapter and clear them. */\n  public detachAll(): void {\n    this._indies.forEach((i) => i.clearMessages())\n    this._indies.clear()\n  }\n\n  /**\n   * Public: Capture the diagnostics sent from a langguage server, convert them to the Linter V2 format and forward them\n   * on to any attached {V2IndieDelegate}s.\n   *\n   * @param params The {PublishDiagnosticsParams} received from the language server that should be captured and\n   *   forwarded on to any attached {V2IndieDelegate}s.\n   */\n  public captureDiagnostics(params: PublishDiagnosticsParams): void {\n    const path = Convert.uriToPath(params.uri)\n    const codeMap = new Map<string, Diagnostic>()\n    const messages = params.diagnostics.map((d) => {\n      const linterMessage = lsDiagnosticToV2Message(path, d)\n      codeMap.set(getMessageKey(linterMessage), d)\n      return linterMessage\n    })\n    this._diagnosticMap.set(path, messages)\n    this._lsDiagnosticMap.set(path, codeMap)\n    this._indies.forEach((i) => i.setMessages(path, messages))\n  }\n\n  /**\n   * Public: Convert a single {Diagnostic} received from a language server into a single {V2Message} expected by the\n   * Linter V2 API.\n   *\n   * @param path A string representing the path of the file the diagnostic belongs to.\n   * @param diagnostics A {Diagnostic} object received from the language server.\n   * @returns A {V2Message} equivalent to the {Diagnostic} object supplied by the language server.\n   */\n  public diagnosticToV2Message(path: string, diagnostic: Diagnostic): linter.Message {\n    return {\n      location: {\n        file: path,\n        position: Convert.lsRangeToAtomRange(diagnostic.range),\n      },\n      excerpt: diagnostic.message,\n      linterName: diagnostic.source,\n      severity: LinterPushV2Adapter.diagnosticSeverityToSeverity(diagnostic.severity || -1),\n    }\n  }\n\n  /**\n   * Public: get diagnostics for the given linter messages\n   *\n   * @param linterMessages An array of linter {V2Message}\n   * @returns An array of LS {Diagnostic[]}\n   */\n  public getLSDiagnosticsForMessages(linterMessages: linter.Message[]): Diagnostic[] {\n    return (\n      linterMessages\n        .map(this.getLSDiagnosticForMessage)\n        // filter out undefined\n        .filter((diagnostic) => diagnostic !== undefined) as Diagnostic[]\n    )\n  }\n\n  /**\n   * Public: Get the {Diagnostic} that is associated with the given Base Linter v2 {Message}.\n   *\n   * @param message The {Message} object to fetch the {Diagnostic} for.\n   * @returns The associated {Diagnostic}.\n   */\n  public getLSDiagnosticForMessage(message: linter.Message): Diagnostic | undefined {\n    return this._lsDiagnosticMap.get(message.location.file)?.get(getMessageKey(message))\n  }\n\n  /**\n   * Public: Convert a diagnostic severity number obtained from the language server into the textual equivalent for a\n   * Linter {V2Message}.\n   *\n   * @param severity A number representing the severity of the diagnostic.\n   * @returns A string of 'error', 'warning' or 'info' depending on the severity.\n   */\n  public static diagnosticSeverityToSeverity(severity: number): \"error\" | \"warning\" | \"info\" {\n    switch (severity) {\n      case DiagnosticSeverity.Error:\n        return \"error\"\n      case DiagnosticSeverity.Warning:\n        return \"warning\"\n      case DiagnosticSeverity.Information:\n      case DiagnosticSeverity.Hint:\n      default:\n        return \"info\"\n    }\n  }\n}\n\n/**\n * Public: Convert a single {Diagnostic} received from a language server into a single {Message} expected by the Linter V2 API.\n *\n * @param path A string representing the path of the file the diagnostic belongs to.\n * @param diagnostics A {Diagnostic} object received from the language server.\n * @returns A {Message} equivalent to the {Diagnostic} object supplied by the language server.\n */\nfunction lsDiagnosticToV2Message(path: string, diagnostic: Diagnostic): linter.Message {\n  return {\n    location: {\n      file: path,\n      position: Convert.lsRangeToAtomRange(diagnostic.range),\n    },\n    reference: relatedInformationToReference(diagnostic.relatedInformation),\n    url: diagnostic.codeDescription?.href,\n    icon: iconForLSSeverity(diagnostic.severity ?? DiagnosticSeverity.Error),\n    excerpt: diagnostic.message,\n    linterName: diagnostic.source,\n    severity: lsSeverityToV2MessageSeverity(diagnostic.severity ?? DiagnosticSeverity.Error),\n    // BLOCKED: on steelbrain/linter#1722\n    solutions: undefined,\n  }\n}\n\n/**\n * Convert a severity level of an LSP {Diagnostic} to that of a Base Linter v2 {Message}. Note: this conversion is lossy\n * due to the v2 Message not being able to represent hints.\n *\n * @param severity A severity level of of an LSP {Diagnostic} to be converted.\n * @returns A severity level a Base Linter v2 {Message}.\n */\nfunction lsSeverityToV2MessageSeverity(severity: DiagnosticSeverity): linter.Message[\"severity\"] {\n  switch (severity) {\n    case DiagnosticSeverity.Error:\n      return \"error\"\n    case DiagnosticSeverity.Warning:\n      return \"warning\"\n    case DiagnosticSeverity.Information:\n    case DiagnosticSeverity.Hint:\n      return \"info\"\n    default:\n      throw Error(`Unexpected diagnostic severity '${severity}'`)\n  }\n}\n\n/**\n * Convert a diagnostic severity number obtained from the language server into an Octicon icon.\n *\n * @param severity A number representing the severity of the diagnostic.\n * @returns An Octicon name.\n */\nfunction iconForLSSeverity(severity: DiagnosticSeverity): string | undefined {\n  switch (severity) {\n    case DiagnosticSeverity.Error:\n      return \"stop\"\n    case DiagnosticSeverity.Warning:\n      return \"warning\"\n    case DiagnosticSeverity.Information:\n      return \"info\"\n    case DiagnosticSeverity.Hint:\n      return \"light-bulb\"\n    default:\n      return undefined\n  }\n}\n\n/**\n * Convert the related information from a diagnostic into a reference point for a Linter {V2Message}.\n *\n * @param relatedInfo Several related information objects (only the first is used).\n * @returns A value that is suitable for using as {V2Message}.reference.\n */\nfunction relatedInformationToReference(\n  relatedInfo: DiagnosticRelatedInformation[] | undefined\n): linter.Message[\"reference\"] {\n  if (relatedInfo === undefined || relatedInfo.length === 0) {\n    return undefined\n  }\n\n  const location = relatedInfo[0].location\n  return {\n    file: Convert.uriToPath(location.uri),\n    position: Convert.lsRangeToAtomRange(location.range).start,\n  }\n}\n\n/**\n * Get a unique key for a Linter v2 Message\n *\n * @param message A {Message} object\n * @returns ${string} a unique key\n */\nfunction getMessageKey(message: linter.Message): string {\n  if (typeof message.key !== \"string\") {\n    updateMessageKey(message)\n  }\n  return message.key as string // updateMessageKey adds message.key string\n}\n\n/**\n * Construct an unique key for a Linter v2 Message and store it in `Message.key`\n *\n * @param message A {Message} object to serialize.\n * @returns ${string} a unique key\n */\nfunction updateMessageKey(message: linter.Message): void {\n  // From https://github.com/steelbrain/linter/blob/fadd462914ef0a8ed5b73a489f662a9393bdbe9f/lib/helpers.ts#L50-L64\n  const { reference, location } = message\n  const nameStr = `$LINTER:${message.linterName}`\n  const locationStr = `$LOCATION:${location.file}$${location.position.start.row}$${location.position.start.column}$${location.position.end.row}$${location.position.end.column}`\n  const referenceStr = reference\n    ? `$REFERENCE:${reference.file}$${\n        reference.position ? `${reference.position.row}$${reference.position.column}` : \"\"\n      }`\n    : \"$REFERENCE:null\"\n  const excerptStr = `$EXCERPT:${message.excerpt}`\n  const severityStr = `$SEVERITY:${message.severity}`\n  const iconStr = message.icon ? `$ICON:${message.icon}` : \"$ICON:null\"\n  const urlStr = message.url ? `$URL:${message.url}` : \"$URL:null\"\n  const descriptionStr =\n    typeof message.description === \"string\" ? `$DESCRIPTION:${message.description}` : \"$DESCRIPTION:null\"\n  message.key = `${nameStr}${locationStr}${referenceStr}${excerptStr}${severityStr}${iconStr}${urlStr}${descriptionStr}`\n}\n"]}
\No newline at end of file