1
1
'use strict'
2
2
3
- const path = require ( 'path' )
4
3
const log = require ( 'npmlog' )
5
4
const semver = require ( 'semver' )
6
5
const cp = require ( 'child_process' )
7
6
const extend = require ( 'util' ) . _extend // eslint-disable-line
8
7
const win = process . platform === 'win32'
9
8
const logWithPrefix = require ( './util' ) . logWithPrefix
10
9
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
+
11
27
function PythonFinder ( configPython , callback ) {
12
28
this . callback = callback
13
29
this . configPython = configPython
@@ -18,17 +34,14 @@ PythonFinder.prototype = {
18
34
log : logWithPrefix ( log , 'find Python' ) ,
19
35
argsExecutable : [ '-c' , 'import sys; print(sys.executable);' ] ,
20
36
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' ,
22
38
23
39
// These can be overridden for testing:
24
40
execFile : cp . execFile ,
25
41
env : process . env ,
26
42
win : win ,
27
43
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 ,
32
45
33
46
// Logs a message at verbose level, but also saves it to be displayed later
34
47
// at error level if an error occurs. This should help diagnose the problem.
@@ -96,11 +109,6 @@ PythonFinder.prototype = {
96
109
before : ( ) => { this . addLog ( 'checking if "python" can be used' ) } ,
97
110
check : this . checkCommand ,
98
111
arg : 'python'
99
- } ,
100
- {
101
- before : ( ) => { this . addLog ( 'checking if "python2" can be used' ) } ,
102
- check : this . checkCommand ,
103
- arg : 'python2'
104
112
}
105
113
]
106
114
@@ -119,7 +127,7 @@ PythonFinder.prototype = {
119
127
checks . push ( {
120
128
before : ( ) => {
121
129
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 ' )
123
131
} ,
124
132
check : this . checkPyLauncher
125
133
} )
@@ -188,10 +196,15 @@ PythonFinder.prototype = {
188
196
// Distributions of Python on Windows by default install with the "py.exe"
189
197
// Python launcher which is more likely to exist than the Python executable
190
198
// 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.
191
204
checkPyLauncher : function checkPyLauncher ( errorCallback ) {
192
205
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 ,
195
208
function ( err , execPath ) {
196
209
// Possible outcomes: same as checkCommand
197
210
if ( err ) {
0 commit comments