Skip to content

Commit 1bd18f3

Browse files
authoredMar 26, 2021
lib: drop Python 2 support in find-python.js (#2333)
Co-authored-by: Christian Clauss <[email protected]> PR-URL: #2333 Reviewed-By: Christian Clauss <[email protected]> Reviewed-By: Jiawen Geng <[email protected]>
1 parent e81602e commit 1bd18f3

File tree

2 files changed

+36
-27
lines changed

2 files changed

+36
-27
lines changed
 

‎lib/find-python.js

+27-14
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,29 @@
11
'use strict'
22

3-
const path = require('path')
43
const log = require('npmlog')
54
const semver = require('semver')
65
const cp = require('child_process')
76
const extend = require('util')._extend // eslint-disable-line
87
const win = process.platform === 'win32'
98
const logWithPrefix = require('./util').logWithPrefix
109

10+
const systemDrive = process.env.SystemDrive || 'C:'
11+
const username = process.env.USERNAME || process.env.USER || require('os').userInfo().username
12+
const localAppData = process.env.LOCALAPPDATA || `${systemDrive}\\${username}\\AppData\\Local`
13+
const programFiles = process.env.ProgramW6432 || process.env.ProgramFiles || `${systemDrive}\\Program Files`
14+
const programFilesX86 = process.env['ProgramFiles(x86)'] || `${programFiles} (x86)`
15+
16+
const winDefaultLocationsArray = []
17+
for (const majorMinor of ['39', '38', '37', '36']) {
18+
winDefaultLocationsArray.push(
19+
`${localAppData}\\Programs\\Python\\Python${majorMinor}\\python.exe`,
20+
`${programFiles}\\Python${majorMinor}\\python.exe`,
21+
`${localAppData}\\Programs\\Python\\Python${majorMinor}-32\\python.exe`,
22+
`${programFiles}\\Python${majorMinor}-32\\python.exe`,
23+
`${programFilesX86}\\Python${majorMinor}-32\\python.exe`
24+
)
25+
}
26+
1127
function PythonFinder (configPython, callback) {
1228
this.callback = callback
1329
this.configPython = configPython
@@ -18,17 +34,14 @@ PythonFinder.prototype = {
1834
log: logWithPrefix(log, 'find Python'),
1935
argsExecutable: ['-c', 'import sys; print(sys.executable);'],
2036
argsVersion: ['-c', 'import sys; print("%s.%s.%s" % sys.version_info[:3]);'],
21-
semverRange: '2.7.x || >=3.5.0',
37+
semverRange: '>=3.6.0',
2238

2339
// These can be overridden for testing:
2440
execFile: cp.execFile,
2541
env: process.env,
2642
win: win,
2743
pyLauncher: 'py.exe',
28-
winDefaultLocations: [
29-
path.join(process.env.SystemDrive || 'C:', 'Python37', 'python.exe'),
30-
path.join(process.env.SystemDrive || 'C:', 'Python27', 'python.exe')
31-
],
44+
winDefaultLocations: winDefaultLocationsArray,
3245

3346
// Logs a message at verbose level, but also saves it to be displayed later
3447
// at error level if an error occurs. This should help diagnose the problem.
@@ -96,11 +109,6 @@ PythonFinder.prototype = {
96109
before: () => { this.addLog('checking if "python" can be used') },
97110
check: this.checkCommand,
98111
arg: 'python'
99-
},
100-
{
101-
before: () => { this.addLog('checking if "python2" can be used') },
102-
check: this.checkCommand,
103-
arg: 'python2'
104112
}
105113
]
106114

@@ -119,7 +127,7 @@ PythonFinder.prototype = {
119127
checks.push({
120128
before: () => {
121129
this.addLog(
122-
'checking if the py launcher can be used to find Python')
130+
'checking if the py launcher can be used to find Python 3')
123131
},
124132
check: this.checkPyLauncher
125133
})
@@ -188,10 +196,15 @@ PythonFinder.prototype = {
188196
// Distributions of Python on Windows by default install with the "py.exe"
189197
// Python launcher which is more likely to exist than the Python executable
190198
// being in the $PATH.
199+
// Because the Python launcher supports Python 2 and Python 3, we should
200+
// explicitly request a Python 3 version. This is done by supplying "-3" as
201+
// the first command line argument. Since "py.exe -3" would be an invalid
202+
// executable for "execFile", we have to use the launcher to figure out
203+
// where the actual "python.exe" executable is located.
191204
checkPyLauncher: function checkPyLauncher (errorCallback) {
192205
this.log.verbose(
193-
`- executing "${this.pyLauncher}" to get Python executable path`)
194-
this.run(this.pyLauncher, this.argsExecutable, false,
206+
`- executing "${this.pyLauncher}" to get Python 3 executable path`)
207+
this.run(this.pyLauncher, ['-3', ...this.argsExecutable], false,
195208
function (err, execPath) {
196209
// Possible outcomes: same as checkCommand
197210
if (err) {

‎test/test-find-python.js

+9-13
Original file line numberDiff line numberDiff line change
@@ -16,13 +16,8 @@ test('find python', function (t) {
1616
t.strictEqual(err, null)
1717
var proc = execFile(found, ['-V'], function (err, stdout, stderr) {
1818
t.strictEqual(err, null)
19-
if (/Python 2/.test(stderr)) {
20-
t.strictEqual(stdout, '')
21-
t.ok(/Python 2/.test(stderr))
22-
} else {
23-
t.ok(/Python 3/.test(stdout))
24-
t.strictEqual(stderr, '')
25-
}
19+
t.ok(/Python 3/.test(stdout))
20+
t.strictEqual(stderr, '')
2621
})
2722
proc.stdout.setEncoding('utf-8')
2823
proc.stderr.setEncoding('utf-8')
@@ -66,7 +61,7 @@ test('find python - python', function (t) {
6661
poison(f, 'execFile')
6762
t.strictEqual(program, '/path/python')
6863
t.ok(/sys\.version_info/.test(args[1]))
69-
cb(null, '2.7.15')
64+
cb(null, '3.9.1')
7065
}
7166
t.strictEqual(program,
7267
process.platform === 'win32' ? '"python"' : 'python')
@@ -146,13 +141,14 @@ test('find python - no python2, no python, unix', function (t) {
146141
})
147142

148143
test('find python - no python, use python launcher', function (t) {
149-
t.plan(3)
144+
t.plan(4)
150145

151146
var f = new TestPythonFinder(null, done)
152147
f.win = true
153148

154149
f.execFile = function (program, args, opts, cb) {
155150
if (program === 'py.exe') {
151+
t.notEqual(args.indexOf('-3'), -1)
156152
t.notEqual(args.indexOf('-c'), -1)
157153
return cb(null, 'Z:\\snake.exe')
158154
}
@@ -162,7 +158,7 @@ test('find python - no python, use python launcher', function (t) {
162158
cb(new Error('not found'))
163159
} else if (/sys\.version_info/.test(args[args.length - 1])) {
164160
if (program === 'Z:\\snake.exe') {
165-
cb(null, '2.7.14')
161+
cb(null, '3.9.0')
166162
} else {
167163
t.fail()
168164
}
@@ -181,17 +177,17 @@ test('find python - no python, use python launcher', function (t) {
181177
test('find python - no python, no python launcher, good guess', function (t) {
182178
t.plan(2)
183179

184-
var re = /C:[\\/]Python37[\\/]python[.]exe/
185180
var f = new TestPythonFinder(null, done)
186181
f.win = true
182+
const expectedProgram = f.winDefaultLocations[0]
187183

188184
f.execFile = function (program, args, opts, cb) {
189185
if (program === 'py.exe') {
190186
return cb(new Error('not found'))
191187
}
192188
if (/sys\.executable/.test(args[args.length - 1])) {
193189
cb(new Error('not found'))
194-
} else if (re.test(program) &&
190+
} else if (program === expectedProgram &&
195191
/sys\.version_info/.test(args[args.length - 1])) {
196192
cb(null, '3.7.3')
197193
} else {
@@ -202,7 +198,7 @@ test('find python - no python, no python launcher, good guess', function (t) {
202198

203199
function done (err, python) {
204200
t.strictEqual(err, null)
205-
t.ok(re.test(python))
201+
t.ok(python === expectedProgram)
206202
}
207203
})
208204

0 commit comments

Comments
 (0)