Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Uses NODE_OPTIONS with PnP instead of CLI args #6629

Merged
merged 4 commits into from
Nov 5, 2018
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,14 @@ Please add one entry in this file for each change in Yarn's behavior. Use the sa

## Master

**Important:** This release contains a cache bump. It will cause the very first install following the upgrade to take slightly more time, especially if you don't use the [Offline Mirror](https://yarnpkg.com/blog/2016/11/24/offline-mirror/) feature. After that everything will be back to normal.

- Uses `NODE_OPTIONS` to instruct Node to load the PnP hook, instead of raw CLI arguments

**Caveat:** This change might cause issues for PnP users having a space inside their cwd (cf [nodejs/node#24065](https://github.com/nodejs/node/pull/24065))

[#6479](https://github.com/yarnpkg/yarn/pull/6629) - [**Maël Nison**](https://twitter.com/arcanis)

- Fixes Gulp when used with Plug'n'Play

[#6623](https://github.com/yarnpkg/yarn/pull/6623) - [**Maël Nison**](https://twitter.com/arcanis)
Expand Down
69 changes: 69 additions & 0 deletions packages/pkg-tests/pkg-tests-specs/sources/pnp.js
Original file line number Diff line number Diff line change
Expand Up @@ -1340,5 +1340,74 @@ module.exports = makeTemporaryEnv => {
},
),
);

test(
`it should not break spawning new Node processes ('node' command)`,
makeTemporaryEnv(
{
dependencies: {[`no-deps`]: `1.0.0`},
},
{plugNPlay: true},
async ({path, run, source}) => {
await run(`install`);

await writeFile(`${path}/script.js`, `console.log(JSON.stringify(require('no-deps')))`);

await expect(
source(
`JSON.parse(require('child_process').execFileSync(process.execPath, [${JSON.stringify(
`${path}/script.js`,
)}]).toString())`,
),
).resolves.toMatchObject({
name: `no-deps`,
version: `1.0.0`,
});
},
),
);

test(
`it should not break spawning new Node processes ('run' command)`,
makeTemporaryEnv(
{
dependencies: {[`no-deps`]: `1.0.0`},
scripts: {[`script`]: `node main.js`},
},
{plugNPlay: true},
async ({path, run, source}) => {
await run(`install`);

await writeFile(`${path}/sub.js`, `console.log(JSON.stringify(require('no-deps')))`);
await writeFile(
`${path}/main.js`,
`console.log(require('child_process').execFileSync(process.execPath, [${JSON.stringify(
`${path}/sub.js`,
)}]).toString())`,
);

expect(JSON.parse((await run(`run`, `script`)).stdout)).toMatchObject({
name: `no-deps`,
version: `1.0.0`,
});
},
),
);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we need two more tests:

  1. For cases where we spawn another script and purposefully omit node args which simulates the scenario you are solving for (is this the second test, I can't really tell?)
  2. Have an already set NODE_OPTIONS env variable and make sure it is preserved.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The first test tests yarn node my-script.js', and the second tests yarn run my-script. I think that should cover the first case? Will add a test for empty NODE_OPTIONS 👍


test(
`it should properly forward the NODE_OPTIONS environment variable`,
makeTemporaryEnv({}, {plugNPlay: true}, async ({path, run, source}) => {
await run(`install`);

await writeFile(`${path}/foo.js`, `console.log(42);`);

await expect(
run(`node`, `-e`, `console.log(21);`, {env: {NODE_OPTIONS: `--require ${path}/foo`}}),
).resolves.toMatchObject({
// Note that '42' is present twice: the first one because Node executes Yarn, and the second one because Yarn spawns Node
stdout: `42\n42\n21\n`,
});
}),
);
});
};
3 changes: 2 additions & 1 deletion packages/pkg-tests/yarn.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ const pkgDriver = generatePkgDriver({
async runDriver(
path,
[command, ...args],
{cwd, projectFolder, registryUrl, plugNPlay, plugnplayShebang, plugnplayBlacklist},
{cwd, projectFolder, registryUrl, plugNPlay, plugnplayShebang, plugnplayBlacklist, env},
) {
let beforeArgs = [];
let middleArgs = [];
Expand Down Expand Up @@ -49,6 +49,7 @@ const pkgDriver = generatePkgDriver({
[`PATH`]: `${path}/bin${delimiter}${process.env.PATH}`,
},
plugNPlay ? {[`YARN_PLUGNPLAY_OVERRIDE`]: plugNPlay ? `1` : `0`} : {},
env,
),
cwd: cwd || path,
},
Expand Down
4 changes: 3 additions & 1 deletion src/cli/commands/node.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,14 +21,16 @@ export function hasWrapper(commander: Object, args: Array<string>): boolean {
export async function run(config: Config, reporter: Reporter, flags: Object, args: Array<string>): Promise<void> {
const pnpPath = `${config.lockfileFolder}/${PNP_FILENAME}`;

let nodeOptions = process.env.NODE_OPTIONS || '';
if (await fs.exists(pnpPath)) {
args = ['-r', pnpPath, ...args];
nodeOptions += ` --require ${pnpPath}`;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm guessing the leading space wouldn't be an issue when NODE_OPTIONS is empty?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yep, will work 👍

Copy link
Contributor

@jdalton jdalton Jan 20, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👆 It looks like

nodeOptions += ` --require ${pnpPath}`

should be

nodeOptions = `--require ${pnpPath} ${nodeOptions}`

so that pnp is the first preloaded module.

Update:

Issue filed #6941; PR #6942.

}

try {
await child.spawn(NODE_BIN_PATH, args, {
stdio: 'inherit',
cwd: flags.into || config.cwd,
env: {...process.env, NODE_OPTIONS: nodeOptions},
});
} catch (err) {
throw err;
Expand Down
19 changes: 8 additions & 11 deletions src/util/execute-lifecycle-script.js
Original file line number Diff line number Diff line change
Expand Up @@ -26,14 +26,6 @@ export const IGNORE_MANIFEST_KEYS: Set<string> = new Set(['readme', 'notice', 'l
// See https://github.com/yarnpkg/yarn/issues/2286.
const IGNORE_CONFIG_KEYS = ['lastUpdateCheck'];

async function getPnpParameters(config: Config): Promise<Array<string>> {
if (await fs.exists(`${config.lockfileFolder}/${constants.PNP_FILENAME}`)) {
return ['-r', `${config.lockfileFolder}/${constants.PNP_FILENAME}`];
} else {
return [];
}
}

let wrappersFolder = null;

export async function getWrappersFolder(config: Config): Promise<string> {
Expand All @@ -45,7 +37,6 @@ export async function getWrappersFolder(config: Config): Promise<string> {

await makePortableProxyScript(process.execPath, wrappersFolder, {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Shall we unify this logic in the future and use node at all times?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What do you mean? At most we should replace makePortableScript with cmdShim (which already covers the behavior we're using here, I think)

proxyBasename: 'node',
prependArguments: [...(await getPnpParameters(config))],
});

await makePortableProxyScript(process.execPath, wrappersFolder, {
Expand Down Expand Up @@ -212,8 +203,9 @@ export async function makeEnv(
}
}

if (await fs.exists(`${config.lockfileFolder}/${constants.PNP_FILENAME}`)) {
const pnpApi = dynamicRequire(`${config.lockfileFolder}/${constants.PNP_FILENAME}`);
const pnpFile = `${config.lockfileFolder}/${constants.PNP_FILENAME}`;
if (await fs.exists(pnpFile)) {
const pnpApi = dynamicRequire(pnpFile);

const packageLocator = pnpApi.findPackageLocator(`${config.cwd}/`);
const packageInformation = pnpApi.getPackageInformation(packageLocator);
Expand All @@ -227,6 +219,11 @@ export async function makeEnv(

pathParts.unshift(`${dependencyInformation.packageLocation}/.bin`);
}

// Note that NODE_OPTIONS doesn't support any style of quoting its arguments at the moment
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Referring to your PR on Node would be nice. Also, would be great if we can centralize some of this PnP path logic.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think I'll keep it duplicated for now - since it's very few lines, I'm concerned about over-abstracting it (since it's only used in two places and there are no reasons to think it will change)

// For this reason, it won't work if the user has a space inside its $PATH
env.NODE_OPTIONS = env.NODE_OPTIONS || '';
env.NODE_OPTIONS += ` --require ${pnpFile}`;
}

pathParts.unshift(await getWrappersFolder(config));
Expand Down