diff --git a/server/package-lock.json b/server/package-lock.json new file mode 100644 index 00000000..dc93420a --- /dev/null +++ b/server/package-lock.json @@ -0,0 +1,388 @@ +{ + "name": "vscode-k8s", + "version": "0.0.1", + "lockfileVersion": 1, + "dependencies": { + "@types/mocha": { + "version": "2.2.41", + "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-2.2.41.tgz", + "integrity": "sha1-4nzwgXFT658nE7LT9saPHhw8pgg=" + }, + "@types/node": { + "version": "6.0.79", + "resolved": "https://registry.npmjs.org/@types/node/-/node-6.0.79.tgz", + "integrity": "sha512-7F3/P6MkTPA0QxOstRqfcnoReCUy5V/QG92cyBoZSPnqdX44L8TtNELSVfN56gAttm3YWj9cEi8FRIPVq0WmeQ==", + "dev": true + }, + "agent-base": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-1.0.2.tgz", + "integrity": "sha1-aJDT+yFwBLYrcPiSjg+uX4lSpwY=" + }, + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", + "dev": true + }, + "arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", + "dev": true + }, + "balanced-match": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", + "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" + }, + "brace-expansion": { + "version": "1.1.8", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.8.tgz", + "integrity": "sha1-wHshHHyVLsH479Uad+8NHTmQopI=" + }, + "browser-stdout": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.0.tgz", + "integrity": "sha1-81HTKWnTL6XXpVZxVCY9korjvR8=" + }, + "chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "dependencies": { + "supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true + } + } + }, + "commander": { + "version": "2.9.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.9.0.tgz", + "integrity": "sha1-nJkJQXbhIkDLItbFFGCYQA/g99Q=" + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" + }, + "debug": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.0.tgz", + "integrity": "sha1-vFlryr52F/Edn6FTYe3tVgi4SZs=" + }, + "diff": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-3.2.0.tgz", + "integrity": "sha1-yc45Okt8vQsFinJck98pkCeGj/k=" + }, + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=" + }, + "extend": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.1.tgz", + "integrity": "sha1-p1Xqe8Gt/MWjHOfnYtuq3F5jZEQ=" + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" + }, + "glob": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz", + "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==" + }, + "graceful-readlink": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/graceful-readlink/-/graceful-readlink-1.0.1.tgz", + "integrity": "sha1-TK+tdrxi8C+gObL5Tpo906ORpyU=" + }, + "growl": { + "version": "1.9.2", + "resolved": "https://registry.npmjs.org/growl/-/growl-1.9.2.tgz", + "integrity": "sha1-Dqd0NxXbjY3ixe3hd14bRayFwC8=" + }, + "has-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", + "dev": true + }, + "has-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-1.0.0.tgz", + "integrity": "sha1-nZ55MWXOAXoA8AQYxD+UKnsdEfo=" + }, + "http-proxy-agent": { + "version": "0.2.7", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-0.2.7.tgz", + "integrity": "sha1-4X/aZfCQLZUs55IeYsf/iGJlWl4=" + }, + "https-proxy-agent": { + "version": "0.3.6", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-0.3.6.tgz", + "integrity": "sha1-cT+jjl01P1DrFKNC/r4pAz7RYZs=" + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=" + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" + }, + "json3": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/json3/-/json3-3.3.2.tgz", + "integrity": "sha1-PAQ0dD35Pi9cQq7nsZvLSDV19OE=" + }, + "jsonc-parser": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/jsonc-parser/-/jsonc-parser-0.4.2.tgz", + "integrity": "sha1-pLLK9n0acjlCgwZgYBMPcVBRMx0=" + }, + "lodash._baseassign": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/lodash._baseassign/-/lodash._baseassign-3.2.0.tgz", + "integrity": "sha1-jDigmVAPIVrQnlnxci/QxSv+Ck4=" + }, + "lodash._basecopy": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/lodash._basecopy/-/lodash._basecopy-3.0.1.tgz", + "integrity": "sha1-jaDmqHbPNEwK2KVIghEd08XHyjY=" + }, + "lodash._basecreate": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash._basecreate/-/lodash._basecreate-3.0.3.tgz", + "integrity": "sha1-G8ZhYU2qf8MRt9A78WgGoCE8+CE=" + }, + "lodash._getnative": { + "version": "3.9.1", + "resolved": "https://registry.npmjs.org/lodash._getnative/-/lodash._getnative-3.9.1.tgz", + "integrity": "sha1-VwvH3t5G1hzc3mh9ZdPuy6o6r/U=" + }, + "lodash._isiterateecall": { + "version": "3.0.9", + "resolved": "https://registry.npmjs.org/lodash._isiterateecall/-/lodash._isiterateecall-3.0.9.tgz", + "integrity": "sha1-UgOte6Ql+uhCRg5pbbnPPmqsBXw=" + }, + "lodash.create": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/lodash.create/-/lodash.create-3.1.1.tgz", + "integrity": "sha1-1/KEnw29p+BGgruM1yqwIkYd6+c=" + }, + "lodash.isarguments": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/lodash.isarguments/-/lodash.isarguments-3.1.0.tgz", + "integrity": "sha1-L1c9hcaiQon/AGY7SRwdM4/zRYo=" + }, + "lodash.isarray": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/lodash.isarray/-/lodash.isarray-3.0.4.tgz", + "integrity": "sha1-eeTriMNqgSKvhvhEqpvNhRtfu1U=" + }, + "lodash.keys": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/lodash.keys/-/lodash.keys-3.1.2.tgz", + "integrity": "sha1-TbwEcrFWvlCgsoaFXRvQsMZWCYo=" + }, + "make-error": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.0.tgz", + "integrity": "sha1-Uq06M5zPEM5itAQLcI/nByRLi5Y=", + "dev": true + }, + "minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==" + }, + "minimist": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", + "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" + }, + "mkdirp": { + "version": "0.5.1", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", + "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=" + }, + "mocha": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/mocha/-/mocha-3.4.2.tgz", + "integrity": "sha1-0O9NMyEm2/GNDWQMmzgt1IvpdZQ=", + "dependencies": { + "glob": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.1.tgz", + "integrity": "sha1-gFIR3wT6rxxjo2ADBs31reULLsg=" + } + } + }, + "ms": { + "version": "0.7.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.2.tgz", + "integrity": "sha1-riXPJRKziFodldfwN4aNhDESR2U=" + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=" + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" + }, + "request-light": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/request-light/-/request-light-0.2.1.tgz", + "integrity": "sha1-mG9agok+nRymqJbr5vRsUca0VX8=" + }, + "source-map": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.6.tgz", + "integrity": "sha1-dc449SvwczxafwwRjYEzSiu19BI=", + "dev": true + }, + "source-map-support": { + "version": "0.4.15", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.4.15.tgz", + "integrity": "sha1-AyAt9lwG0r2MfsI2KhkwVv7407E=", + "dev": true + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true + }, + "strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", + "dev": true + }, + "strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", + "dev": true + }, + "supports-color": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-3.1.2.tgz", + "integrity": "sha1-cqJiiU2dQIuVbKBf83su2KbiotU=" + }, + "triesearch": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/triesearch/-/triesearch-1.0.2.tgz", + "integrity": "sha1-fZnBuyoJZWGKGsHsgvXvfIaO1+0=" + }, + "ts-node": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-3.1.0.tgz", + "integrity": "sha1-p17FrrSPMFixuUXbp2XxFQuoj4w=", + "dev": true, + "dependencies": { + "minimist": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", + "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=", + "dev": true + } + } + }, + "tsconfig": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/tsconfig/-/tsconfig-6.0.0.tgz", + "integrity": "sha1-aw6DdgA9evGGT434+J3QBZ/80DI=", + "dev": true + }, + "typescript": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-2.1.5.tgz", + "integrity": "sha1-b+lHngDgGFUkfOohbnVhuvzbzUo=", + "dev": true + }, + "user-home": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/user-home/-/user-home-1.1.1.tgz", + "integrity": "sha1-K1viOjK2Onyd640PKNSFcko98ZA=", + "dev": true + }, + "v8flags": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/v8flags/-/v8flags-2.1.1.tgz", + "integrity": "sha1-qrGh+jDUX4jdMhFIh1rALAtV5bQ=", + "dev": true + }, + "vscode-json-languageservice": { + "version": "2.0.12", + "resolved": "https://registry.npmjs.org/vscode-json-languageservice/-/vscode-json-languageservice-2.0.12.tgz", + "integrity": "sha1-NmHQ6xv60JZ2IwjWh2qi5vUQYF0=" + }, + "vscode-jsonrpc": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/vscode-jsonrpc/-/vscode-jsonrpc-3.3.1.tgz", + "integrity": "sha512-iLlG27498AJF0j4GJ4yua7Z9bpJfLfwpAaAA9mihe6VDoYHwK8TyFgnpXdgjoTb8X9/DnzimQeg0bjIWINvPWw==" + }, + "vscode-languageserver": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/vscode-languageserver/-/vscode-languageserver-3.3.0.tgz", + "integrity": "sha512-8YvEbjl77Nl24/cgcSbQC/CPEhSoJwvLLKGJZJbIF4DsGo4jrVDbuARXfmUt9S6vCEEr++o3fbbZ17iLZ0QH0A==" + }, + "vscode-languageserver-types": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/vscode-languageserver-types/-/vscode-languageserver-types-3.3.0.tgz", + "integrity": "sha512-5BMClM4D3mRl5JlWFxIxhhJAbcVW9dFviz8ubppmG8epCTzl1bPpndcnvsjOjUlVsO9V8l8Ktklqc70Ew6soew==" + }, + "vscode-nls": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/vscode-nls/-/vscode-nls-2.0.2.tgz", + "integrity": "sha1-gIUiOAhEuK0VNJmvXDsDkhrqAto=" + }, + "vscode-uri": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/vscode-uri/-/vscode-uri-1.0.0.tgz", + "integrity": "sha1-BzaeLQlr/kK0Br0hhgtapP56etA=" + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" + }, + "yaml-ast-parser": { + "version": "0.0.32", + "resolved": "https://registry.npmjs.org/yaml-ast-parser/-/yaml-ast-parser-0.0.32.tgz", + "integrity": "sha1-2g8cKJl4BfFZf+777U+SvWdla28=" + }, + "yaml-ast-parser-beta": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/yaml-ast-parser-beta/-/yaml-ast-parser-beta-0.0.33.tgz", + "integrity": "sha1-oGqrAmAjGA56imaUFiL2FtaIKNA=" + }, + "yn": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/yn/-/yn-2.0.0.tgz", + "integrity": "sha1-5a2ryKz0CPY4X8dklWhMiOavaJo=", + "dev": true + } + } +} diff --git a/server/package.json b/server/package.json index 48fe48bd..267d8bc3 100755 --- a/server/package.json +++ b/server/package.json @@ -14,6 +14,7 @@ "mocha": "^3.4.2", "request-light": "^0.2.0", "triesearch": "^1.0.2", + "vscode-json-languageservice": "^2.0.12", "vscode-languageserver": "^3.1.0", "vscode-nls": "^2.0.2", "vscode-uri": "1.0.0", diff --git a/server/src/languageModelCache.ts b/server/src/languageModelCache.ts new file mode 100644 index 00000000..b12f77b5 --- /dev/null +++ b/server/src/languageModelCache.ts @@ -0,0 +1,83 @@ +/*--------------------------------------------------------------------------------------------- + * Copyright (c) Microsoft Corporation. All rights reserved. + * Licensed under the MIT License. See License.txt in the project root for license information. + *--------------------------------------------------------------------------------------------*/ +'use strict'; + +import { TextDocument } from 'vscode-languageserver'; + +export interface LanguageModelCache { + get(document: TextDocument): T; + onDocumentRemoved(document: TextDocument): void; + dispose(): void; +} + +export function getLanguageModelCache(maxEntries: number, cleanupIntervalTimeInSec: number, parse: (document: TextDocument) => T): LanguageModelCache { + let languageModels: { [uri: string]: { version: number, languageId: string, cTime: number, languageModel: T } } = {}; + let nModels = 0; + + let cleanupInterval = void 0; + if (cleanupIntervalTimeInSec > 0) { + cleanupInterval = setInterval(() => { + let cutoffTime = Date.now() - cleanupIntervalTimeInSec * 1000; + let uris = Object.keys(languageModels); + for (let uri of uris) { + let languageModelInfo = languageModels[uri]; + if (languageModelInfo.cTime < cutoffTime) { + delete languageModels[uri]; + nModels--; + } + } + }, cleanupIntervalTimeInSec * 1000); + } + + return { + get(document: TextDocument): T { + let version = document.version; + let languageId = document.languageId; + let languageModelInfo = languageModels[document.uri]; + if (languageModelInfo && languageModelInfo.version === version && languageModelInfo.languageId === languageId) { + languageModelInfo.cTime = Date.now(); + return languageModelInfo.languageModel; + } + let languageModel = parse(document); + languageModels[document.uri] = { languageModel, version, languageId, cTime: Date.now() }; + if (!languageModelInfo) { + nModels++; + } + + if (nModels === maxEntries) { + let oldestTime = Number.MAX_VALUE; + let oldestUri = null; + for (let uri in languageModels) { + let languageModelInfo = languageModels[uri]; + if (languageModelInfo.cTime < oldestTime) { + oldestUri = uri; + oldestTime = languageModelInfo.cTime; + } + } + if (oldestUri) { + delete languageModels[oldestUri]; + nModels--; + } + } + return languageModel; + + }, + onDocumentRemoved(document: TextDocument) { + let uri = document.uri; + if (languageModels[uri]) { + delete languageModels[uri]; + nModels--; + } + }, + dispose() { + if (typeof cleanupInterval !== 'undefined') { + clearInterval(cleanupInterval); + cleanupInterval = void 0; + languageModels = {}; + nModels = 0; + } + } + }; +} \ No newline at end of file diff --git a/server/src/languageService/parser/documentPositionCalculator.ts b/server/src/languageService/parser/documentPositionCalculator.ts new file mode 100644 index 00000000..b408a38e --- /dev/null +++ b/server/src/languageService/parser/documentPositionCalculator.ts @@ -0,0 +1,62 @@ +"use strict" + +export function insertionPointReturnValue(pt: number) { + return ((-pt) - 1) +} + +export function binarySearch(array: number[], sought: number) { + + let lower = 0 + let upper = array.length - 1 + + while (lower <= upper) { + let idx = Math.floor((lower + upper) / 2) + const value = array[idx] + + if (value === sought) { + return idx; + } + + if (lower === upper) { + const insertionPoint = (value < sought) ? idx + 1 : idx + return insertionPointReturnValue(insertionPoint) + } + + if (sought > value) { + lower = idx + 1; + } else if (sought < value) { + upper = idx - 1; + } + } +} + +export function getLineStartPositions(text: string) { + const lineStartPositions = [0]; + for (var i = 0; i < text.length; i++) { + const c = text[i]; + + if (c === '\r') { + // Check for Windows encoding, otherwise we are old Mac + if (i + 1 < text.length && text[i + 1] == '\n') { + i++; + } + + lineStartPositions.push(i + 1); + } else if (c === '\n'){ + lineStartPositions.push(i + 1); + } + } + + return lineStartPositions; +} + +export function getPosition(pos: number, lineStartPositions: number[]){ + let line = binarySearch(lineStartPositions, pos) + + if (line < 0){ + const insertionPoint = -1 * line - 1; + line = insertionPoint - 1; + } + + return {line, column: pos - lineStartPositions[line]} +} \ No newline at end of file diff --git a/server/src/languageService/parser/yamlParser.ts b/server/src/languageService/parser/yamlParser.ts new file mode 100644 index 00000000..ef619602 --- /dev/null +++ b/server/src/languageService/parser/yamlParser.ts @@ -0,0 +1,377 @@ +'use strict'; + +import { JSONDocument, ASTNode, ErrorCode, BooleanASTNode, NullASTNode, ArrayASTNode, NumberASTNode, ObjectASTNode, PropertyASTNode, StringASTNode } from './jsonParser'; + +import * as nls from 'vscode-nls'; +const localize = nls.loadMessageBundle(); + +import * as Yaml from 'yaml-ast-parser' +import { Kind } from 'yaml-ast-parser' + +import { getLineStartPositions, getPosition } from './documentPositionCalculator' + +export class YAMLDocument extends JSONDocument { + private lines; + + constructor(lines: number[]) { + super({disallowComments: false, ignoreDanglingComma: true}); + this.lines = lines; + } + + // TODO: This is complicated, messy and probably buggy + // It should be re-written. + // To get the correct behavior, it probably needs to be aware of + // the type of the nodes it is processing since there are no delimiters + // like in JSON. (ie. so it correctly returns 'object' vs 'property') + public getNodeFromOffsetEndInclusive(offset: number): ASTNode { + if (!this.root) { + return; + } + if (offset < this.root.start || offset > this.root.end) { + // We somehow are completely outside the document + // This is unexpected + console.log("Attempting to resolve node outside of document") + return null; + } + + const children = this.root.getChildNodes() + + function* sliding2(nodes: ASTNode[]) { + var i = 0; + while (i < nodes.length) { + yield [nodes[i], (i === nodes.length) ? null : nodes[i + 1]] + i++; + } + } + + const onLaterLine = (offset: number, node: ASTNode) => { + const { line: actualLine } = getPosition(offset, this.lines) + const { line: nodeEndLine } = getPosition(node.end, this.lines) + + return actualLine > nodeEndLine; + } + + let findNode = (nodes: ASTNode[]): ASTNode => { + if (nodes.length === 0) { + return null; + } + + var gen = sliding2(nodes); + + let result: IteratorResult = { done: false, value: undefined } + + for (let [first, second] of gen) { + const end = (second) ? second.start : first.parent.end + if (offset >= first.start && offset < end) { + const children = first.getChildNodes(); + + const foundChild = findNode(children) + + if (foundChild) { + if (foundChild['isKey'] && foundChild.end < offset) { + return foundChild.parent; + } + + if (foundChild.type === "null") { + return null; + } + } + + if (!foundChild && onLaterLine(offset, first)) { + return this.getNodeByIndent(this.lines, offset, this.root) + } + + return foundChild || first; + } + } + + return null; + } + + return findNode(children) || this.root; + } + + public getNodeFromOffset(offset: number): ASTNode { + return this.getNodeFromOffsetEndInclusive(offset); + } + + private getNodeByIndent = (lines: number[], offset: number, node: ASTNode) => { + + const { line, column: indent } = getPosition(offset, this.lines) + + const children = node.getChildNodes() + + function findNode(children) { + for (var idx = 0; idx < children.length; idx++) { + var child = children[idx]; + + const { line: childLine, column: childCol } = getPosition(child.start, lines); + + if (childCol > indent) { + return null; + } + + const newChildren = child.getChildNodes() + const foundNode = findNode(newChildren) + + if (foundNode) { + return foundNode; + } + + // We have the right indentation, need to return based on line + if (childLine == line) { + return child; + } + if (childLine > line) { + // Get previous + (idx - 1) >= 0 ? children[idx - 1] : child; + } + // Else continue loop to try next element + } + + // Special case, we found the correct + return children[children.length - 1] + } + + return findNode(children) || node + } +} + + +function recursivelyBuildAst(parent: ASTNode, node: Yaml.YAMLNode): ASTNode { + + if (!node) { + return; + } + + switch (node.kind) { + case Yaml.Kind.MAP: { + const instance = node; + + const result = new ObjectASTNode(parent, null, node.startPosition, node.endPosition) + result.addProperty + + for (const mapping of instance.mappings) { + result.addProperty(recursivelyBuildAst(result, mapping)) + } + + return result; + } + case Yaml.Kind.MAPPING: { + const instance = node; + const key = instance.key; + + // Technically, this is an arbitrary node in YAML + // I doubt we would get a better string representation by parsing it + const keyNode = new StringASTNode(null, null, true, key.startPosition, key.endPosition); + keyNode.value = key.value; + + const result = new PropertyASTNode(parent, keyNode) + result.end = instance.endPosition + + const valueNode = (instance.value) ? recursivelyBuildAst(result, instance.value) : new NullASTNode(parent, key.value, instance.endPosition, instance.endPosition) + valueNode.location = key.value + + result.setValue(valueNode) + + return result; + } + case Yaml.Kind.SEQ: { + const instance = node; + + const result = new ArrayASTNode(parent, null, instance.startPosition, instance.endPosition); + + let count = 0; + for (const item of instance.items) { + if (item === null && count === instance.items.length - 1) { + break; + } + + // Be aware of https://github.com/nodeca/js-yaml/issues/321 + // Cannot simply work around it here because we need to know if we are in Flow or Block + var itemNode = (item === null) ? new NullASTNode(parent, null, instance.endPosition, instance.endPosition) : recursivelyBuildAst(result, item); + + itemNode.location = count++; + result.addItem(itemNode); + } + + return result; + } + case Yaml.Kind.SCALAR: { + const instance = node; + + const type = determineScalarType(instance) + + // The name is set either by the sequence or the mapping case. + const name = null; + const value = instance.value; + + switch (type) { + case ScalarType.null: { + return new NullASTNode(parent, name, instance.startPosition, instance.endPosition); + } + case ScalarType.bool: { + return new BooleanASTNode(parent, name, parseYamlBoolean(value), node.startPosition, node.endPosition) + } + case ScalarType.int: { + const result = new NumberASTNode(parent, name, node.startPosition, node.endPosition); + result.value = parseYamlInteger(value); + result.isInteger = true; + return result; + } + case ScalarType.float: { + const result = new NumberASTNode(parent, name, node.startPosition, node.endPosition); + result.value = parseYamlFloat(value); + result.isInteger = false; + return result; + } + case ScalarType.string: { + const result = new StringASTNode(parent, name, false, node.startPosition, node.endPosition); + result.value = node.value; + return result; + } + } + + break; + } + case Yaml.Kind.ANCHOR_REF: { + const instance = (node).value + + return recursivelyBuildAst(parent, instance) || + new NullASTNode(parent, null, node.startPosition, node.endPosition); + } + case Yaml.Kind.INCLUDE_REF: { + // Issue Warning + console.log("Unsupported feature, node kind: " + node.kind); + break; + } + } +} + +export function parseYamlBoolean(input: string): boolean { + if (["true", "True", "TRUE"].lastIndexOf(input) >= 0) { + return true; + } + else if (["false", "False", "FALSE"].lastIndexOf(input) >= 0) { + return false; + } + throw `Invalid boolean "${input}"` +} + +function safeParseYamlInteger(input: string): number { + // Use startsWith when es6 methods becomes available + if (input.lastIndexOf('0o', 0) === 0) { + return parseInt(input.substring(2), 8) + } + + return parseInt(input); +} + +export function parseYamlInteger(input: string): number { + const result = safeParseYamlInteger(input) + + if (isNaN(result)) { + throw `Invalid integer "${input}"` + } + + return result; +} + +export function parseYamlFloat(input: string): number { + + if ([".nan", ".NaN", ".NAN"].lastIndexOf(input) >= 0) { + return NaN; + } + + const infinity = /^([-+])?(?:\.inf|\.Inf|\.INF)$/ + const match = infinity.exec(input) + if (match) { + return (match[1] === '-') ? -Infinity : Infinity; + } + + const result = parseFloat(input) + + if (!isNaN(result)) { + return result; + } + + throw `Invalid float "${input}"` +} + +export enum ScalarType { + null, bool, int, float, string +} + +export function determineScalarType(node: Yaml.YAMLScalar): ScalarType { + if (node === undefined) { + return ScalarType.null; + } + + if (node.doubleQuoted || !node.plainScalar || node['singleQuoted']) { + return ScalarType.string + } + + const value = node.value; + + if (["null", "Null", "NULL", "~", ''].indexOf(value) >= 0) { + return ScalarType.null; + } + + if (value === null || value === undefined) { + return ScalarType.null; + } + + if (["true", "True", "TRUE", "false", "False", "FALSE"].indexOf(value) >= 0) { + return ScalarType.bool; + } + + const base10 = /^[-+]?[0-9]+$/ + const base8 = /^0o[0-7]+$/ + const base16 = /^0x[0-9a-fA-F]+$/ + + if (base10.test(value) || base8.test(value) || base16.test(value)) { + return ScalarType.int; + } + + const float = /^[-+]?(\.[0-9]+|[0-9]+(\.[0-9]*)?)([eE][-+]?[0-9]+)?$/ + const infinity = /^[-+]?(\.inf|\.Inf|\.INF)$/ + if (float.test(value) || infinity.test(value) || [".nan", ".NaN", ".NAN"].indexOf(value) >= 0) { + return ScalarType.float; + } + + return ScalarType.string; +} + +function convertError(e: Yaml.YAMLException) { + // Subtract 2 because \n\0 is added by the parser (see loader.ts/loadDocuments) + const bufferLength = e.mark.buffer.length - 2; + + // TODO determine correct positioning. + return { message: `${e.message}`, location: { start: Math.min(e.mark.position, bufferLength - 1), end: bufferLength, code: ErrorCode.Undefined } } +} + +export function parse(text: string): JSONDocument { + + const startPositions = getLineStartPositions(text) + let _doc = new YAMLDocument(startPositions); + // This is documented to return a YAMLNode even though the + // typing only returns a YAMLDocument + const yamlDoc = Yaml.safeLoad(text, {}) + + _doc.root = recursivelyBuildAst(null, yamlDoc) + + if (!_doc.root) { + // TODO: When this is true, consider not pushing the other errors. + _doc.errors.push({ message: localize('Invalid symbol', 'Expected a YAML object, array or literal'), code: ErrorCode.Undefined, location: { start: yamlDoc.startPosition, end: yamlDoc.endPosition } }); + } + + const duplicateKeyReason = 'duplicate key' + + const errors = yamlDoc.errors.filter(e => e.reason !== duplicateKeyReason).map(e => convertError(e)) + const warnings = yamlDoc.errors.filter(e => e.reason === duplicateKeyReason).map(e => convertError(e)) + + errors.forEach(e => _doc.errors.push(e)); + warnings.forEach(e => _doc.warnings.push(e)); + + return _doc; +} \ No newline at end of file diff --git a/server/src/languageService/utils/astServices.ts b/server/src/languageService/utils/astServices.ts index 7e1102ba..479206f6 100644 --- a/server/src/languageService/utils/astServices.ts +++ b/server/src/languageService/utils/astServices.ts @@ -72,47 +72,18 @@ export function generateChildren(node){ case Kind.MAP : let yamlMappingNodeList = []; ( node).mappings.forEach(node => { - let gen = this.generateChildren(node); + let gen = generateChildren(node); yamlMappingNodeList.push(gen); }); return [].concat([], yamlMappingNodeList); case Kind.SEQ : let yamlSeqNodeList = []; ( node).items.forEach(node => { - let gen = this.generateChildren(node); + let gen = generateChildren(node); gen.forEach(element => { yamlSeqNodeList.push(element); }); }); return [].concat([], yamlSeqNodeList); } - } - -export function traverseForSymbols(node: YAMLNode){ - if(!node) return; - switch(node.kind){ - case Kind.SCALAR: - let scalar = node; - return [{type: 0, value: scalar}]; - case Kind.SEQ: - let seq = node; - let seqList = []; - seq.items.forEach(item=>{ - seqList.push(traverseForSymbols(item)); - }); - return seqList.concat({kind: 9, value: seq.key}); - case Kind.MAPPING: - let mapping = node; - return traverseForSymbols(mapping.value).concat({kind: 9, value: mapping.key}); - case Kind.MAP: - let map = node; - let mapList = []; - map.mappings.forEach(mapping=>{ - mapList.push(traverseForSymbols(mapping)); - }); - return mapList; - case Kind.ANCHOR_REF: - let anchor = node; - return traverseForSymbols(anchor.value).concat({kind: 9, value: anchor.key}); - } } \ No newline at end of file diff --git a/server/src/server.ts b/server/src/server.ts index 170f3c98..5c808dab 100755 --- a/server/src/server.ts +++ b/server/src/server.ts @@ -14,8 +14,10 @@ import Strings = require( './languageService/utils/strings'); import URI from './languageService/utils/uri'; import * as URL from 'url'; import fs = require('fs'); +import { getLanguageModelCache } from './languageModelCache'; import {snippetAutocompletor} from './SnippetSupport/snippet'; -import {traverseForSymbols} from './languageService/utils/astServices'; +import {parse as parseYaml} from './languageService/parser/yamlParser'; +import {JSONDocument, getLanguageService as getJsonLanguageService } from 'vscode-json-languageservice'; var glob = require('glob'); namespace VSCodeContentRequest { @@ -88,6 +90,7 @@ let schemaRequestService = (uri: string): Thenable => { }; let languageService = getLanguageService(schemaRequestService, workspaceContext); +let jsonLanguageService = getJsonLanguageService(schemaRequestService); // The content of a text document has changed. This event is emitted // when the text document first opened or when its content has changed. @@ -248,23 +251,25 @@ connection.onCompletionResolve((item: CompletionItem): CompletionItem => { return item; }); +let yamlDocuments = getLanguageModelCache(10, 60, document => parseYaml(document.getText())); + +documents.onDidClose(e => { + yamlDocuments.onDocumentRemoved(e.document); +}); + +connection.onShutdown(() => { + yamlDocuments.dispose(); +}); + +function getJSONDocument(document: TextDocument): JSONDocument { + return yamlDocuments.get(document); +} + connection.onDocumentSymbol(params => { - let doc = documents.get(params.textDocument.uri); - let yamlDoc:YAMLDocument = yamlLoader(doc.getText(),{}); - let symbols = traverseForSymbols(yamlDoc); - let flattenedSymbols = flatten(symbols); - let documentSymbols = []; - flattenedSymbols.forEach(obj => { - if(obj !== null && obj !== undefined && obj.value){ - documentSymbols.push({ - name: obj.value.value, - kind: obj.kind, - location: Location.create(params.textDocument.uri, Range.create(doc.positionAt(obj.value.startPosition), doc.positionAt(obj.value.endPosition))) - }); - } - }); - - return documentSymbols; + + let document = documents.get(params.textDocument.uri); + let jsonDocument = getJSONDocument(document); + return jsonLanguageService.findDocumentSymbols(document, jsonDocument); });