-
Notifications
You must be signed in to change notification settings - Fork 173
/
mkshims.ts
78 lines (60 loc) · 2.92 KB
/
mkshims.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
import cmdShim from '@zkochan/cmd-shim';
import fs from 'fs';
import path from 'path';
import {Engine} from './sources/Engine';
import {SupportedPackageManagerSet} from './sources/types';
const engine = new Engine();
const distDir = path.join(__dirname, `dist`);
const shimsDir = path.join(__dirname, `shims`);
const physicalNodewinDir = path.join(shimsDir, `nodewin`);
const virtualNodewinDir = path.join(physicalNodewinDir, `node_modules/corepack`);
fs.mkdirSync(distDir, {recursive: true});
fs.mkdirSync(shimsDir, {recursive: true});
fs.mkdirSync(physicalNodewinDir, {recursive: true});
async function main() {
const corepackPath = path.join(distDir, `corepack.js`);
fs.writeFileSync(corepackPath, [
`#!/usr/bin/env node`,
`require('./lib/corepack.cjs').runMain(process.argv.slice(2));`,
].join(`\n`));
fs.chmodSync(corepackPath, 0o755);
for (const packageManager of SupportedPackageManagerSet) {
const binSet = engine.getBinariesFor(packageManager);
for (const binaryName of binSet) {
const entryPath = path.join(distDir, `${binaryName}.js`);
const entryScript = [
`#!/usr/bin/env node`,
`require('./lib/corepack.cjs').runMain(['${binaryName}', ...process.argv.slice(2)]);`,
].join(`\n`);
fs.writeFileSync(entryPath, entryScript);
fs.chmodSync(entryPath, 0o755);
}
}
for (const entry of fs.readdirSync(distDir, {withFileTypes: true})) {
if (entry.isDirectory())
continue;
await cmdShim(path.join(distDir, entry.name), path.join(shimsDir, path.basename(entry.name, `.js`)), {createCmdFile: true});
}
// The Node distribution doesn't support symlinks, so they copy the shims into
// the target folder. Since our shims have relative paths, it doesn't work
// super well... To make this process easier, we ship with a set of shims
// compatible for this use case. Not great, but better than text replacement
// in batch scripts.
// Last note: cmdShim generates shims with relative paths, so it doesn't matter
// that the target files don't truly exist, as long as we mock the `stat` function.
const remapPath = (p: string) => path.resolve(__dirname, path.relative(virtualNodewinDir, p));
const easyStatFs = Object.assign(Object.create(fs), {
readFile: (p: string, encoding: BufferEncoding, cb: (err: any, str: string) => void) => fs.readFile(remapPath(p), encoding, cb),
stat: (p: string, cb: () => void) => fs.stat(remapPath(p), cb),
});
for (const entry of fs.readdirSync(distDir, {withFileTypes: true})) {
if (entry.isDirectory())
continue;
await cmdShim(path.join(virtualNodewinDir, `dist/${entry.name}`), path.join(physicalNodewinDir, path.basename(entry.name, `.js`)), {createCmdFile: true, fs: easyStatFs});
}
console.log(`All shims have been generated.`);
}
main().catch(err => {
console.log(err.stack);
process.exitCode = 1;
});