Skip to content

Commit 8c5ea63

Browse files
dmitry-shibanovMaxim Lobanov
and
Maxim Lobanov
authoredDec 17, 2020
Adding support for more PyPy versions and installing them on-flight (#168)
* add support to install pypy * resolved comments, update readme, add e2e tests. * resolve throw error * Add pypy unit tests to cover code * add tests * Update test-pypy.yml * Update test-python.yml * Update test-python.yml * Update README.md * fixing tests * change order Co-authored-by: Maxim Lobanov <[email protected]> * add pypy tests and fix issue with pypy-3-nightly Co-authored-by: Maxim Lobanov <[email protected]>
1 parent 2831efe commit 8c5ea63

14 files changed

+1896
-34
lines changed
 

‎.github/workflows/test-pypy.yml

+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
name: Validate PyPy e2e
2+
on:
3+
push:
4+
branches:
5+
- main
6+
paths-ignore:
7+
- '**.md'
8+
pull_request:
9+
paths-ignore:
10+
- '**.md'
11+
schedule:
12+
- cron: 30 3 * * *
13+
14+
jobs:
15+
setup-pypy:
16+
name: Setup PyPy ${{ matrix.pypy }} ${{ matrix.os }}
17+
runs-on: ${{ matrix.os }}
18+
strategy:
19+
fail-fast: false
20+
matrix:
21+
os: [macos-latest, windows-latest, ubuntu-18.04, ubuntu-20.04]
22+
pypy:
23+
- 'pypy-2.7'
24+
- 'pypy-3.6'
25+
- 'pypy-3.7'
26+
- 'pypy-2.7-v7.3.2'
27+
- 'pypy-3.6-v7.3.2'
28+
- 'pypy-3.7-v7.3.2'
29+
- 'pypy-3.6-v7.3.x'
30+
- 'pypy-3.7-v7.x'
31+
- 'pypy-3.6-v7.3.3rc1'
32+
- 'pypy-3.7-nightly'
33+
34+
steps:
35+
- name: Checkout
36+
uses: actions/checkout@v2
37+
38+
- name: setup-python ${{ matrix.pypy }}
39+
uses: ./
40+
with:
41+
python-version: ${{ matrix.pypy }}
42+
43+
- name: PyPy and Python version
44+
run: python --version
45+
46+
- name: Run simple code
47+
run: python -c 'import math; print(math.factorial(5))'

‎.github/workflows/test.yml ‎.github/workflows/test-python.yml

+7-7
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
name: Validate 'setup-python'
1+
name: Validate Python e2e
22
on:
33
push:
44
branches:
@@ -9,7 +9,7 @@ on:
99
paths-ignore:
1010
- '**.md'
1111
schedule:
12-
- cron: 0 0 * * *
12+
- cron: 30 3 * * *
1313

1414
jobs:
1515
default-version:
@@ -18,7 +18,7 @@ jobs:
1818
strategy:
1919
fail-fast: false
2020
matrix:
21-
os: [macos-latest, windows-latest, ubuntu-16.04, ubuntu-18.04]
21+
os: [macos-latest, windows-latest, ubuntu-16.04, ubuntu-18.04, ubuntu-20.04]
2222
steps:
2323
- name: Checkout
2424
uses: actions/checkout@v2
@@ -38,7 +38,7 @@ jobs:
3838
strategy:
3939
fail-fast: false
4040
matrix:
41-
os: [macos-latest, windows-latest, ubuntu-16.04, ubuntu-18.04]
41+
os: [macos-latest, windows-latest, ubuntu-16.04, ubuntu-18.04, ubuntu-20.04]
4242
python: [3.5.4, 3.6.7, 3.7.5, 3.8.1]
4343
steps:
4444
- name: Checkout
@@ -68,7 +68,7 @@ jobs:
6868
strategy:
6969
fail-fast: false
7070
matrix:
71-
os: [macos-latest, windows-latest, ubuntu-16.04, ubuntu-18.04]
71+
os: [macos-latest, windows-latest, ubuntu-16.04, ubuntu-18.04, ubuntu-20.04]
7272
steps:
7373
- name: Checkout
7474
uses: actions/checkout@v2
@@ -91,13 +91,13 @@ jobs:
9191
- name: Run simple code
9292
run: python -c 'import math; print(math.factorial(5))'
9393

94-
setup-pypy:
94+
setup-pypy-legacy:
9595
name: Setup PyPy ${{ matrix.os }}
9696
runs-on: ${{ matrix.os }}
9797
strategy:
9898
fail-fast: false
9999
matrix:
100-
os: [macos-latest, windows-latest, ubuntu-16.04, ubuntu-18.04]
100+
os: [macos-latest, windows-latest, ubuntu-16.04, ubuntu-18.04, ubuntu-20.04]
101101
steps:
102102
- name: Checkout
103103
uses: actions/checkout@v2

‎README.md

+53-4
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ This action sets up a Python environment for use in actions by:
1717
- Allows for pinning to a specific patch version of Python without the worry of it ever being removed or changed.
1818
- Automatic setup and download of Python packages if using a self-hosted runner.
1919
- Support for pre-release versions of Python.
20+
- Support for installing any version of PyPy on-flight
2021

2122
# Usage
2223

@@ -40,7 +41,7 @@ jobs:
4041
runs-on: ubuntu-latest
4142
strategy:
4243
matrix:
43-
python-version: [ '2.x', '3.x', 'pypy2', 'pypy3' ]
44+
python-version: [ '2.x', '3.x', 'pypy-2.7', 'pypy-3.6', 'pypy-3.7' ]
4445
name: Python ${{ matrix.python-version }} sample
4546
steps:
4647
- uses: actions/checkout@v2
@@ -60,7 +61,7 @@ jobs:
6061
strategy:
6162
matrix:
6263
os: [ubuntu-latest, macos-latest, windows-latest]
63-
python-version: [2.7, 3.6, 3.7, 3.8, pypy2, pypy3]
64+
python-version: [2.7, 3.6, 3.7, 3.8, pypy-2.7, pypy-3.6]
6465
exclude:
6566
- os: macos-latest
6667
python-version: 3.8
@@ -91,7 +92,6 @@ jobs:
9192
with:
9293
python-version: ${{ matrix.python-version }}
9394
- run: python my_script.py
94-
9595
```
9696
9797
Download and set up an accurate pre-release version of Python:
@@ -114,6 +114,27 @@ steps:
114114
- run: python my_script.py
115115
```
116116
117+
Download and set up PyPy:
118+
119+
```yaml
120+
jobs:
121+
build:
122+
runs-on: ubuntu-latest
123+
strategy:
124+
matrix:
125+
python-version:
126+
- pypy-3.6 # the latest available version of PyPy that supports Python 3.6
127+
- pypy-3.7 # the latest available version of PyPy that supports Python 3.7
128+
- pypy-3.7-v7.3.3 # Python 3.7 and PyPy 7.3.3
129+
steps:
130+
- uses: actions/checkout@v2
131+
- uses: actions/setup-python@v2
132+
with:
133+
python-version: ${{ matrix.python-version }}
134+
- run: python my_script.py
135+
```
136+
More details on PyPy syntax and examples of using preview / nightly versions of PyPy can be found in the [Available versions of PyPy](#available-versions-of-pypy) section.
137+
117138
# Getting started with Python + Actions
118139
119140
Check out our detailed guide on using [Python with GitHub Actions](https://help.github.com/en/actions/automating-your-workflow-with-github-actions/using-python-with-github-actions).
@@ -129,7 +150,21 @@ Check out our detailed guide on using [Python with GitHub Actions](https://help.
129150
- If the exact patch version doesn't matter to you, specifying just the major and minor version will get you the latest preinstalled patch version. In the previous example, the version spec `3.8` will use the `3.8.2` Python version found in the cache.
130151
- Downloadable Python versions from GitHub Releases ([actions/python-versions](https://github.com/actions/python-versions/releases)).
131152
- All available versions are listed in the [version-manifest.json](https://github.com/actions/python-versions/blob/main/versions-manifest.json) file.
132-
- If there is a specific version of Python that is not available, you can open an issue here.
153+
- If there is a specific version of Python that is not available, you can open an issue here
154+
155+
# Available versions of PyPy
156+
157+
`setup-python` is able to configure PyPy from two sources:
158+
159+
- Preinstalled versions of PyPy in the tools cache on GitHub-hosted runners
160+
- For detailed information regarding the available versions of PyPy that are installed see [Supported software](https://docs.github.com/en/actions/reference/specifications-for-github-hosted-runners#supported-software).
161+
- For the latest PyPy release, all versions of Python are cached.
162+
- Cache is updated with a 1-2 week delay. If you specify the PyPy version as `pypy-3.6`, the cached version will be used although a newer version is available. If you need to start using the recently released version right after release, you should specify the exact PyPy version using `pypy-3.6-v7.3.3`.
163+
164+
- Downloadable PyPy versions from the [official PyPy site](https://downloads.python.org/pypy/).
165+
- All available versions that we can download are listed in [versions.json](https://downloads.python.org/pypy/versions.json) file.
166+
- PyPy < 7.3.3 are not available to install on-flight.
167+
- If some versions are not available, you can open an issue in https://foss.heptapod.net/pypy/pypy/
133168

134169
# Hosted Tool Cache
135170

@@ -155,6 +190,20 @@ You should specify only a major and minor version if you are okay with the most
155190
- There will be a single patch version already installed on each runner for every minor version of Python that is supported.
156191
- The patch version that will be preinstalled, will generally be the latest and every time there is a new patch released, the older version that is preinstalled will be replaced.
157192
- Using the most recent patch version will result in a very quick setup since no downloads will be required since a locally installed version Python on the runner will be used.
193+
194+
# Specifying a PyPy version
195+
The version of PyPy should be specified in the format `pypy-<python_version>[-v<pypy_version>]`.
196+
The `<pypy_version>` parameter is optional and can be skipped. The latest version will be used in this case.
197+
198+
```
199+
pypy-3.6 # the latest available version of PyPy that supports Python 3.6
200+
pypy-3.7 # the latest available version of PyPy that supports Python 3.7
201+
pypy-2.7 # the latest available version of PyPy that supports Python 2.7
202+
pypy-3.7-v7.3.3 # Python 3.7 and PyPy 7.3.3
203+
pypy-3.7-v7.x # Python 3.7 and the latest available PyPy 7.x
204+
pypy-3.7-v7.3.3rc1 # Python 3.7 and preview version of PyPy
205+
pypy-3.7-nightly # Python 3.7 and nightly PyPy
206+
```
158207
159208
# Using `setup-python` with a self hosted runner
160209

‎__tests__/data/pypy.json

+494
Large diffs are not rendered by default.

‎__tests__/find-pypy.test.ts

+237
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,237 @@
1+
import fs from 'fs';
2+
3+
import * as utils from '../src/utils';
4+
import {HttpClient} from '@actions/http-client';
5+
import * as ifm from '@actions/http-client/interfaces';
6+
import * as tc from '@actions/tool-cache';
7+
import * as exec from '@actions/exec';
8+
9+
import * as path from 'path';
10+
import * as semver from 'semver';
11+
12+
import * as finder from '../src/find-pypy';
13+
import {
14+
IPyPyManifestRelease,
15+
IS_WINDOWS,
16+
validateVersion,
17+
getPyPyVersionFromPath
18+
} from '../src/utils';
19+
20+
const manifestData = require('./data/pypy.json');
21+
22+
let architecture: string;
23+
24+
if (IS_WINDOWS) {
25+
architecture = 'x86';
26+
} else {
27+
architecture = 'x64';
28+
}
29+
30+
const toolDir = path.join(__dirname, 'runner', 'tools');
31+
const tempDir = path.join(__dirname, 'runner', 'temp');
32+
33+
describe('parsePyPyVersion', () => {
34+
it.each([
35+
['pypy-3.6-v7.3.3', {pythonVersion: '3.6', pypyVersion: 'v7.3.3'}],
36+
['pypy-3.6-v7.3.x', {pythonVersion: '3.6', pypyVersion: 'v7.3.x'}],
37+
['pypy-3.6-v7.x', {pythonVersion: '3.6', pypyVersion: 'v7.x'}],
38+
['pypy-3.6', {pythonVersion: '3.6', pypyVersion: 'x'}],
39+
['pypy-3.6-nightly', {pythonVersion: '3.6', pypyVersion: 'nightly'}],
40+
['pypy-3.6-v7.3.3rc1', {pythonVersion: '3.6', pypyVersion: 'v7.3.3-rc.1'}]
41+
])('%s -> %s', (input, expected) => {
42+
expect(finder.parsePyPyVersion(input)).toEqual(expected);
43+
});
44+
45+
it('throw on invalid input', () => {
46+
expect(() => finder.parsePyPyVersion('pypy-')).toThrowError(
47+
"Invalid 'version' property for PyPy. PyPy version should be specified as 'pypy-<python-version>'. See README for examples and documentation."
48+
);
49+
});
50+
});
51+
52+
describe('getPyPyVersionFromPath', () => {
53+
it('/fake/toolcache/PyPy/3.6.5/x64 -> 3.6.5', () => {
54+
expect(getPyPyVersionFromPath('/fake/toolcache/PyPy/3.6.5/x64')).toEqual(
55+
'3.6.5'
56+
);
57+
});
58+
});
59+
60+
describe('findPyPyToolCache', () => {
61+
const actualPythonVersion = '3.6.17';
62+
const actualPyPyVersion = '7.5.4';
63+
const pypyPath = path.join('PyPy', actualPythonVersion, architecture);
64+
let tcFind: jest.SpyInstance;
65+
let spyReadExactPyPyVersion: jest.SpyInstance;
66+
67+
beforeEach(() => {
68+
tcFind = jest.spyOn(tc, 'find');
69+
tcFind.mockImplementation((toolname: string, pythonVersion: string) => {
70+
const semverVersion = new semver.Range(pythonVersion);
71+
return semver.satisfies(actualPythonVersion, semverVersion)
72+
? pypyPath
73+
: '';
74+
});
75+
76+
spyReadExactPyPyVersion = jest.spyOn(utils, 'readExactPyPyVersionFile');
77+
spyReadExactPyPyVersion.mockImplementation(() => actualPyPyVersion);
78+
});
79+
80+
afterEach(() => {
81+
jest.resetAllMocks();
82+
jest.clearAllMocks();
83+
jest.restoreAllMocks();
84+
});
85+
86+
it('PyPy exists on the path and versions are satisfied', () => {
87+
expect(finder.findPyPyToolCache('3.6.17', 'v7.5.4', architecture)).toEqual({
88+
installDir: pypyPath,
89+
resolvedPythonVersion: actualPythonVersion,
90+
resolvedPyPyVersion: actualPyPyVersion
91+
});
92+
});
93+
94+
it('PyPy exists on the path and versions are satisfied with semver', () => {
95+
expect(finder.findPyPyToolCache('3.6', 'v7.5.x', architecture)).toEqual({
96+
installDir: pypyPath,
97+
resolvedPythonVersion: actualPythonVersion,
98+
resolvedPyPyVersion: actualPyPyVersion
99+
});
100+
});
101+
102+
it("PyPy exists on the path, but Python version doesn't match", () => {
103+
expect(finder.findPyPyToolCache('3.7', 'v7.5.4', architecture)).toEqual({
104+
installDir: '',
105+
resolvedPythonVersion: '',
106+
resolvedPyPyVersion: ''
107+
});
108+
});
109+
110+
it("PyPy exists on the path, but PyPy version doesn't match", () => {
111+
expect(finder.findPyPyToolCache('3.6', 'v7.5.1', architecture)).toEqual({
112+
installDir: null,
113+
resolvedPythonVersion: '',
114+
resolvedPyPyVersion: ''
115+
});
116+
});
117+
});
118+
119+
describe('findPyPyVersion', () => {
120+
let tcFind: jest.SpyInstance;
121+
let spyExtractZip: jest.SpyInstance;
122+
let spyExtractTar: jest.SpyInstance;
123+
let spyHttpClient: jest.SpyInstance;
124+
let spyExistsSync: jest.SpyInstance;
125+
let spyExec: jest.SpyInstance;
126+
let spySymlinkSync: jest.SpyInstance;
127+
let spyDownloadTool: jest.SpyInstance;
128+
let spyReadExactPyPyVersion: jest.SpyInstance;
129+
let spyFsReadDir: jest.SpyInstance;
130+
let spyWriteExactPyPyVersionFile: jest.SpyInstance;
131+
let spyCacheDir: jest.SpyInstance;
132+
let spyChmodSync: jest.SpyInstance;
133+
134+
beforeEach(() => {
135+
tcFind = jest.spyOn(tc, 'find');
136+
tcFind.mockImplementation((tool: string, version: string) => {
137+
const semverRange = new semver.Range(version);
138+
let pypyPath = '';
139+
if (semver.satisfies('3.6.12', semverRange)) {
140+
pypyPath = path.join(toolDir, 'PyPy', '3.6.12', architecture);
141+
}
142+
return pypyPath;
143+
});
144+
145+
spyWriteExactPyPyVersionFile = jest.spyOn(
146+
utils,
147+
'writeExactPyPyVersionFile'
148+
);
149+
spyWriteExactPyPyVersionFile.mockImplementation(() => null);
150+
151+
spyReadExactPyPyVersion = jest.spyOn(utils, 'readExactPyPyVersionFile');
152+
spyReadExactPyPyVersion.mockImplementation(() => '7.3.3');
153+
154+
spyDownloadTool = jest.spyOn(tc, 'downloadTool');
155+
spyDownloadTool.mockImplementation(() => path.join(tempDir, 'PyPy'));
156+
157+
spyExtractZip = jest.spyOn(tc, 'extractZip');
158+
spyExtractZip.mockImplementation(() => tempDir);
159+
160+
spyExtractTar = jest.spyOn(tc, 'extractTar');
161+
spyExtractTar.mockImplementation(() => tempDir);
162+
163+
spyFsReadDir = jest.spyOn(fs, 'readdirSync');
164+
spyFsReadDir.mockImplementation((directory: string) => ['PyPyTest']);
165+
166+
spyHttpClient = jest.spyOn(HttpClient.prototype, 'getJson');
167+
spyHttpClient.mockImplementation(
168+
async (): Promise<ifm.ITypedResponse<IPyPyManifestRelease[]>> => {
169+
const result = JSON.stringify(manifestData);
170+
return {
171+
statusCode: 200,
172+
headers: {},
173+
result: JSON.parse(result) as IPyPyManifestRelease[]
174+
};
175+
}
176+
);
177+
178+
spyExec = jest.spyOn(exec, 'exec');
179+
spyExec.mockImplementation(() => undefined);
180+
181+
spySymlinkSync = jest.spyOn(fs, 'symlinkSync');
182+
spySymlinkSync.mockImplementation(() => undefined);
183+
184+
spyExistsSync = jest.spyOn(fs, 'existsSync');
185+
spyExistsSync.mockReturnValue(true);
186+
});
187+
188+
afterEach(() => {
189+
jest.resetAllMocks();
190+
jest.clearAllMocks();
191+
jest.restoreAllMocks();
192+
});
193+
194+
it('found PyPy in toolcache', async () => {
195+
await expect(
196+
finder.findPyPyVersion('pypy-3.6-v7.3.x', architecture)
197+
).resolves.toEqual({
198+
resolvedPythonVersion: '3.6.12',
199+
resolvedPyPyVersion: '7.3.3'
200+
});
201+
});
202+
203+
it('throw on invalid input format', async () => {
204+
await expect(
205+
finder.findPyPyVersion('pypy3.7-v7.3.x', architecture)
206+
).rejects.toThrow();
207+
});
208+
209+
it('throw on invalid input format pypy3.7-7.3.x', async () => {
210+
await expect(
211+
finder.findPyPyVersion('pypy3.7-v7.3.x', architecture)
212+
).rejects.toThrow();
213+
});
214+
215+
it('found and install successfully', async () => {
216+
spyCacheDir = jest.spyOn(tc, 'cacheDir');
217+
spyCacheDir.mockImplementation(() =>
218+
path.join(toolDir, 'PyPy', '3.7.7', architecture)
219+
);
220+
spyChmodSync = jest.spyOn(fs, 'chmodSync');
221+
spyChmodSync.mockImplementation(() => undefined);
222+
await expect(
223+
finder.findPyPyVersion('pypy-3.7-v7.3.x', architecture)
224+
).resolves.toEqual({
225+
resolvedPythonVersion: '3.7.9',
226+
resolvedPyPyVersion: '7.3.3'
227+
});
228+
});
229+
230+
it('throw if release is not found', async () => {
231+
await expect(
232+
finder.findPyPyVersion('pypy-3.7-v7.5.x', architecture)
233+
).rejects.toThrowError(
234+
`PyPy version 3.7 (v7.5.x) with arch ${architecture} not found`
235+
);
236+
});
237+
});

‎__tests__/install-pypy.test.ts

+230
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,230 @@
1+
import fs from 'fs';
2+
3+
import {HttpClient} from '@actions/http-client';
4+
import * as ifm from '@actions/http-client/interfaces';
5+
import * as tc from '@actions/tool-cache';
6+
import * as exec from '@actions/exec';
7+
import * as path from 'path';
8+
9+
import * as installer from '../src/install-pypy';
10+
import {
11+
IPyPyManifestRelease,
12+
IPyPyManifestAsset,
13+
IS_WINDOWS
14+
} from '../src/utils';
15+
16+
const manifestData = require('./data/pypy.json');
17+
18+
let architecture: string;
19+
if (IS_WINDOWS) {
20+
architecture = 'x86';
21+
} else {
22+
architecture = 'x64';
23+
}
24+
25+
const toolDir = path.join(__dirname, 'runner', 'tools');
26+
const tempDir = path.join(__dirname, 'runner', 'temp');
27+
28+
describe('pypyVersionToSemantic', () => {
29+
it.each([
30+
['7.3.3rc1', '7.3.3-rc.1'],
31+
['7.3.3', '7.3.3'],
32+
['7.3.x', '7.3.x'],
33+
['7.x', '7.x'],
34+
['nightly', 'nightly']
35+
])('%s -> %s', (input, expected) => {
36+
expect(installer.pypyVersionToSemantic(input)).toEqual(expected);
37+
});
38+
});
39+
40+
describe('findRelease', () => {
41+
const result = JSON.stringify(manifestData);
42+
const releases = JSON.parse(result) as IPyPyManifestRelease[];
43+
const extension = IS_WINDOWS ? '.zip' : '.tar.bz2';
44+
const extensionName = IS_WINDOWS
45+
? `${process.platform}${extension}`
46+
: `${process.platform}64${extension}`;
47+
const files: IPyPyManifestAsset = {
48+
filename: `pypy3.6-v7.3.3-${extensionName}`,
49+
arch: architecture,
50+
platform: process.platform,
51+
download_url: `https://test.download.python.org/pypy/pypy3.6-v7.3.3-${extensionName}`
52+
};
53+
54+
it("Python version is found, but PyPy version doesn't match", () => {
55+
const pythonVersion = '3.6';
56+
const pypyVersion = '7.3.7';
57+
expect(
58+
installer.findRelease(releases, pythonVersion, pypyVersion, architecture)
59+
).toEqual(null);
60+
});
61+
62+
it('Python version is found and PyPy version matches', () => {
63+
const pythonVersion = '3.6';
64+
const pypyVersion = '7.3.3';
65+
expect(
66+
installer.findRelease(releases, pythonVersion, pypyVersion, architecture)
67+
).toEqual({
68+
foundAsset: files,
69+
resolvedPythonVersion: '3.6.12',
70+
resolvedPyPyVersion: pypyVersion
71+
});
72+
});
73+
74+
it('Python version is found in toolcache and PyPy version matches semver', () => {
75+
const pythonVersion = '3.6';
76+
const pypyVersion = '7.x';
77+
expect(
78+
installer.findRelease(releases, pythonVersion, pypyVersion, architecture)
79+
).toEqual({
80+
foundAsset: files,
81+
resolvedPythonVersion: '3.6.12',
82+
resolvedPyPyVersion: '7.3.3'
83+
});
84+
});
85+
86+
it('Python and preview version of PyPy are found', () => {
87+
const pythonVersion = '3.7';
88+
const pypyVersion = installer.pypyVersionToSemantic('7.3.3rc2');
89+
expect(
90+
installer.findRelease(releases, pythonVersion, pypyVersion, architecture)
91+
).toEqual({
92+
foundAsset: {
93+
filename: `test${extension}`,
94+
arch: architecture,
95+
platform: process.platform,
96+
download_url: `test${extension}`
97+
},
98+
resolvedPythonVersion: '3.7.7',
99+
resolvedPyPyVersion: '7.3.3rc2'
100+
});
101+
});
102+
103+
it('Python version with latest PyPy is found', () => {
104+
const pythonVersion = '3.6';
105+
const pypyVersion = 'x';
106+
expect(
107+
installer.findRelease(releases, pythonVersion, pypyVersion, architecture)
108+
).toEqual({
109+
foundAsset: files,
110+
resolvedPythonVersion: '3.6.12',
111+
resolvedPyPyVersion: '7.3.3'
112+
});
113+
});
114+
115+
it('Nightly release is found', () => {
116+
const pythonVersion = '3.6';
117+
const pypyVersion = 'nightly';
118+
const filename = IS_WINDOWS ? 'filename.zip' : 'filename.tar.bz2';
119+
expect(
120+
installer.findRelease(releases, pythonVersion, pypyVersion, architecture)
121+
).toEqual({
122+
foundAsset: {
123+
filename: filename,
124+
arch: architecture,
125+
platform: process.platform,
126+
download_url: `http://nightlyBuilds.org/${filename}`
127+
},
128+
resolvedPythonVersion: '3.6',
129+
resolvedPyPyVersion: pypyVersion
130+
});
131+
});
132+
});
133+
134+
describe('installPyPy', () => {
135+
let tcFind: jest.SpyInstance;
136+
let spyExtractZip: jest.SpyInstance;
137+
let spyExtractTar: jest.SpyInstance;
138+
let spyFsReadDir: jest.SpyInstance;
139+
let spyFsWriteFile: jest.SpyInstance;
140+
let spyHttpClient: jest.SpyInstance;
141+
let spyExistsSync: jest.SpyInstance;
142+
let spyExec: jest.SpyInstance;
143+
let spySymlinkSync: jest.SpyInstance;
144+
let spyDownloadTool: jest.SpyInstance;
145+
let spyCacheDir: jest.SpyInstance;
146+
let spyChmodSync: jest.SpyInstance;
147+
148+
beforeEach(() => {
149+
tcFind = jest.spyOn(tc, 'find');
150+
tcFind.mockImplementation(() => path.join('PyPy', '3.6.12', architecture));
151+
152+
spyDownloadTool = jest.spyOn(tc, 'downloadTool');
153+
spyDownloadTool.mockImplementation(() => path.join(tempDir, 'PyPy'));
154+
155+
spyExtractZip = jest.spyOn(tc, 'extractZip');
156+
spyExtractZip.mockImplementation(() => tempDir);
157+
158+
spyExtractTar = jest.spyOn(tc, 'extractTar');
159+
spyExtractTar.mockImplementation(() => tempDir);
160+
161+
spyFsReadDir = jest.spyOn(fs, 'readdirSync');
162+
spyFsReadDir.mockImplementation(() => ['PyPyTest']);
163+
164+
spyFsWriteFile = jest.spyOn(fs, 'writeFileSync');
165+
spyFsWriteFile.mockImplementation(() => undefined);
166+
167+
spyHttpClient = jest.spyOn(HttpClient.prototype, 'getJson');
168+
spyHttpClient.mockImplementation(
169+
async (): Promise<ifm.ITypedResponse<IPyPyManifestRelease[]>> => {
170+
const result = JSON.stringify(manifestData);
171+
return {
172+
statusCode: 200,
173+
headers: {},
174+
result: JSON.parse(result) as IPyPyManifestRelease[]
175+
};
176+
}
177+
);
178+
179+
spyExec = jest.spyOn(exec, 'exec');
180+
spyExec.mockImplementation(() => undefined);
181+
182+
spySymlinkSync = jest.spyOn(fs, 'symlinkSync');
183+
spySymlinkSync.mockImplementation(() => undefined);
184+
185+
spyExistsSync = jest.spyOn(fs, 'existsSync');
186+
spyExistsSync.mockImplementation(() => false);
187+
});
188+
189+
afterEach(() => {
190+
jest.resetAllMocks();
191+
jest.clearAllMocks();
192+
jest.restoreAllMocks();
193+
});
194+
195+
it('throw if release is not found', async () => {
196+
await expect(
197+
installer.installPyPy('7.3.3', '3.6.17', architecture)
198+
).rejects.toThrowError(
199+
`PyPy version 3.6.17 (7.3.3) with arch ${architecture} not found`
200+
);
201+
202+
expect(spyHttpClient).toHaveBeenCalled();
203+
expect(spyDownloadTool).not.toHaveBeenCalled();
204+
expect(spyExec).not.toHaveBeenCalled();
205+
});
206+
207+
it('found and install PyPy', async () => {
208+
spyCacheDir = jest.spyOn(tc, 'cacheDir');
209+
spyCacheDir.mockImplementation(() =>
210+
path.join(toolDir, 'PyPy', '3.6.12', architecture)
211+
);
212+
213+
spyChmodSync = jest.spyOn(fs, 'chmodSync');
214+
spyChmodSync.mockImplementation(() => undefined);
215+
216+
await expect(
217+
installer.installPyPy('7.3.x', '3.6.12', architecture)
218+
).resolves.toEqual({
219+
installDir: path.join(toolDir, 'PyPy', '3.6.12', architecture),
220+
resolvedPythonVersion: '3.6.12',
221+
resolvedPyPyVersion: '7.3.3'
222+
});
223+
224+
expect(spyHttpClient).toHaveBeenCalled();
225+
expect(spyDownloadTool).toHaveBeenCalled();
226+
expect(spyExistsSync).toHaveBeenCalled();
227+
expect(spyCacheDir).toHaveBeenCalled();
228+
expect(spyExec).toHaveBeenCalled();
229+
});
230+
});

‎__tests__/utils.test.ts

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
import {
2+
validateVersion,
3+
validatePythonVersionFormatForPyPy
4+
} from '../src/utils';
5+
6+
describe('validatePythonVersionFormatForPyPy', () => {
7+
it.each([
8+
['3.6', true],
9+
['3.7', true],
10+
['3.6.x', false],
11+
['3.7.x', false],
12+
['3.x', false],
13+
['3', false]
14+
])('%s -> %s', (input, expected) => {
15+
expect(validatePythonVersionFormatForPyPy(input)).toEqual(expected);
16+
});
17+
});
18+
19+
describe('validateVersion', () => {
20+
it.each([
21+
['v7.3.3', true],
22+
['v7.3.x', true],
23+
['v7.x', true],
24+
['x', true],
25+
['v7.3.3-rc.1', true],
26+
['nightly', true],
27+
['v7.3.b', false],
28+
['3.6', true],
29+
['3.b', false],
30+
['3', true]
31+
])('%s -> %s', (input, expected) => {
32+
expect(validateVersion(input)).toEqual(expected);
33+
});
34+
});

‎dist/index.js

+360-14
Large diffs are not rendered by default.

‎src/find-pypy.ts

+131
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,131 @@
1+
import * as path from 'path';
2+
import * as pypyInstall from './install-pypy';
3+
import {
4+
IS_WINDOWS,
5+
validateVersion,
6+
getPyPyVersionFromPath,
7+
readExactPyPyVersionFile,
8+
validatePythonVersionFormatForPyPy
9+
} from './utils';
10+
11+
import * as semver from 'semver';
12+
import * as core from '@actions/core';
13+
import * as tc from '@actions/tool-cache';
14+
15+
interface IPyPyVersionSpec {
16+
pypyVersion: string;
17+
pythonVersion: string;
18+
}
19+
20+
export async function findPyPyVersion(
21+
versionSpec: string,
22+
architecture: string
23+
): Promise<{resolvedPyPyVersion: string; resolvedPythonVersion: string}> {
24+
let resolvedPyPyVersion = '';
25+
let resolvedPythonVersion = '';
26+
let installDir: string | null;
27+
28+
const pypyVersionSpec = parsePyPyVersion(versionSpec);
29+
30+
// PyPy only precompiles binaries for x86, but the architecture parameter defaults to x64.
31+
if (IS_WINDOWS && architecture === 'x64') {
32+
architecture = 'x86';
33+
}
34+
35+
({installDir, resolvedPythonVersion, resolvedPyPyVersion} = findPyPyToolCache(
36+
pypyVersionSpec.pythonVersion,
37+
pypyVersionSpec.pypyVersion,
38+
architecture
39+
));
40+
41+
if (!installDir) {
42+
({
43+
installDir,
44+
resolvedPythonVersion,
45+
resolvedPyPyVersion
46+
} = await pypyInstall.installPyPy(
47+
pypyVersionSpec.pypyVersion,
48+
pypyVersionSpec.pythonVersion,
49+
architecture
50+
));
51+
}
52+
53+
const pipDir = IS_WINDOWS ? 'Scripts' : 'bin';
54+
const _binDir = path.join(installDir, pipDir);
55+
const pythonLocation = pypyInstall.getPyPyBinaryPath(installDir);
56+
core.exportVariable('pythonLocation', pythonLocation);
57+
core.addPath(pythonLocation);
58+
core.addPath(_binDir);
59+
60+
return {resolvedPyPyVersion, resolvedPythonVersion};
61+
}
62+
63+
export function findPyPyToolCache(
64+
pythonVersion: string,
65+
pypyVersion: string,
66+
architecture: string
67+
) {
68+
let resolvedPyPyVersion = '';
69+
let resolvedPythonVersion = '';
70+
let installDir: string | null = tc.find('PyPy', pythonVersion, architecture);
71+
72+
if (installDir) {
73+
// 'tc.find' finds tool based on Python version but we also need to check
74+
// whether PyPy version satisfies requested version.
75+
resolvedPythonVersion = getPyPyVersionFromPath(installDir);
76+
resolvedPyPyVersion = readExactPyPyVersionFile(installDir);
77+
78+
const isPyPyVersionSatisfies = semver.satisfies(
79+
resolvedPyPyVersion,
80+
pypyVersion
81+
);
82+
if (!isPyPyVersionSatisfies) {
83+
installDir = null;
84+
resolvedPyPyVersion = '';
85+
resolvedPythonVersion = '';
86+
}
87+
}
88+
89+
if (!installDir) {
90+
core.info(
91+
`PyPy version ${pythonVersion} (${pypyVersion}) was not found in the local cache`
92+
);
93+
}
94+
95+
return {installDir, resolvedPythonVersion, resolvedPyPyVersion};
96+
}
97+
98+
export function parsePyPyVersion(versionSpec: string): IPyPyVersionSpec {
99+
const versions = versionSpec.split('-').filter(item => !!item);
100+
101+
if (versions.length < 2 || versions[0] != 'pypy') {
102+
throw new Error(
103+
"Invalid 'version' property for PyPy. PyPy version should be specified as 'pypy-<python-version>'. See README for examples and documentation."
104+
);
105+
}
106+
107+
const pythonVersion = versions[1];
108+
let pypyVersion: string;
109+
if (versions.length > 2) {
110+
pypyVersion = pypyInstall.pypyVersionToSemantic(versions[2]);
111+
} else {
112+
pypyVersion = 'x';
113+
}
114+
115+
if (!validateVersion(pythonVersion) || !validateVersion(pypyVersion)) {
116+
throw new Error(
117+
"Invalid 'version' property for PyPy. Both Python version and PyPy versions should satisfy SemVer notation. See README for examples and documentation."
118+
);
119+
}
120+
121+
if (!validatePythonVersionFormatForPyPy(pythonVersion)) {
122+
throw new Error(
123+
"Invalid format of Python version for PyPy. Python version should be specified in format 'x.y'. See README for examples and documentation."
124+
);
125+
}
126+
127+
return {
128+
pypyVersion: pypyVersion,
129+
pythonVersion: pythonVersion
130+
};
131+
}

‎src/find-python.ts

+1-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import * as os from 'os';
22
import * as path from 'path';
3+
import {IS_WINDOWS, IS_LINUX} from './utils';
34

45
import * as semver from 'semver';
56

@@ -8,9 +9,6 @@ import * as installer from './install-python';
89
import * as core from '@actions/core';
910
import * as tc from '@actions/tool-cache';
1011

11-
const IS_WINDOWS = process.platform === 'win32';
12-
const IS_LINUX = process.platform === 'linux';
13-
1412
// Python has "scripts" or "bin" directories where command-line tools that come with packages are installed.
1513
// This is where pip is, along with anything that pip installs.
1614
// There is a seperate directory for `pip install --user`.

‎src/install-pypy.ts

+193
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,193 @@
1+
import * as path from 'path';
2+
import * as core from '@actions/core';
3+
import * as tc from '@actions/tool-cache';
4+
import * as semver from 'semver';
5+
import * as httpm from '@actions/http-client';
6+
import * as exec from '@actions/exec';
7+
import fs from 'fs';
8+
9+
import {
10+
IS_WINDOWS,
11+
IPyPyManifestRelease,
12+
createSymlinkInFolder,
13+
isNightlyKeyword,
14+
writeExactPyPyVersionFile
15+
} from './utils';
16+
17+
export async function installPyPy(
18+
pypyVersion: string,
19+
pythonVersion: string,
20+
architecture: string
21+
) {
22+
let downloadDir;
23+
24+
const releases = await getAvailablePyPyVersions();
25+
if (!releases || releases.length === 0) {
26+
throw new Error('No release was found in PyPy version.json');
27+
}
28+
29+
const releaseData = findRelease(
30+
releases,
31+
pythonVersion,
32+
pypyVersion,
33+
architecture
34+
);
35+
36+
if (!releaseData || !releaseData.foundAsset) {
37+
throw new Error(
38+
`PyPy version ${pythonVersion} (${pypyVersion}) with arch ${architecture} not found`
39+
);
40+
}
41+
42+
const {foundAsset, resolvedPythonVersion, resolvedPyPyVersion} = releaseData;
43+
let downloadUrl = `${foundAsset.download_url}`;
44+
45+
core.info(`Downloading PyPy from "${downloadUrl}" ...`);
46+
const pypyPath = await tc.downloadTool(downloadUrl);
47+
48+
core.info('Extracting downloaded archive...');
49+
if (IS_WINDOWS) {
50+
downloadDir = await tc.extractZip(pypyPath);
51+
} else {
52+
downloadDir = await tc.extractTar(pypyPath, undefined, 'x');
53+
}
54+
55+
// root folder in archive can have unpredictable name so just take the first folder
56+
// downloadDir is unique folder under TEMP and can't contain any other folders
57+
const archiveName = fs.readdirSync(downloadDir)[0];
58+
59+
const toolDir = path.join(downloadDir, archiveName);
60+
let installDir = toolDir;
61+
if (!isNightlyKeyword(resolvedPyPyVersion)) {
62+
installDir = await tc.cacheDir(
63+
toolDir,
64+
'PyPy',
65+
resolvedPythonVersion,
66+
architecture
67+
);
68+
}
69+
70+
writeExactPyPyVersionFile(installDir, resolvedPyPyVersion);
71+
72+
const binaryPath = getPyPyBinaryPath(installDir);
73+
await createPyPySymlink(binaryPath, resolvedPythonVersion);
74+
await installPip(binaryPath);
75+
76+
return {installDir, resolvedPythonVersion, resolvedPyPyVersion};
77+
}
78+
79+
async function getAvailablePyPyVersions() {
80+
const url = 'https://downloads.python.org/pypy/versions.json';
81+
const http: httpm.HttpClient = new httpm.HttpClient('tool-cache');
82+
83+
const response = await http.getJson<IPyPyManifestRelease[]>(url);
84+
if (!response.result) {
85+
throw new Error(
86+
`Unable to retrieve the list of available PyPy versions from '${url}'`
87+
);
88+
}
89+
90+
return response.result;
91+
}
92+
93+
async function createPyPySymlink(
94+
pypyBinaryPath: string,
95+
pythonVersion: string
96+
) {
97+
const version = semver.coerce(pythonVersion)!;
98+
const pythonBinaryPostfix = semver.major(version);
99+
const pypyBinaryPostfix = pythonBinaryPostfix === 2 ? '' : '3';
100+
let binaryExtension = IS_WINDOWS ? '.exe' : '';
101+
102+
core.info('Creating symlinks...');
103+
createSymlinkInFolder(
104+
pypyBinaryPath,
105+
`pypy${pypyBinaryPostfix}${binaryExtension}`,
106+
`python${pythonBinaryPostfix}${binaryExtension}`,
107+
true
108+
);
109+
110+
createSymlinkInFolder(
111+
pypyBinaryPath,
112+
`pypy${pypyBinaryPostfix}${binaryExtension}`,
113+
`python${binaryExtension}`,
114+
true
115+
);
116+
}
117+
118+
async function installPip(pythonLocation: string) {
119+
core.info('Installing and updating pip');
120+
const pythonBinary = path.join(pythonLocation, 'python');
121+
await exec.exec(`${pythonBinary} -m ensurepip`);
122+
123+
await exec.exec(
124+
`${pythonLocation}/python -m pip install --ignore-installed pip`
125+
);
126+
}
127+
128+
export function findRelease(
129+
releases: IPyPyManifestRelease[],
130+
pythonVersion: string,
131+
pypyVersion: string,
132+
architecture: string
133+
) {
134+
const filterReleases = releases.filter(item => {
135+
const isPythonVersionSatisfied = semver.satisfies(
136+
semver.coerce(item.python_version)!,
137+
pythonVersion
138+
);
139+
const isPyPyNightly =
140+
isNightlyKeyword(pypyVersion) && isNightlyKeyword(item.pypy_version);
141+
const isPyPyVersionSatisfied =
142+
isPyPyNightly ||
143+
semver.satisfies(pypyVersionToSemantic(item.pypy_version), pypyVersion);
144+
const isArchPresent =
145+
item.files &&
146+
item.files.some(
147+
file => file.arch === architecture && file.platform === process.platform
148+
);
149+
return isPythonVersionSatisfied && isPyPyVersionSatisfied && isArchPresent;
150+
});
151+
152+
if (filterReleases.length === 0) {
153+
return null;
154+
}
155+
156+
const sortedReleases = filterReleases.sort((previous, current) => {
157+
return (
158+
semver.compare(
159+
semver.coerce(pypyVersionToSemantic(current.pypy_version))!,
160+
semver.coerce(pypyVersionToSemantic(previous.pypy_version))!
161+
) ||
162+
semver.compare(
163+
semver.coerce(current.python_version)!,
164+
semver.coerce(previous.python_version)!
165+
)
166+
);
167+
});
168+
169+
const foundRelease = sortedReleases[0];
170+
const foundAsset = foundRelease.files.find(
171+
item => item.arch === architecture && item.platform === process.platform
172+
);
173+
174+
return {
175+
foundAsset,
176+
resolvedPythonVersion: foundRelease.python_version,
177+
resolvedPyPyVersion: foundRelease.pypy_version
178+
};
179+
}
180+
181+
/** Get PyPy binary location from the tool of installation directory
182+
* - On Linux and macOS, the Python interpreter is in 'bin'.
183+
* - On Windows, it is in the installation root.
184+
*/
185+
export function getPyPyBinaryPath(installDir: string) {
186+
const _binDir = path.join(installDir, 'bin');
187+
return IS_WINDOWS ? installDir : _binDir;
188+
}
189+
190+
export function pypyVersionToSemantic(versionSpec: string) {
191+
const prereleaseVersion = /(\d+\.\d+\.\d+)((?:a|b|rc))(\d*)/g;
192+
return versionSpec.replace(prereleaseVersion, '$1-$2.$3');
193+
}

‎src/install-python.ts

+1-4
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ import * as core from '@actions/core';
33
import * as tc from '@actions/tool-cache';
44
import * as exec from '@actions/exec';
55
import {ExecOptions} from '@actions/exec/lib/interfaces';
6-
import {stderr} from 'process';
6+
import {IS_WINDOWS, IS_LINUX} from './utils';
77

88
const TOKEN = core.getInput('token');
99
const AUTH = !TOKEN || isGhes() ? undefined : `token ${TOKEN}`;
@@ -12,9 +12,6 @@ const MANIFEST_REPO_NAME = 'python-versions';
1212
const MANIFEST_REPO_BRANCH = 'main';
1313
export const MANIFEST_URL = `https://raw.githubusercontent.com/${MANIFEST_REPO_OWNER}/${MANIFEST_REPO_NAME}/${MANIFEST_REPO_BRANCH}/versions-manifest.json`;
1414

15-
const IS_WINDOWS = process.platform === 'win32';
16-
const IS_LINUX = process.platform === 'linux';
17-
1815
export async function findReleaseFromManifest(
1916
semanticVersionSpec: string,
2017
architecture: string

‎src/setup-python.ts

+16-2
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,29 @@
11
import * as core from '@actions/core';
22
import * as finder from './find-python';
3+
import * as finderPyPy from './find-pypy';
34
import * as path from 'path';
45
import * as os from 'os';
56

7+
function isPyPyVersion(versionSpec: string) {
8+
return versionSpec.startsWith('pypy-');
9+
}
10+
611
async function run() {
712
try {
813
let version = core.getInput('python-version');
914
if (version) {
1015
const arch: string = core.getInput('architecture') || os.arch();
11-
const installed = await finder.findPythonVersion(version, arch);
12-
core.info(`Successfully setup ${installed.impl} (${installed.version})`);
16+
if (isPyPyVersion(version)) {
17+
const installed = await finderPyPy.findPyPyVersion(version, arch);
18+
core.info(
19+
`Successfully setup PyPy ${installed.resolvedPyPyVersion} with Python (${installed.resolvedPythonVersion})`
20+
);
21+
} else {
22+
const installed = await finder.findPythonVersion(version, arch);
23+
core.info(
24+
`Successfully setup ${installed.impl} (${installed.version})`
25+
);
26+
}
1327
}
1428
const matchersPath = path.join(__dirname, '..', '.github');
1529
core.info(`##[add-matcher]${path.join(matchersPath, 'python.json')}`);

‎src/utils.ts

+92
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
import fs from 'fs';
2+
import * as path from 'path';
3+
import * as semver from 'semver';
4+
5+
export const IS_WINDOWS = process.platform === 'win32';
6+
export const IS_LINUX = process.platform === 'linux';
7+
const PYPY_VERSION_FILE = 'PYPY_VERSION';
8+
9+
export interface IPyPyManifestAsset {
10+
filename: string;
11+
arch: string;
12+
platform: string;
13+
download_url: string;
14+
}
15+
16+
export interface IPyPyManifestRelease {
17+
pypy_version: string;
18+
python_version: string;
19+
stable: boolean;
20+
latest_pypy: boolean;
21+
files: IPyPyManifestAsset[];
22+
}
23+
24+
/** create Symlinks for downloaded PyPy
25+
* It should be executed only for downloaded versions in runtime, because
26+
* toolcache versions have this setup.
27+
*/
28+
export function createSymlinkInFolder(
29+
folderPath: string,
30+
sourceName: string,
31+
targetName: string,
32+
setExecutable = false
33+
) {
34+
const sourcePath = path.join(folderPath, sourceName);
35+
const targetPath = path.join(folderPath, targetName);
36+
if (fs.existsSync(targetPath)) {
37+
return;
38+
}
39+
40+
fs.symlinkSync(sourcePath, targetPath);
41+
if (!IS_WINDOWS && setExecutable) {
42+
fs.chmodSync(targetPath, '755');
43+
}
44+
}
45+
46+
export function validateVersion(version: string) {
47+
return isNightlyKeyword(version) || Boolean(semver.validRange(version));
48+
}
49+
50+
export function isNightlyKeyword(pypyVersion: string) {
51+
return pypyVersion === 'nightly';
52+
}
53+
54+
export function getPyPyVersionFromPath(installDir: string) {
55+
return path.basename(path.dirname(installDir));
56+
}
57+
58+
/**
59+
* In tool-cache, we put PyPy to '<toolcache_root>/PyPy/<python_version>/x64'
60+
* There is no easy way to determine what PyPy version is located in specific folder
61+
* 'pypy --version' is not reliable enough since it is not set properly for preview versions
62+
* "7.3.3rc1" is marked as '7.3.3' in 'pypy --version'
63+
* so we put PYPY_VERSION file to PyPy directory when install it to VM and read it when we need to know version
64+
* PYPY_VERSION contains exact version from 'versions.json'
65+
*/
66+
export function readExactPyPyVersionFile(installDir: string) {
67+
let pypyVersion = '';
68+
let fileVersion = path.join(installDir, PYPY_VERSION_FILE);
69+
if (fs.existsSync(fileVersion)) {
70+
pypyVersion = fs.readFileSync(fileVersion).toString();
71+
}
72+
73+
return pypyVersion;
74+
}
75+
76+
export function writeExactPyPyVersionFile(
77+
installDir: string,
78+
resolvedPyPyVersion: string
79+
) {
80+
const pypyFilePath = path.join(installDir, PYPY_VERSION_FILE);
81+
fs.writeFileSync(pypyFilePath, resolvedPyPyVersion);
82+
}
83+
84+
/**
85+
* Python version should be specified explicitly like "x.y" (2.7, 3.6, 3.7)
86+
* "3.x" or "3" are not supported
87+
* because it could cause ambiguity when both PyPy version and Python version are not precise
88+
*/
89+
export function validatePythonVersionFormatForPyPy(version: string) {
90+
const re = /^\d+\.\d+$/;
91+
return re.test(version);
92+
}

0 commit comments

Comments
 (0)
Please sign in to comment.