From 5eb19c6d4b98dc612cd8a0374b62afb7ba3e1106 Mon Sep 17 00:00:00 2001 From: Danilo Sampaio Date: Thu, 3 Sep 2020 05:34:51 -0300 Subject: [PATCH 1/7] Fix #158 handle properly dependencies cache when deno/deps does not exists (#169) Co-authored-by: Jose Danilo Sampaio da Silva (Venturus) --- core/deno_deps.ts | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/core/deno_deps.ts b/core/deno_deps.ts index 06281c40..bd3905e7 100644 --- a/core/deno_deps.ts +++ b/core/deno_deps.ts @@ -31,7 +31,16 @@ export type Range = { export async function getAllDenoCachedDeps(): Promise { const depsRootDir = getDenoDepsDir(); const deps: Deps[] = []; - const protocols = await fs.readdir(depsRootDir); + let protocols: string[] = []; + + try { + protocols = await fs.readdir(depsRootDir); + } catch (error) { + //deno/deps directory does not exists + if (error.code === "ENOENT") { + return deps; + } + } await Promise.all( protocols.map(async (protocol) => { From 6f05bab521ab793957fffe8f42f9f92774dd2527 Mon Sep 17 00:00:00 2001 From: Jesse Jackson Date: Fri, 4 Sep 2020 08:20:46 -0500 Subject: [PATCH 2/7] Add support for URLs with non-default ports (#173) Co-authored-by: Luca Casonato --- core/deno_cache.ts | 2 +- core/module_resolver.test.ts | 21 +++++++++++++++++++ core/module_resolver.ts | 2 +- ...7363be7846d6d3b84a2b8697b9bd50c60d4509ab60 | 0 ...b84a2b8697b9bd50c60d4509ab60.metadata.json | 4 ++++ 5 files changed, 27 insertions(+), 2 deletions(-) create mode 100644 core/testdata/deno_dir_manifest/deps/http/localhost_PORT3000/51a61f14fdad404fd2c9187363be7846d6d3b84a2b8697b9bd50c60d4509ab60 create mode 100644 core/testdata/deno_dir_manifest/deps/http/localhost_PORT3000/51a61f14fdad404fd2c9187363be7846d6d3b84a2b8697b9bd50c60d4509ab60.metadata.json diff --git a/core/deno_cache.ts b/core/deno_cache.ts index 6a79a136..9a1f695a 100644 --- a/core/deno_cache.ts +++ b/core/deno_cache.ts @@ -125,7 +125,7 @@ export class CacheModule implements DenoCacheModule { targetOriginDir = path.join( getDenoDepsDir(), url.protocol.replace(/:$/, ""), // https: -> https - url.hostname + `${url.hostname}${url.port ? `_PORT${url.port}` : ""}` ); } // invalid diff --git a/core/module_resolver.test.ts b/core/module_resolver.test.ts index ce68e039..0b3653f4 100644 --- a/core/module_resolver.test.ts +++ b/core/module_resolver.test.ts @@ -28,6 +28,8 @@ test("core / module_resolver: resolve module from Deno cache", () => { "./sub/mod.ts", "/esm/mod.ts", "https://example.com/esm/mod.ts", + "https://example.com:443/esm/mod.ts", + "http://localhost:3000/mod.ts", "./module_not_exist.ts", "https://module.not.exist.com/mod.ts", ]) @@ -56,6 +58,25 @@ test("core / module_resolver: resolve module from Deno cache", () => { "8afd52da760dab7f2deda4b7453197f50421f310372c5da3f3847ffd062fa1cf" ), }, + { + extension: ".ts", + origin: "https://example.com:443/esm/mod.ts", + filepath: path.join( + path.dirname(cacheFilepath), + "8afd52da760dab7f2deda4b7453197f50421f310372c5da3f3847ffd062fa1cf" + ), + }, + { + extension: ".ts", + origin: "http://localhost:3000/mod.ts", + filepath: path.join( + denoDir, + "deps", + "http", + "localhost_PORT3000", + "51a61f14fdad404fd2c9187363be7846d6d3b84a2b8697b9bd50c60d4509ab60" + ), + }, undefined, undefined, ] as ResolvedModule[]); diff --git a/core/module_resolver.ts b/core/module_resolver.ts index ce40ca9a..d9cd6892 100644 --- a/core/module_resolver.ts +++ b/core/module_resolver.ts @@ -75,7 +75,7 @@ export class ModuleResolver implements ModuleResolverInterface { const originDir = path.join( getDenoDepsDir(), url.protocol.replace(/:$/, ""), // https: -> https - url.hostname + `${url.hostname}${url.port ? `_PORT${url.port}` : ""}` // hostname.xyz:3000 -> hostname.xyz_PORT3000 ); const hash = hashURL(url); diff --git a/core/testdata/deno_dir_manifest/deps/http/localhost_PORT3000/51a61f14fdad404fd2c9187363be7846d6d3b84a2b8697b9bd50c60d4509ab60 b/core/testdata/deno_dir_manifest/deps/http/localhost_PORT3000/51a61f14fdad404fd2c9187363be7846d6d3b84a2b8697b9bd50c60d4509ab60 new file mode 100644 index 00000000..e69de29b diff --git a/core/testdata/deno_dir_manifest/deps/http/localhost_PORT3000/51a61f14fdad404fd2c9187363be7846d6d3b84a2b8697b9bd50c60d4509ab60.metadata.json b/core/testdata/deno_dir_manifest/deps/http/localhost_PORT3000/51a61f14fdad404fd2c9187363be7846d6d3b84a2b8697b9bd50c60d4509ab60.metadata.json new file mode 100644 index 00000000..5f3b30f5 --- /dev/null +++ b/core/testdata/deno_dir_manifest/deps/http/localhost_PORT3000/51a61f14fdad404fd2c9187363be7846d6d3b84a2b8697b9bd50c60d4509ab60.metadata.json @@ -0,0 +1,4 @@ +{ + "headers": {}, + "url": "http://localhost:3000/mod.ts" +} From dd25cff71cc34bb565d6be8b40e55d789ca5f018 Mon Sep 17 00:00:00 2001 From: CGQAQ Date: Fri, 4 Sep 2020 21:49:52 +0800 Subject: [PATCH 3/7] feat: IntelliSense for std and deno.land/x imports (#172) --- client/src/extension.ts | 35 +++ client/src/import_enhancement_provider.ts | 255 ++++++++++++++++++++++ client/src/import_utils.test.ts | 106 +++++++++ client/src/import_utils.ts | 167 ++++++++++++++ client/src/types/vscode-cache.d.ts | 27 +++ package.json | 28 ++- snippets/import_enhancement.json | 14 ++ 7 files changed, 631 insertions(+), 1 deletion(-) create mode 100644 client/src/import_enhancement_provider.ts create mode 100644 client/src/import_utils.test.ts create mode 100644 client/src/import_utils.ts create mode 100644 client/src/types/vscode-cache.d.ts create mode 100644 snippets/import_enhancement.json diff --git a/client/src/extension.ts b/client/src/extension.ts index adf18a3f..99ab3e47 100644 --- a/client/src/extension.ts +++ b/client/src/extension.ts @@ -32,6 +32,11 @@ import execa from "execa"; import * as semver from "semver"; import { TreeViewProvider } from "./tree_view_provider"; +import { + ImportEnhancementCompletionProvider, + CACHE_STATE, +} from "./import_enhancement_provider"; + import { ImportMap } from "../../core/import_map"; import { HashMeta } from "../../core/hash_meta"; import { isInDeno } from "../../core/deno"; @@ -113,6 +118,9 @@ export class Extension { }, executablePath: "", }; + // CGQAQ: ImportEnhancementCompletionProvider instance + private import_enhancement_completion_provider = new ImportEnhancementCompletionProvider(); + // get configuration of Deno public getConfiguration(uri?: Uri): ConfigurationField { const config: ConfigurationField = {}; @@ -465,6 +473,14 @@ Executable ${this.denoInfo.executablePath}`; await window.showInformationMessage(`Copied to clipboard.`); }); + // CGQAQ: deno._clear_import_enhencement_cache + this.registerCommand("_clear_import_enhencement_cache", async () => { + this.import_enhancement_completion_provider + .clearCache() + .then(() => window.showInformationMessage("Clear success!")) + .catch(() => window.showErrorMessage("Clear failed!")); + }); + this.registerQuickFix({ _fetch_remote_module: async (editor, text) => { const config = this.getConfiguration(editor.document.uri); @@ -595,6 +611,23 @@ Executable ${this.denoInfo.executablePath}`; window.registerTreeDataProvider("deno", treeView) ); + // CGQAQ: activate import enhance feature + this.import_enhancement_completion_provider.activate(this.context); + + // CGQAQ: Start caching full module list + this.import_enhancement_completion_provider + .cacheModList() + .then((state) => { + if (state === CACHE_STATE.CACHE_SUCCESS) { + window.showInformationMessage( + "deno.land/x module list cached successfully!" + ); + } + }) + .catch(() => + window.showErrorMessage("deno.land/x module list failed to cache!") + ); + this.sync(window.activeTextEditor?.document); const extension = extensions.getExtension(this.id); @@ -607,6 +640,8 @@ Executable ${this.denoInfo.executablePath}`; public async deactivate(context: ExtensionContext): Promise { this.context = context; + this.import_enhancement_completion_provider.dispose(); + if (this.client) { await this.client.stop(); this.client = undefined; diff --git a/client/src/import_enhancement_provider.ts b/client/src/import_enhancement_provider.ts new file mode 100644 index 00000000..f18a0df9 --- /dev/null +++ b/client/src/import_enhancement_provider.ts @@ -0,0 +1,255 @@ +import { + CompletionItemProvider, + TextDocument, + Position, + CompletionItem, + Disposable, + CompletionItemKind, + CompletionList, + DocumentSelector, + languages, + ExtensionContext, + Range, + Command, + window, + ProgressLocation, +} from "vscode"; + +import Semver from "semver"; + +import VC = require("vscode-cache"); + +import { + listVersionsOfMod, + modTreeOf, + parseImportStatement, + searchX, + fetchModList, + ModuleInfo, +} from "./import_utils"; + +export enum CACHE_STATE { + ALREADY_CACHED, + CACHE_SUCCESS, +} + +export class ImportEnhancementCompletionProvider + implements CompletionItemProvider, Disposable { + vc?: VC; + async provideCompletionItems( + document: TextDocument, + position: Position + // _token: CancellationToken, + // _context: CompletionContext + ): Promise { + const line_text = document.lineAt(position).text; + + if (/import.+?from\W+['"].*?['"]/.test(line_text)) { + // We're at import statement line + const imp_info = parseImportStatement(line_text); + if (imp_info?.domain !== "deno.land") { + return undefined; + } + // We'll handle the completion only if the domain is `deno.land` and mod name is not empty + const at_index = line_text.indexOf("@"); + if ( + /.*?deno.land\/(x\/)?\w+@[\w.-]*$/.test( + line_text.substring(0, position.character) + ) && + position.character > at_index + ) { + // Version completion + const vers = await listVersionsOfMod(imp_info.module); + + const result = vers.versions + .sort((a, b) => { + const av = Semver.clean(a); + const bv = Semver.clean(b); + if ( + av === null || + bv === null || + !Semver.valid(av) || + !Semver.valid(bv) + ) { + return 0; + } + return Semver.gt(av, bv) ? -1 : 1; + }) + .map((it, i) => { + // let latest version on top + const ci = new CompletionItem(it, CompletionItemKind.Value); + ci.sortText = `a${String.fromCharCode(i) + 1}`; + ci.filterText = it; + ci.range = new Range( + position.line, + at_index + 1, + position.line, + position.character + ); + return ci; + }); + return new CompletionList(result); + } + + if ( + /.*?deno\.land\/x\/\w*$/.test( + line_text.substring(line_text.indexOf("'") + 1, position.character) + ) + ) { + // x module name completion + if (this.vc !== undefined) { + const result: { name: string; description: string }[] = await searchX( + this.vc, + imp_info.module + ); + const r = result.map((it) => { + const ci = new CompletionItem(it.name, CompletionItemKind.Module); + ci.detail = it.description; + ci.sortText = String.fromCharCode(1); + ci.filterText = it.name; + return ci; + }); + return r; + } else { + return []; + } + } + + if ( + !/.*?deno\.land\/(x\/)?\w+(@[\w.-]*)?\//.test( + line_text.substring(0, position.character) + ) + ) { + return []; + } + + const result = await modTreeOf( + this.vc, + imp_info.module, + imp_info.version + ); + const arr_path = imp_info.path.split("/"); + const path = arr_path.slice(0, arr_path.length - 1).join("/") + "/"; + + const r = result.directory_listing + .filter((it) => it.path.startsWith(path)) + .map((it) => ({ + path: + path.length > 1 ? it.path.replace(path, "") : it.path.substring(1), + size: it.size, + type: it.type, + })) + .filter((it) => it.path.split("/").length < 2) + .filter( + (it) => + // exclude tests + !(it.path.endsWith("_test.ts") || it.path.endsWith("_test.js")) && + // include only js and ts + (it.path.endsWith(".ts") || + it.path.endsWith(".js") || + it.path.endsWith(".tsx") || + it.path.endsWith(".jsx") || + it.path.endsWith(".mjs") || + it.type !== "file") && + // exclude privates + !it.path.startsWith("_") && + // exclude hidden file/folder + !it.path.startsWith(".") && + // exclude testdata dir + (it.path !== "testdata" || it.type !== "dir") && + it.path.length !== 0 + ) + // .sort((a, b) => a.path.length - b.path.length) + .map((it) => { + const r = new CompletionItem( + it.path, + it.type === "dir" + ? CompletionItemKind.Folder + : CompletionItemKind.File + ); + r.sortText = it.type === "dir" ? "a" : "b"; + r.insertText = it.type === "dir" ? it.path + "/" : it.path; + r.range = new Range( + position.line, + line_text.substring(0, position.character).lastIndexOf("/") + 1, + position.line, + position.character + ); + if (it.type === "dir") { + // https://github.com/microsoft/vscode-extension-samples/blob/bb4a0c3a5dd9460a5cd64290b4d5c4f6bd79bdc4/completions-sample/src/extension.ts#L37 + r.command = { + command: "editor.action.triggerSuggest", + title: "Re-trigger completions...", + }; + } + return r; + }); + return new CompletionList(r, false); + } + } + + async clearCache(): Promise { + await this.vc?.flush(); + } + + async cacheModList(): Promise { + return window.withProgress( + { + location: ProgressLocation.Notification, + title: "Fetching deno.land/x module list...", + }, + async (progress) => { + const mod_list_key = "mod_list"; + if (this.vc?.isExpired(mod_list_key) || !this.vc?.has(mod_list_key)) { + this.vc?.forget(mod_list_key); + progress.report({ increment: 0 }); + for await (const modules of fetchModList()) { + if (this.vc?.has(mod_list_key)) { + const list_in_cache = this.vc?.get(mod_list_key) as ModuleInfo[]; + list_in_cache.push(...modules.data); + await this.vc?.put( + mod_list_key, + list_in_cache, + 60 * 60 * 24 * 7 /* expiration in a week */ + ); + } else { + this.vc?.put( + mod_list_key, + modules.data, + 60 * 60 * 24 * 7 /* expiration in a week */ + ); + } + progress.report({ + increment: (1 / modules.total) * 100, + }); + } + return CACHE_STATE.CACHE_SUCCESS; + } + return CACHE_STATE.ALREADY_CACHED; + } + ); + } + + activate(ctx: ExtensionContext): void { + this.vc = new VC(ctx, "import-enhanced"); + + const document_selector = [ + { language: "javascript" }, + { language: "typescript" }, + { language: "javascriptreact" }, + { language: "typescriptreact" }, + ]; + const trigger_word = ["@", "/"]; + ctx.subscriptions.push( + languages.registerCompletionItemProvider( + document_selector, + this, + ...trigger_word + ) + ); + } + + dispose(): void { + /* eslint-disable */ + } +} diff --git a/client/src/import_utils.test.ts b/client/src/import_utils.test.ts new file mode 100644 index 00000000..3f92cc35 --- /dev/null +++ b/client/src/import_utils.test.ts @@ -0,0 +1,106 @@ +import { + listVersionsOfMod, + modTreeOf, + parseImportStatement, +} from "./import_utils"; + +test("import-enhance listVersionsOfMod", async () => { + const result = await listVersionsOfMod("std"); + expect(result.latest).toBeTruthy(); + expect(result.versions.length).not.toEqual(0); +}); + +test("import-enhance modTreeOf", async () => { + const result = await modTreeOf(undefined, "std"); + expect(result.uploaded_at).toBeTruthy(); + expect(result.directory_listing.length).not.toEqual(0); +}); + +test("import-enhance parseImportStatement", async () => { + const test_cases = [ + { + imp: "import * from 'http://a.c/xx/a.ts'", + expect: { + domain: "a.c", + module: "xx", + version: "latest", + path: "/a.ts", + }, + }, + { + imp: "import * from 'http://deno.land/std@0.66.0/fs/copy.ts'", + expect: { + domain: "deno.land", + module: "std", + version: "0.66.0", + path: "/fs/copy.ts", + }, + }, + { + imp: "import * from 'https://deno.land/x/sha2@1.0.0/mod/sha224.ts'", + expect: { + domain: "deno.land", + module: "sha2", + version: "1.0.0", + path: "/mod/sha224.ts", + }, + }, + { + imp: "import {} from 'https://deno.land/std@/';", + expect: { + domain: "deno.land", + module: "std", + version: "latest", + path: "/", + }, + }, + // non semver verions + { + imp: "import {} from 'https://deno.land/std@1.0.0-alpha/';", + expect: { + domain: "deno.land", + module: "std", + version: "1.0.0-alpha", + path: "/", + }, + }, + { + imp: "import {} from 'https://deno.land/std@v1/';", + expect: { + domain: "deno.land", + module: "std", + version: "v1", + path: "/", + }, + }, + { + imp: "import {} from 'https://deno.land/x/@/';", + expect: { + domain: "deno.land", + module: "", + version: "latest", + path: "/", + }, + }, + { + imp: "import { } from 'https://deno.land/x/sq'", + expect: { + domain: "deno.land", + module: "sq", + version: "latest", + path: "/", + }, + }, + ] as { + imp: string; + expect: { domain: string; module: string; version: string; path: string }; + }[]; + + for (const test_case of test_cases) { + const result = parseImportStatement(test_case.imp); + expect(result?.domain).toEqual(test_case.expect.domain); + expect(result?.module).toEqual(test_case.expect.module); + expect(result?.version).toEqual(test_case.expect.version); + expect(result?.path).toEqual(test_case.expect.path); + } +}); diff --git a/client/src/import_utils.ts b/client/src/import_utils.ts new file mode 100644 index 00000000..8c75cd43 --- /dev/null +++ b/client/src/import_utils.ts @@ -0,0 +1,167 @@ +/* eslint @typescript-eslint/triple-slash-reference: "off" */ +// CGQAQ: Without next line the test will fail, using import won't work +/// + +import got from "got"; +import VC from "vscode-cache"; + +export interface ModuleInfo { + name: string; + description: string; + star_count: string; + search_score: number; +} + +export async function* fetchModList(): AsyncGenerator<{ + current: number; + total: number; + data: ModuleInfo[]; +}> { + // https://api.deno.land/modules?limit=100&query=$QUERY + + let response: { + success: boolean; + data: { + total_count: number; + results: ModuleInfo[]; + }; + }; + let page = 1; + do { + response = await got( + `https://api.deno.land/modules?limit=100&page=${page}` + ).json(); + if (Array.isArray(response.data.results)) { + yield { + current: page, + total: Math.ceil(response.data.total_count / 100), + data: response.data.results, + }; + } + page++; + } while (response.success && response.data.results.length > 0); +} + +// this function now is search from cache only +export async function searchX( + cache: VC, + keyword: string +): Promise { + if (cache.has("mod_list")) { + const buff = cache.get("mod_list") as ModuleInfo[]; + return buff + .filter((it) => it.name.startsWith(keyword)) + .sort((a, b) => b.search_score - a.search_score); + } else { + return []; + } +} + +interface ModVersions { + latest: string; + versions: string[]; +} +export async function listVersionsOfMod( + module_name: string +): Promise { + // https://cdn.deno.land/$MODULE/meta/versions.json + const response: ModVersions = await got( + `https://cdn.deno.land/${encodeURIComponent( + module_name + )}/meta/versions.json` + ).json(); + return response; +} + +interface ModTreeItem { + path: string; + size: number; + type: string; +} + +interface ModTree { + uploaded_at: string; // Use this to update cache + directory_listing: ModTreeItem[]; +} +export async function modTreeOf( + vc: VC | undefined, + module_name: string, + version = "latest" +): Promise { + // https://cdn.deno.land/$MODULE/versions/$VERSION/meta/meta.json + let ver = version; + if (ver === "latest") { + const vers = await listVersionsOfMod(module_name); + ver = vers.latest; + } + + const cache_key = `mod_tree:${module_name}@${ver}`; + if (vc?.has(cache_key)) { + // use cache + return vc.get(cache_key) as ModTree; + } + + const response: ModTree = await got( + `https://cdn.deno.land/${encodeURIComponent( + module_name + )}/versions/${ver}/meta/meta.json` + ).json(); + + // cache it + vc?.put(cache_key, response); + + return response; +} + +interface ImportUrlInfo { + domain: string; + module: string; + version: string; + path: string; +} + +export function parseImportStatement(text: string): ImportUrlInfo | undefined { + const reg_groups = text.match(/.*?['"](?.*?)['"]/)?.groups; + if (!reg_groups) { + return undefined; + } + const import_url = reg_groups["url"] ?? ""; + try { + const url = new URL(import_url); + const components = url.pathname.split("/"); + const parse = (components: string[]) => { + const module_info = components[0].split("@"); + if (module_info.length > 1) { + const module = module_info[0]; + const version = module_info[1].length === 0 ? "latest" : module_info[1]; + const path = "/" + components.slice(1, components.length).join("/"); + return { + domain: url.hostname, + module, + version, + path, + }; + } else { + const module = module_info[0]; + const version = "latest"; + const path = "/" + components.slice(1, components.length).join("/"); + return { + domain: url.hostname, + module, + version, + path, + }; + } + }; + if (components.length > 2) { + const m = components[1]; + if (m === "x") { + return parse(components.slice(2, components.length)); + } else { + return parse(components.slice(1, components.length)); + } + } + } catch { + return undefined; + } +} diff --git a/client/src/types/vscode-cache.d.ts b/client/src/types/vscode-cache.d.ts new file mode 100644 index 00000000..48279d11 --- /dev/null +++ b/client/src/types/vscode-cache.d.ts @@ -0,0 +1,27 @@ +/* eslint-disable */ +declare module "vscode-cache" { + export = Cache; + class Cache { + constructor(context: any, namespace?: string); + put(key: string, value: any, expiration?: number): Promise; + set(key: any, value: any, expiration: any): Promise; + save(key: any, value: any, expiration: any): Promise; + store(key: any, value: any, expiration: any): Promise; + cache: (key: any, value: any, expiration: any) => Promise; + get(key: string, defaultValue?: any): any; + fetch(key: any, defaultValue: any): any; + retrieve(key: any, defaultValue: any): any; + has(key: string): boolean; + exists(key: any): boolean; + forget(key: string): any; + remove(key: any): any; + delete(key: any): any; + keys(): string[]; + all(): object; + getAll(): object; + flush(): any; + clearAll(): any; + getExpiration(key: string): number; + isExpired(key: any): boolean; + } +} diff --git a/package.json b/package.json index 4a217571..d6063fee 100644 --- a/package.json +++ b/package.json @@ -54,6 +54,11 @@ "light": "resource/icon/refresh.light.svg", "dark": "resource/icon/refresh.dark.svg" } + }, + { + "command": "deno._clear_import_enhencement_cache", + "title": "Clear import enhencement cache", + "category": "deno" } ], "views": { @@ -134,6 +139,24 @@ ], "url": "./schemas/import_map.schema.json" } + ], + "snippets": [ + { + "language": "javascript", + "path": "snippets/import_enhancement.json" + }, + { + "language": "typescript", + "path": "snippets/import_enhancement.json" + }, + { + "language": "javascriptreact", + "path": "snippets/import_enhancement.json" + }, + { + "language": "typescriptreact", + "path": "snippets/import_enhancement.json" + } ] }, "scripts": { @@ -146,6 +169,7 @@ "format": "prettier \"**/*.md\" \"**/*.json\" \"**/*.ts\" \"**/*.yml\" --config ./.prettierrc.json --write", "test": "jest --coverage", "test-coveralls": "jest --coverage --coverageReporters=text-lcov | coveralls", + "test.import-enhance": "jest -t 'import-enhance'", "build": "npx vsce package -o ./vscode-deno.vsix", "lint": "eslint . --ext .js,.jsx,.ts,.tsx" }, @@ -171,8 +195,10 @@ "dependencies": { "deep-equal": "2.0.3", "deepmerge": "^4.2.2", + "got": "^11.5.2", "json5": "^2.1.3", + "semver": "7.3.2", "typescript-deno-plugin": "./typescript-deno-plugin", - "semver": "7.3.2" + "vscode-cache": "^0.3.0" } } diff --git a/snippets/import_enhancement.json b/snippets/import_enhancement.json new file mode 100644 index 00000000..c8cbf1a9 --- /dev/null +++ b/snippets/import_enhancement.json @@ -0,0 +1,14 @@ +{ + "Import deno.land/std": { + "scope": "javascript,typescript,javascriptreact,typescriptreact", + "prefix": ["impstd", "importstd"], + "body": "import {$3} from 'https://deno.land/std@$1/$2';$0", + "description": "Use this snippet to import a deno.land/std module." + }, + "Import deno.land/x module": { + "scope": "javascript,typescript,javascriptreact,typescriptreact", + "prefix": ["impx", "importx"], + "body": "import {$4} from 'https://deno.land/x/$1@$2/$3';$0", + "description": "Use this snippet to import a deno.land/x module." + } +} From dff16646a52dd5f17d7afb29dac08bba510ae563 Mon Sep 17 00:00:00 2001 From: Luca Casonato Date: Fri, 4 Sep 2020 16:43:34 +0200 Subject: [PATCH 4/7] v2.1.0 (#176) --- Releases.md | 9 +++++++++ package.json | 2 +- 2 files changed, 10 insertions(+), 1 deletion(-) diff --git a/Releases.md b/Releases.md index f8e9e1d2..6ce45c9c 100644 --- a/Releases.md +++ b/Releases.md @@ -2,6 +2,15 @@ Releases of the extension can be downloaded from [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=denoland.vscode-deno). +### [v2.1.0](https://github.com/denoland/vscode_deno/compare/v2.0.16...v2.1.0) / 2020.09.04 + +- feat: IntelliSense support for std and deno.land/x imports (#172) +- fix: add support for URLs with non default ports (#173) +- fix: correctly handle non existing \$DENO_DIR/deps (#169) +- refactor: simplify import map json validation (#167) +- chore: update dependencies (#165) +- docs: remove non english readme (#164) + ### [v2.0.16](https://github.com/denoland/vscode_deno/compare/v2.0.15...v2.0.16) / 2020.08.29 - fix: autocomplete supports adding extensions (#156) diff --git a/package.json b/package.json index d6063fee..474d00b8 100644 --- a/package.json +++ b/package.json @@ -2,7 +2,7 @@ "name": "vscode-deno", "displayName": "Deno", "description": "Deno support for VSCode", - "version": "2.0.16", + "version": "2.1.0", "publisher": "denoland", "icon": "deno.png", "galleryBanner": { From 9adc3a266dba1d0ac46ae21757ef3c3c97d13679 Mon Sep 17 00:00:00 2001 From: Luca Casonato Date: Fri, 4 Sep 2020 17:12:13 +0200 Subject: [PATCH 5/7] v2.1.1 (#177) --- Releases.md | 4 ++++ package.json | 2 +- typescript-deno-plugin/package.json | 3 ++- 3 files changed, 7 insertions(+), 2 deletions(-) diff --git a/Releases.md b/Releases.md index 6ce45c9c..a3217cc9 100644 --- a/Releases.md +++ b/Releases.md @@ -2,6 +2,10 @@ Releases of the extension can be downloaded from [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=denoland.vscode-deno). +### [v2.1.1](https://github.com/denoland/vscode_deno/compare/v2.1.0...v2.1.1) / 2020.09.04 + +- fix: typescript not found error (#177) + ### [v2.1.0](https://github.com/denoland/vscode_deno/compare/v2.0.16...v2.1.0) / 2020.09.04 - feat: IntelliSense support for std and deno.land/x imports (#172) diff --git a/package.json b/package.json index 474d00b8..a88e1a5b 100644 --- a/package.json +++ b/package.json @@ -2,7 +2,7 @@ "name": "vscode-deno", "displayName": "Deno", "description": "Deno support for VSCode", - "version": "2.1.0", + "version": "2.1.1", "publisher": "denoland", "icon": "deno.png", "galleryBanner": { diff --git a/typescript-deno-plugin/package.json b/typescript-deno-plugin/package.json index 964e843f..16c49140 100644 --- a/typescript-deno-plugin/package.json +++ b/typescript-deno-plugin/package.json @@ -10,6 +10,7 @@ }, "devDependencies": { "@types/deep-equal": "1.0.1", - "@types/json5": "0.0.30" + "@types/json5": "0.0.30", + "typescript": "4.0.2" } } From 714b945b61665d27e5bcdcef8e7116c20da33350 Mon Sep 17 00:00:00 2001 From: Luca Casonato Date: Fri, 4 Sep 2020 17:38:30 +0200 Subject: [PATCH 6/7] v2.1.2 (#178) --- Releases.md | 4 ++++ server/package.json | 1 + 2 files changed, 5 insertions(+) diff --git a/Releases.md b/Releases.md index a3217cc9..14162ada 100644 --- a/Releases.md +++ b/Releases.md @@ -2,6 +2,10 @@ Releases of the extension can be downloaded from [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=denoland.vscode-deno). +### [v2.1.2](https://github.com/denoland/vscode_deno/compare/v2.1.1...v2.1.2) / 2020.09.04 + +- fix: another typescript not found error (#178) + ### [v2.1.1](https://github.com/denoland/vscode_deno/compare/v2.1.0...v2.1.1) / 2020.09.04 - fix: typescript not found error (#177) diff --git a/server/package.json b/server/package.json index 99bd12c2..b43dc0f4 100644 --- a/server/package.json +++ b/server/package.json @@ -4,6 +4,7 @@ }, "dependencies": { "execa": "4.0.3", + "typescript": "4.0.2", "vscode-languageserver": "6.1.1", "vscode-languageserver-textdocument": "1.0.1", "vscode-uri": "2.1.2", From 0a70a93056fe2b23f8c0f95a9b8a399ee20d8c8f Mon Sep 17 00:00:00 2001 From: Luca Casonato Date: Fri, 4 Sep 2020 17:42:14 +0200 Subject: [PATCH 7/7] fix package.json version --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index a88e1a5b..770d2710 100644 --- a/package.json +++ b/package.json @@ -2,7 +2,7 @@ "name": "vscode-deno", "displayName": "Deno", "description": "Deno support for VSCode", - "version": "2.1.1", + "version": "2.1.2", "publisher": "denoland", "icon": "deno.png", "galleryBanner": {