Skip to content

Commit e88214c

Browse files
authored
Merge pull request davidmarkclements#168 from BridgeAR/use-same-node-version
Make sure the same Node.js executable is used
2 parents 086213e + cedd18c commit e88214c

File tree

5 files changed

+71
-55
lines changed

5 files changed

+71
-55
lines changed

cmd.js

+47-32
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ const { version } = require('./package.json')
1313

1414
const defaultBanner = `
1515
0x ${version}
16-
16+
1717
0x [flags] -- node [nodeFlags] script.js [scriptFlags]
1818
1919
`
@@ -24,10 +24,21 @@ if (module.parent === null) {
2424
debug(err)
2525
process.exit(err.code || 1)
2626
})
27-
} else module.exports = cmd
27+
} else {
28+
module.exports = cmd
29+
}
2830

2931
async function cmd (argv, banner = defaultBanner) {
30-
var args = minimist(argv, {
32+
if (semver.lt(process.version, '8.5.0') === true) {
33+
throw Error(
34+
`Node version unsupported. Current Node version is ${process.version}\n` +
35+
'Support extends from Node 8.5.0 and above.\n\n' +
36+
' npm i -g 0x@3 for Node 6.x.x – 8.4.0\n' +
37+
' npm i -g 0x@2 for Node 4\n'
38+
)
39+
}
40+
41+
const args = minimist(argv, {
3142
stopEarly: true,
3243
'--': true,
3344
boolean: [
@@ -55,61 +66,65 @@ async function cmd (argv, banner = defaultBanner) {
5566
}
5667
})
5768

58-
if (semver.lt(process.version, '8.5.0') === true) {
59-
throw Error(
60-
'Node version unsupported. Current Node version is ' + process.version + '\n' +
61-
'Support extends from Node 8.5.0 and above.\n\n' +
62-
' npm i -g 0x@3 for Node 6.x.x – 8.4.0\n' +
63-
' npm i -g 0x@2 for Node 4\n'
64-
)
65-
}
66-
6769
if (args.help || argv.length === 0) {
6870
process.stdout.write(banner)
6971
return fs.createReadStream(join(__dirname, 'usage.txt')).pipe(process.stdout)
7072
}
7173

72-
if (args.version) return console.log('0x ' + version)
74+
if (args.version) {
75+
return console.log(`0x ${version}`)
76+
}
77+
7378
const status = createStatus(args)
79+
const { pathToNodeBinary, subprocessArgv } = parseSubprocessCommand(args)
80+
7481
args.workingDir = process.cwd()
7582
args.status = status
76-
const { pathToNodeBinary, subprocessArgv } = parseSubprocessCommand(args)
7783
args.argv = subprocessArgv
7884
args.pathToNodeBinary = pathToNodeBinary
7985

80-
if (args.visualizeOnly) status(`Creating flamegraph from ${args.visualizeOnly}`)
86+
if (args.visualizeOnly) {
87+
status(`Creating flamegraph from ${args.visualizeOnly}`)
88+
}
8189

8290
const assetPath = await zeroEks(args)
8391

84-
if (args.collectOnly) status(`Stats collected in folder file://${assetPath}\n`)
85-
else {
86-
status('Flamegraph generated in\n' + assetPath + '\n')
87-
if (args.open) launch(assetPath, {wait: false})
92+
if (args.collectOnly) {
93+
status(`Stats collected in folder file://${assetPath}\n`)
94+
} else {
95+
status(`Flamegraph generated in\n${assetPath}\n`)
96+
if (args.open) {
97+
launch(assetPath, { wait: false })
98+
}
8899
}
89100

90101
return assetPath
91102
}
92103

93104
function parseSubprocessCommand (args) {
94105
const dashDash = args['--']
95-
if (dashDash[0] && dashDash[0][0] === '-') {
96-
throw Error(`The node binary must immediately follow double dash (--)
97-
0x [flags] -- node [nodeFlags] script.js [scriptFlags]
98-
`)
99-
}
100-
var pathToNodeBinary = false
101-
var subprocessArgv = args._
102-
if (dashDash[0]) {
103-
if (dashDash[0][0] !== 'node') pathToNodeBinary = dashDash[0]
106+
let pathToNodeBinary = process.argv[0]
107+
let subprocessArgv = args._
108+
109+
if (dashDash.length !== 0) {
110+
const dashEntry = dashDash[0][0];
111+
if (dashEntry === '-') {
112+
throw Error(
113+
'The node binary must immediately follow double dash (--)\n' +
114+
'0x [flags] -- node [nodeFlags] script.js [scriptFlags]'
115+
)
116+
}
117+
pathToNodeBinary = dashDash[0]
104118
dashDash.shift()
105119
subprocessArgv = dashDash
106120
}
107121
return { pathToNodeBinary, subprocessArgv }
108122
}
109123

110-
function createStatus ({silent, quiet}) {
111-
const statusStream = process.stderr
112-
if (quiet || silent) return () => {}
113-
const status = sll(statusStream)
124+
function createStatus ({ silent, quiet }) {
125+
if (quiet || silent) {
126+
return () => {}
127+
}
128+
const status = sll(process.stderr)
114129
return hasUnicode ? (s) => status(`🔥 ${s}`) : status
115130
}

index.js

+17-16
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ const { sun, linux, windows, v8 } = require('./platform')
44
const debug = require('debug')('0x')
55
const { join, isAbsolute, relative } = require('path')
66
const fs = require('fs')
7+
const { promisify } = require('util')
78
const validate = require('./lib/validate')(require('./schema.json'))
89
const traceStacksToTicks = require('./lib/trace-stacks-to-ticks')
910
const v8LogToTicks = require('./lib/v8-log-to-ticks')
@@ -13,6 +14,8 @@ const render = require('./lib/render')
1314
const platform = process.platform
1415
const { tidy, noop, isSudo } = require('./lib/util')
1516

17+
const fsWriteFilePromise = promisify(fs.writeFile)
18+
1619
module.exports = zeroEks
1720

1821
async function zeroEks (args) {
@@ -26,7 +29,7 @@ async function zeroEks (args) {
2629

2730
if (visualizeOnly) return visualize(args)
2831

29-
args.title = args.title || 'node ' + args.argv.join(' ')
32+
args.title = args.title || `node ${args.argv.join(' ')}`
3033
const binary = args.pathToNodeBinary
3134
var { ticks, pid, folder, inlined } = await startProcessAndCollectTraceData(args, binary)
3235

@@ -35,8 +38,6 @@ async function zeroEks (args) {
3538
fs.writeFileSync(`${folder}/stacks.${pid}.json`, JSON.stringify(tree, 0, 2))
3639
}
3740

38-
fs.writeFileSync(`${folder}/meta.json`, JSON.stringify({...args, inlined}))
39-
4041
if (collectOnly === true) {
4142
debug('collect-only mode bailing on rendering')
4243
tidy()
@@ -45,7 +46,10 @@ async function zeroEks (args) {
4546
}
4647

4748
try {
48-
const file = await generateFlamegraph({...args, ticks, inlined, pid, folder})
49+
const [file] = await Promise.all([
50+
generateFlamegraph({...args, ticks, inlined, pid, folder}),
51+
fsWriteFilePromise(`${folder}/meta.json`, JSON.stringify({...args, inlined}))
52+
])
4953
return file
5054
} catch (err) {
5155
tidy()
@@ -68,17 +72,12 @@ async function startProcessAndCollectTraceData (args, binary) {
6872
}
6973

7074
async function generateFlamegraph (opts) {
71-
try {
72-
const file = await render(opts)
73-
tidy()
74-
return file
75-
} catch (err) {
76-
tidy()
77-
throw err
78-
}
75+
const file = await render(opts)
76+
tidy()
77+
return file
7978
}
8079

81-
async function visualize ({ visualizeOnly, treeDebug, workingDir, title, mapFrames, phase, open, name }) {
80+
async function visualize ({ visualizeOnly, treeDebug, workingDir, title, mapFrames, open, name, pathToNodeBinary }) {
8281
try {
8382
const folder = isAbsolute(visualizeOnly)
8483
? relative(workingDir, visualizeOnly)
@@ -108,7 +107,7 @@ async function visualize ({ visualizeOnly, treeDebug, workingDir, title, mapFram
108107
name = name || meta.name
109108

110109
const ticks = (srcType === 'v8')
111-
? await v8LogToTicks(src)
110+
? await v8LogToTicks(src, pathToNodeBinary)
112111
: traceStacksToTicks(src)
113112

114113
if (treeDebug === true) {
@@ -134,8 +133,10 @@ async function visualize ({ visualizeOnly, treeDebug, workingDir, title, mapFram
134133
} catch (e) {
135134
if (e.code === 'ENOENT') {
136135
throw Error('Invalid data path provided (unable to access/does not exist)')
137-
} else if (e.code === 'ENOTDIR') {
136+
}
137+
if (e.code === 'ENOTDIR') {
138138
throw Error('Invalid data path provided (not a directory)')
139-
} else throw e
139+
}
140+
throw e
140141
}
141142
}

lib/v8-log-to-ticks.js

+4-4
Original file line numberDiff line numberDiff line change
@@ -9,18 +9,18 @@ const pump = require('pump')
99

1010
module.exports = v8LogToTicks
1111

12-
function v8LogToTicks (isolateLogPath) {
12+
function v8LogToTicks (isolateLogPath, node) {
1313
const isJson = extname(isolateLogPath) === '.json'
14-
const sp = isJson || spawn(process.argv[0], [
14+
const sp = isJson || spawn(node, [
1515
'--prof-process', '--preprocess', '-j', isolateLogPath
1616
], {stdio: ['ignore', 'pipe', 'pipe']})
1717
const close = isJson ? () => {} : () => sp.kill()
1818
const srcStream = isJson ? fs.createReadStream(isolateLogPath) : sp.stdout
1919
if (!isJson) {
20-
// unknown code state errors are mostly harmless,
20+
// unknown code state errors are mostly harmless,
2121
// and not fixable within 0x at the moment (because
2222
// it's an error in the V8 internal tick profiler)
23-
// ignore them.
23+
// ignore them.
2424
const ignore = /unknown code state:/
2525
pump(sp.stderr, through((errMsg, _, cb) => {
2626
if (ignore.test(errMsg)) cb()

platform/v8.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -113,7 +113,7 @@ async function v8 (args, binary) {
113113
const isolateLogPath = path.join(folder, isolateLog)
114114
await renameSafe(path.join(args.workingDir, isolateLog), isolateLogPath)
115115
return {
116-
ticks: await v8LogToTicks(isolateLogPath),
116+
ticks: await v8LogToTicks(isolateLogPath, node),
117117
inlined: inlined,
118118
pid: proc.pid,
119119
folder: folder

visualizer/html.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,8 @@ function html (opts) {
1919
<meta charset="utf-8">
2020
<style> ${tachyons} </style>
2121
<style>
22-
button:focus { outline:0; }
23-
rect:hover {opacity: 0.9}
22+
button:focus { outline:0; }
23+
rect:hover {opacity: 0.9}
2424
</style>
2525
</head>
2626
<body class='m0 bg-white sans-serif overflow-hidden'>

0 commit comments

Comments
 (0)