Skip to content

Commit 3f8afdb

Browse files
authoredFeb 21, 2025··
Modernize (#32)
* Use prettier instead of clang-format We're formatting TypeScript, not C/C++. * Modernize This bumps the version to 0.1.19 because the type of `AbortController` is not backwards compatible. - Remove obsolete dependencies. - abort-controller; `AbortController` is stable since Node 15.4.0. - source-map-support; `--enable-source-maps` is supported since Node 12.12.0. - Update dependencies. - Enable strict mode in `tsconfig.json`. - Use `which(..., {nothrow: true})` and distinguish between binary-not-in-PATH and binary-access-failed. Log an error. - Check for presence of clangd binary in zip archive before extracting. - Use native promises APIs instead of `promisify`. - Assign severity to all logs (no more `console.log`).
1 parent 37ee1af commit 3f8afdb

File tree

10 files changed

+2923
-2168
lines changed

10 files changed

+2923
-2168
lines changed
 

‎.github/workflows/ci.yml

+6-18
Original file line numberDiff line numberDiff line change
@@ -1,26 +1,14 @@
11
name: CI
22
on:
33
push:
4-
branches: [ master ]
4+
branches: [master]
55
pull_request:
6-
branches: [ master ]
6+
branches: [master]
77
jobs:
88
build:
99
runs-on: ubuntu-latest
1010
steps:
11-
- uses: actions/checkout@v4
12-
- run: npm ci
13-
- run: npm test
14-
- name: clang-format (diff)
15-
if: github.event_name == 'pull_request'
16-
env:
17-
BASE: ${{ github['base_ref'] }}
18-
run: |
19-
git fetch --no-tags --prune --depth=1 origin +refs/heads/$BASE:refs/remotes/origin/$BASE
20-
npm run git-clang-format -- --diff origin/$BASE |\
21-
awk -v s="^diff" '$0~s{r=1} 1; END{exit(r)}'
22-
- name: clang-format (project)
23-
if: github.event_name == 'push'
24-
run: |
25-
npm run git-clang-format -- --diff --commit $(git hash-object -t tree /dev/null) |\
26-
awk -v s="^diff" '$0~s{r=1} 1; END{exit(r)}'
11+
- uses: actions/checkout@v4
12+
- run: npm ci
13+
- run: npm test
14+
- run: npx prettier . --check

‎.gitignore

-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,2 @@
11
node_modules/
22
out/
3-

‎.prettierignore

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
out

‎.prettierrc

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
{
2+
"bracketSpacing": false,
3+
"singleQuote": true
4+
}

‎README.md

+1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,3 @@
11
# node-clangd
2+
23
Shared features of vscode-clangd and coc-clangd

‎package-lock.json

+2,483-1,841
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎package.json

+21-25
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,15 @@
11
{
22
"name": "@clangd/install",
3-
"version": "0.1.18",
3+
"version": "0.1.19",
44
"description": "Installing clangd binaries from editor plugins.",
55
"main": "out/src/index.js",
66
"files": [
77
"out/src/index.d.ts"
88
],
99
"scripts": {
1010
"compile": "tsc -watch -p ./",
11-
"test": "tsc -p ./ && tape -r source-map-support/register 'out/test/**/*.js' | tap-spec",
12-
"format": "clang-format -i --glob=\"{src,test}/*.ts\"",
13-
"git-clang-format": "git-clang-format",
11+
"test": "tsc -p ./ && NODE_OPTIONS='--enable-source-maps' tape 'out/test/**/*.js' | tap-spec",
12+
"format": "prettier . --write",
1413
"prepare": "tsc -p ./"
1514
},
1615
"repository": {
@@ -27,30 +26,27 @@
2726
"url": "https://github.com/clangd/node-clangd/issues"
2827
},
2928
"dependencies": {
30-
"abort-controller": "^3.0.0",
31-
"adm-zip": "^0.5.10",
32-
"node-fetch": "^2.6.0",
33-
"readdirp": "^3.4.0",
34-
"rimraf": "^3.0.2",
35-
"semver": "^7.3.2",
36-
"which": "^2.0.2"
29+
"adm-zip": "^0.5.14",
30+
"node-fetch": "^2.7.0",
31+
"readdirp": "^4.1.1",
32+
"rimraf": "^6.0.1",
33+
"semver": "^7.6.3",
34+
"which": "^5.0.0"
3735
},
3836
"devDependencies": {
39-
"@types/adm-zip": "^0.5.0",
40-
"@types/node": "^13.13.4",
41-
"@types/node-fetch": "^2.5.7",
42-
"@types/node-static": "^0.7.3",
43-
"@types/rimraf": "^3.0.0",
44-
"@types/semver": "^7.1.0",
45-
"@types/tape": "^4.13.0",
46-
"@types/tmp": "^0.2.0",
47-
"@types/which": "^1.3.2",
48-
"clang-format": "^1.4.0",
37+
"@types/adm-zip": "^0.5.5",
38+
"@types/node": "^22.0.2",
39+
"@types/node-fetch": "^2.6.11",
40+
"@types/node-static": "^0.7.11",
41+
"@types/semver": "^7.5.8",
42+
"@types/tape": "^5.6.4",
43+
"@types/tmp": "^0.2.6",
44+
"@types/which": "^3.0.4",
4945
"node-static": "^0.7.11",
50-
"source-map-support": "^0.5.19",
46+
"prettier": "^3.3.3",
5147
"tap-spec": "^5.0.0",
52-
"tape": "^5.0.0",
53-
"tmp-promise": "^2.1.0",
54-
"typescript": "^3.8.3"
48+
"tape": "^5.8.1",
49+
"tmp-promise": "^3.0.3",
50+
"typescript": "^5.5.4"
5551
}
5652
}

‎src/index.ts

+278-220
Large diffs are not rendered by default.

‎test/index.ts

+120-60
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ import * as http from 'http';
33
import * as nodeStatic from 'node-static';
44
import * as os from 'os';
55
import * as path from 'path';
6-
import * as tape from 'tape';
6+
import tape from 'tape';
77
import * as tmp from 'tmp-promise';
88

99
import * as install from '../src/index';
@@ -12,7 +12,7 @@ const oldClangd = process.cwd() + '/test/assets/fake-clangd-5/clangd';
1212
const newClangdV15 = process.cwd() + '/test/assets/fake-clangd-15/clangd';
1313
const newClangdV16 = process.cwd() + '/test/assets/fake-clangd-16/clangd';
1414
const unversionedClangd =
15-
process.cwd() + '/test/assets/fake-clangd-unversioned/clangd';
15+
process.cwd() + '/test/assets/fake-clangd-unversioned/clangd';
1616
const appleClangd = process.cwd() + '/test/assets/apple-clangd-5/clangd';
1717
const exactLdd = process.cwd() + '/test/assets/ldd/exact';
1818
const oldLdd = process.cwd() + '/test/assets/ldd/old';
@@ -49,9 +49,15 @@ class FakeUI {
4949
console.info(msg, url);
5050
}
5151

52-
promptReload() { this.event('promptReload'); }
53-
promptUpdate() { this.event('promptUpdate'); }
54-
promptInstall() { this.event('promptInstall'); }
52+
promptReload() {
53+
this.event('promptReload');
54+
}
55+
promptUpdate() {
56+
this.event('promptUpdate');
57+
}
58+
promptInstall() {
59+
this.event('promptInstall');
60+
}
5561
public shouldReuseValue = true;
5662
async shouldReuse() {
5763
this.event('shouldReuse');
@@ -62,86 +68,127 @@ class FakeUI {
6268
this.event('slow');
6369
return work;
6470
}
65-
progress<T>(_title: string, _cancel: any,
66-
work: (progress: (fraction: number) => void) => Promise<T>) {
71+
progress<T>(
72+
_title: string,
73+
_cancel: any,
74+
work: (progress: (fraction: number) => void) => Promise<T>,
75+
) {
6776
this.event('progress');
6877
return work((fraction) => console.log('progress% ', 100 * fraction));
6978
}
70-
localize(message: string, ...args: Array<string|number|boolean>): string {
79+
localize(message: string, ...args: Array<string | number | boolean>): string {
7180
let ret = message;
7281
for (const i in args) {
7382
ret.replace(`{${i}}`, args[i].toString());
7483
}
7584
return ret;
7685
}
77-
};
78-
79-
function test(name: string,
80-
body: (assert: tape.Test, ui: FakeUI) => Promise<any>) {
81-
tape(name, async (assert) => tmp.withDir(async dir => {
82-
const ui = new FakeUI(dir.path);
83-
const files = new nodeStatic.Server('test/assets/');
84-
return new Promise((resolve, _reject) => {
85-
const server = http.createServer((req, res) => {
86-
console.log('Fake github:', req.method, req.url);
87-
req.on('end', () => files.serve(req, res)).resume();
88-
})
89-
.listen(9999, '::', async () => {
90-
console.log('Fake github serving...');
91-
install.fakeGitHubReleaseURL(releases);
92-
install.fakeLddCommand(exactLdd);
93-
try {
94-
await body(assert, ui);
95-
} catch (e) {
96-
assert.fail(e);
97-
}
98-
console.log('Fake github stopping...');
99-
server.close();
100-
resolve();
101-
});
102-
});
103-
}, {unsafeCleanup: true}));
86+
}
87+
88+
function test(
89+
name: string,
90+
body: (assert: tape.Test, ui: FakeUI) => Promise<any>,
91+
) {
92+
tape(name, async (assert) =>
93+
tmp.withDir(
94+
async (dir) => {
95+
const ui = new FakeUI(dir.path);
96+
const files = new nodeStatic.Server('test/assets/');
97+
return new Promise((resolve, _reject) => {
98+
const server = http
99+
.createServer((req, res) => {
100+
console.log('Fake github:', req.method, req.url);
101+
req.on('end', () => files.serve(req, res)).resume();
102+
})
103+
.listen(9999, '::', async () => {
104+
console.log('Fake github serving...');
105+
install.fakeGitHubReleaseURL(releases);
106+
install.fakeLddCommand(exactLdd);
107+
try {
108+
await body(assert, ui);
109+
} catch (e) {
110+
assert.fail(e as string);
111+
}
112+
console.log('Fake github stopping...');
113+
server.close();
114+
resolve();
115+
});
116+
});
117+
},
118+
{unsafeCleanup: true},
119+
),
120+
);
104121
}
105122

106123
// Test the actual installation, typically the clangd.install command.
107124

108125
test('install', async (assert, ui) => {
109126
await install.installLatest(ui);
110127

111-
const installedClangd =
112-
path.join(ui.storagePath, 'install', '10.0', 'fake-clangd-10', 'clangd');
113-
assert.true(fs.existsSync(installedClangd),
114-
`Extracted clangd exists: ${installedClangd}`);
128+
const installedClangd = path.join(
129+
ui.storagePath,
130+
'install',
131+
'10.0',
132+
'fake-clangd-10',
133+
'clangd',
134+
);
135+
assert.true(
136+
fs.existsSync(installedClangd),
137+
`Extracted clangd exists: ${installedClangd}`,
138+
);
115139
assert.equal(ui.clangdPath, installedClangd);
116-
assert.deepEqual(
117-
ui.events, [/*download*/ 'progress', /*extract*/ 'slow', 'promptReload']);
140+
assert.deepEqual(ui.events, [
141+
/*download*/ 'progress',
142+
/*extract*/ 'slow',
143+
'promptReload',
144+
]);
118145
});
119146

120147
test('install: no binary for platform', async (assert, ui) => {
121148
install.fakeGitHubReleaseURL(incompatibleReleases);
122149
await install.installLatest(ui);
123150

124-
const installedClangd =
125-
path.join(ui.storagePath, 'install', '10.0', 'fake-clangd-10', 'clangd');
126-
assert.false(fs.existsSync(installedClangd),
127-
`Extracted clangd exists: ${installedClangd}`);
128-
assert.true(ui.clangdPath.endsWith('fake-clangd-5/clangd'),
129-
'clangdPath unmodified');
151+
const installedClangd = path.join(
152+
ui.storagePath,
153+
'install',
154+
'10.0',
155+
'fake-clangd-10',
156+
'clangd',
157+
);
158+
assert.false(
159+
fs.existsSync(installedClangd),
160+
`Extracted clangd exists: ${installedClangd}`,
161+
);
162+
assert.true(
163+
ui.clangdPath.endsWith('fake-clangd-5/clangd'),
164+
'clangdPath unmodified',
165+
);
130166
assert.deepEqual(ui.events, ['showHelp']);
131167
});
132168

133169
test('install: wrong url', async (assert, ui) => {
134170
install.fakeGitHubReleaseURL(wrongUrlReleases);
135171
await install.installLatest(ui);
136172

137-
const installedClangd =
138-
path.join(ui.storagePath, 'install', '10.0', 'fake-clangd-10', 'clangd');
139-
assert.false(fs.existsSync(installedClangd),
140-
`Extracted clangd exists: ${installedClangd}`);
141-
assert.true(ui.clangdPath.endsWith('fake-clangd-5/clangd'),
142-
'clangdPath unmodified');
143-
assert.deepEqual(ui.events,
144-
[/*download*/ 'progress', /*download-fails*/ 'showHelp']);
173+
const installedClangd = path.join(
174+
ui.storagePath,
175+
'install',
176+
'10.0',
177+
'fake-clangd-10',
178+
'clangd',
179+
);
180+
assert.false(
181+
fs.existsSync(installedClangd),
182+
`Extracted clangd exists: ${installedClangd}`,
183+
);
184+
assert.true(
185+
ui.clangdPath.endsWith('fake-clangd-5/clangd'),
186+
'clangdPath unmodified',
187+
);
188+
assert.deepEqual(ui.events, [
189+
/*download*/ 'progress',
190+
/*download-fails*/ 'showHelp',
191+
]);
145192
});
146193

147194
if (os.platform() == 'linux') {
@@ -177,8 +224,13 @@ test('install: reuse existing install', async (assert, ui) => {
177224
ui.shouldReuseValue = true;
178225
await install.installLatest(ui);
179226

180-
const installedClangd =
181-
path.join(ui.storagePath, 'install', '10.0', 'fake-clangd-10', 'clangd');
227+
const installedClangd = path.join(
228+
ui.storagePath,
229+
'install',
230+
'10.0',
231+
'fake-clangd-10',
232+
'clangd',
233+
);
182234
assert.false(fs.existsSync(installedClangd), 'Not extracted');
183235
assert.true(fs.existsSync(existingClangd), 'Not erased');
184236
assert.equal(existingClangd, ui.clangdPath, 'clangdPath is existing install');
@@ -194,13 +246,21 @@ test('install: overwrite existing install', async (assert, ui) => {
194246
ui.shouldReuseValue = false;
195247
await install.installLatest(ui);
196248

197-
const installedClangd =
198-
path.join(ui.storagePath, 'install', '10.0', 'fake-clangd-10', 'clangd');
249+
const installedClangd = path.join(
250+
ui.storagePath,
251+
'install',
252+
'10.0',
253+
'fake-clangd-10',
254+
'clangd',
255+
);
199256
assert.true(fs.existsSync(installedClangd), 'Extracted');
200257
assert.false(fs.existsSync(existingClangd), 'Erased');
201258
assert.equal(installedClangd, ui.clangdPath, 'clangdPath is new install');
202259
assert.deepEqual(ui.events, [
203-
'shouldReuse', /*download*/ 'progress', /*extract*/ 'slow', 'promptReload'
260+
'shouldReuse',
261+
/*download*/ 'progress',
262+
/*extract*/ 'slow',
263+
'promptReload',
204264
]);
205265
});
206266

‎tsconfig.json

+9-3
Original file line numberDiff line numberDiff line change
@@ -4,14 +4,20 @@
44
"target": "es6",
55
"outDir": "out",
66
"lib": [
7-
"es6", "es2015.core", "es2015.collection", "es2015.generator",
8-
"es2015.iterable", "es2015.promise", "es2015.symbol"
7+
"es6",
8+
"es2015.core",
9+
"es2015.collection",
10+
"es2015.generator",
11+
"es2015.iterable",
12+
"es2015.promise",
13+
"es2015.symbol"
914
],
1015
"sourceMap": true,
1116
"declaration": true,
1217
"rootDir": ".",
1318
"alwaysStrict": true,
14-
"strict": false,
19+
"strict": true,
20+
"esModuleInterop": true,
1521
"noEmitOnError": true,
1622
"noFallthroughCasesInSwitch": true,
1723
"noImplicitAny": true,

0 commit comments

Comments
 (0)
Please sign in to comment.