mirror of
https://github.com/pnpm/action-setup.git
synced 2026-05-15 06:43:32 +08:00
Compare commits
15 Commits
update-pnp
...
next
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
3de0023fce | ||
|
|
f489bfced8 | ||
|
|
25df77c6d2 | ||
|
|
53ef38b049 | ||
|
|
1f69135d3d | ||
|
|
c3b2a5fad6 | ||
|
|
0ffe724fa5 | ||
|
|
a722bd2d87 | ||
|
|
49b8837b49 | ||
|
|
9fbc6cd1d1 | ||
|
|
af96d9fd0e | ||
|
|
1fb299a44f | ||
|
|
24a61aa18d | ||
|
|
ab0b84aeb1 | ||
|
|
dc312cdfd7 |
1
.github/CODEOWNERS
vendored
1
.github/CODEOWNERS
vendored
@@ -1 +0,0 @@
|
|||||||
* @zkochan
|
|
||||||
310
.github/workflows/test.yaml
vendored
310
.github/workflows/test.yaml
vendored
@@ -1,76 +1,39 @@
|
|||||||
name: Test Action
|
name: Test Action
|
||||||
|
|
||||||
on:
|
on:
|
||||||
pull_request:
|
- push
|
||||||
push:
|
- pull_request
|
||||||
branches:
|
- workflow_dispatch
|
||||||
- master
|
|
||||||
workflow_dispatch:
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
smoke:
|
test_default_inputs:
|
||||||
# Cross-OS coverage. Exercises the bootstrap install + PATH on each platform,
|
name: Test with default inputs
|
||||||
# the version-respects-request regression (#225 / #230 — Windows PATH shadow),
|
|
||||||
# and the bin_dest output regression (#247). Multi-version coverage on Linux
|
|
||||||
# so we don't pay 3x for major-version differences.
|
|
||||||
name: 'Smoke (${{ matrix.name }})'
|
|
||||||
|
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
pnpm:
|
||||||
- name: 'ubuntu / v9.15.5'
|
- 9.15.5
|
||||||
os: ubuntu-latest
|
os:
|
||||||
version: '9.15.5'
|
- ubuntu-latest
|
||||||
- name: 'ubuntu / v10.33.0'
|
- macos-latest
|
||||||
os: ubuntu-latest
|
- windows-latest
|
||||||
version: '10.33.0'
|
|
||||||
- name: 'ubuntu / v9.15.5 / custom-dest'
|
|
||||||
os: ubuntu-latest
|
|
||||||
version: '9.15.5'
|
|
||||||
dest: '~/test/pnpm'
|
|
||||||
- name: 'macos / v9.15.5'
|
|
||||||
os: macos-latest
|
|
||||||
version: '9.15.5'
|
|
||||||
- name: 'windows / v9.15.5'
|
|
||||||
os: windows-latest
|
|
||||||
version: '9.15.5'
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@34e114876b0b11c390a56381ad16ebd13914f8d5 # v4.3.1
|
- uses: actions/checkout@34e114876b0b11c390a56381ad16ebd13914f8d5 # v4.3.1
|
||||||
|
|
||||||
- id: pnpm
|
- name: Run the action
|
||||||
name: Run the action
|
|
||||||
uses: ./
|
uses: ./
|
||||||
with:
|
with:
|
||||||
version: ${{ matrix.version }}
|
version: 9.15.5
|
||||||
dest: ${{ matrix.dest || '~/setup-pnpm' }}
|
|
||||||
|
|
||||||
- name: 'Test: pnpm/pnpx on PATH report the requested version (incl. via bin_dest)'
|
- name: 'Test: which'
|
||||||
# Pass paths via env, not template interpolation, so Windows
|
run: which pnpm; which pnpx
|
||||||
# backslashes in `bin_dest` aren't eaten by bash's escape handling.
|
|
||||||
env:
|
- name: 'Test: version'
|
||||||
BIN_DEST: ${{ steps.pnpm.outputs.bin_dest }}
|
run: pnpm --version
|
||||||
REQUIRED: ${{ matrix.version }}
|
|
||||||
run: |
|
|
||||||
set -e
|
|
||||||
which pnpm
|
|
||||||
which pnpx
|
|
||||||
actual="$(pnpm --version)"
|
|
||||||
echo "pnpm --version: ${actual}"
|
|
||||||
if [ "${actual}" != "${REQUIRED}" ]; then
|
|
||||||
echo "Expected pnpm version ${REQUIRED}, but got ${actual}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
bin_dest_version="$("$BIN_DEST/pnpm" --version)"
|
|
||||||
echo "bin_dest pnpm --version: ${bin_dest_version}"
|
|
||||||
if [ "${bin_dest_version}" != "${REQUIRED}" ]; then
|
|
||||||
echo "Expected ${REQUIRED} via bin_dest, but got ${bin_dest_version}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
shell: bash
|
|
||||||
|
|
||||||
- name: 'Test: install in a fresh project'
|
- name: 'Test: install in a fresh project'
|
||||||
run: |
|
run: |
|
||||||
@@ -80,95 +43,47 @@ jobs:
|
|||||||
pnpm add is-odd
|
pnpm add is-odd
|
||||||
shell: bash
|
shell: bash
|
||||||
|
|
||||||
manifest_pin:
|
test_dest:
|
||||||
# Folds the old test_package_manager_field, test_dev_engines, and
|
name: Test with dest
|
||||||
# test_dev_engines_on_fail_error jobs. The action's manifest handling is
|
|
||||||
# OS-independent, so ubuntu-only is sufficient.
|
|
||||||
name: 'Manifest pin: ${{ matrix.label }}'
|
|
||||||
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ${{ matrix.os }}
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
pnpm:
|
||||||
- label: 'packageManager pnpm@9.15.5 (#227)'
|
- 9.15.5
|
||||||
manifest: '{"packageManager":"pnpm@9.15.5"}'
|
os:
|
||||||
version: '9.15.5'
|
- ubuntu-latest
|
||||||
- label: 'packageManager pnpm@10.33.0'
|
- macos-latest
|
||||||
manifest: '{"packageManager":"pnpm@10.33.0"}'
|
- windows-latest
|
||||||
version: '10.33.0'
|
|
||||||
- label: 'devEngines onFail=download, exact'
|
|
||||||
manifest: '{"devEngines":{"packageManager":{"name":"pnpm","version":"9.15.5","onFail":"download"}}}'
|
|
||||||
version: '9.15.5'
|
|
||||||
- label: 'devEngines onFail=download, range'
|
|
||||||
manifest: '{"devEngines":{"packageManager":{"name":"pnpm","version":">=9.15.0","onFail":"download"}}}'
|
|
||||||
version: '>=9.15.0'
|
|
||||||
- label: 'devEngines onFail=error, exact (#252)'
|
|
||||||
manifest: '{"devEngines":{"packageManager":{"name":"pnpm","version":"9.15.5","onFail":"error"}}}'
|
|
||||||
version: '9.15.5'
|
|
||||||
- label: 'devEngines onFail=error, range (#252)'
|
|
||||||
manifest: '{"devEngines":{"packageManager":{"name":"pnpm","version":">=9.15.0","onFail":"error"}}}'
|
|
||||||
version: '>=9.15.0'
|
|
||||||
- label: 'explicit version: pnpm_config_pm_on_fail not exported'
|
|
||||||
# Regression guard for the af8e203 scope fix: when the user passes an
|
|
||||||
# explicit `version:` input, the action must NOT export
|
|
||||||
# pnpm_config_pm_on_fail=download, so the user's strict onFail policy
|
|
||||||
# is preserved. Asserted directly on the env var rather than pnpm
|
|
||||||
# runtime behavior — different pnpm majors read devEngines
|
|
||||||
# differently (v10 ignores it, v11+ honors it).
|
|
||||||
manifest: '{"devEngines":{"packageManager":{"name":"pnpm","version":"9.15.5","onFail":"error"}}}'
|
|
||||||
explicit_version: '10.33.0'
|
|
||||||
expect_pm_on_fail_unset: true
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@34e114876b0b11c390a56381ad16ebd13914f8d5 # v4.3.1
|
- uses: actions/checkout@34e114876b0b11c390a56381ad16ebd13914f8d5 # v4.3.1
|
||||||
|
|
||||||
- name: Set up package.json
|
|
||||||
run: echo '${{ matrix.manifest }}' > package.json
|
|
||||||
shell: bash
|
|
||||||
|
|
||||||
- name: Run the action
|
- name: Run the action
|
||||||
uses: ./
|
uses: ./
|
||||||
with:
|
with:
|
||||||
version: ${{ matrix.explicit_version }}
|
version: 9.15.5
|
||||||
|
dest: ~/test/pnpm
|
||||||
|
|
||||||
- name: 'Test: pnpm reports the pinned version'
|
- name: 'Test: which'
|
||||||
if: ${{ !matrix.expect_pm_on_fail_unset }}
|
run: which pnpm && which pnpx
|
||||||
env:
|
|
||||||
REQUIRED: ${{ matrix.version }}
|
|
||||||
run: |
|
|
||||||
set -e
|
|
||||||
actual="$(pnpm --version)"
|
|
||||||
echo "pnpm version: ${actual}"
|
|
||||||
if [ "${REQUIRED}" = ">=9.15.0" ]; then
|
|
||||||
min="9.15.0"
|
|
||||||
if [ "$(printf '%s\n' "${min}" "${actual}" | sort -V | head -n1)" != "${min}" ]; then
|
|
||||||
echo "Expected pnpm version >= ${min}, but got ${actual}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
if [ "${actual}" != "${REQUIRED}" ]; then
|
|
||||||
echo "Expected pnpm version ${REQUIRED}, but got ${actual}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
shell: bash
|
|
||||||
|
|
||||||
- name: 'Test: pnpm_config_pm_on_fail not exported (explicit version preserves strict policy)'
|
- name: 'Test: version'
|
||||||
if: ${{ matrix.expect_pm_on_fail_unset }}
|
run: pnpm --version
|
||||||
run: |
|
|
||||||
if [ -n "${pnpm_config_pm_on_fail:-}" ]; then
|
|
||||||
echo "Expected pnpm_config_pm_on_fail to be unset, but got: '${pnpm_config_pm_on_fail}'"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
echo "pnpm_config_pm_on_fail is unset, as expected"
|
|
||||||
shell: bash
|
|
||||||
|
|
||||||
standalone:
|
test_standalone:
|
||||||
name: Standalone mode
|
name: Test with standalone
|
||||||
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ${{ matrix.os }}
|
||||||
|
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
os:
|
||||||
|
- ubuntu-latest
|
||||||
|
- windows-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@34e114876b0b11c390a56381ad16ebd13914f8d5 # v4.3.1
|
- uses: actions/checkout@34e114876b0b11c390a56381ad16ebd13914f8d5 # v4.3.1
|
||||||
@@ -179,125 +94,34 @@ jobs:
|
|||||||
version: 9.15.0
|
version: 9.15.0
|
||||||
standalone: true
|
standalone: true
|
||||||
|
|
||||||
- name: 'Test: pnpm works'
|
- name: 'Test: which'
|
||||||
|
run: which pnpm
|
||||||
|
|
||||||
|
- name: 'Test: version'
|
||||||
|
run: pnpm --version
|
||||||
|
|
||||||
|
- name: 'Test: install in a fresh project'
|
||||||
run: |
|
run: |
|
||||||
set -e
|
|
||||||
which pnpm
|
|
||||||
actual="$(pnpm --version)"
|
|
||||||
if [ "${actual}" != "9.15.0" ]; then
|
|
||||||
echo "Expected 9.15.0, got ${actual}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
mkdir /tmp/test-standalone
|
mkdir /tmp/test-standalone
|
||||||
cd /tmp/test-standalone
|
cd /tmp/test-standalone
|
||||||
pnpm init
|
pnpm init
|
||||||
pnpm add is-odd
|
pnpm add is-odd
|
||||||
shell: bash
|
shell: bash
|
||||||
|
|
||||||
standalone_windows_self_update:
|
test_run_install:
|
||||||
# Regression guard for the patchPnpmEnv PATH-shadow bug. When
|
name: 'Test with run_install (${{ matrix.run_install.name }}, ${{ matrix.os }})'
|
||||||
# standalone: true on Windows AND the requested pnpm differs from the
|
|
||||||
# bootstrap, the previous patchPnpmEnv prepended node_modules/.bin to
|
|
||||||
# PATH; that directory contains an npm-created pnpm.cmd shim pointing
|
|
||||||
# at the BOOTSTRAP pnpm, which shadowed the self-updated pnpm at
|
|
||||||
# $PNPM_HOME/bin and caused `pnpm install` inside the action to run
|
|
||||||
# under the bootstrap version. Exercising a newer-pnpm-only flag
|
|
||||||
# (`--no-runtime`, added in 11.1.0) makes the regression assertable:
|
|
||||||
# if the bootstrap (11.0.4) handles the install, it errors with
|
|
||||||
# "Unknown option: 'runtime'".
|
|
||||||
name: 'Standalone Windows self-update (PATH regression)'
|
|
||||||
runs-on: windows-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@34e114876b0b11c390a56381ad16ebd13914f8d5 # v4.3.1
|
|
||||||
|
|
||||||
- name: Set up package.json with a minimal manifest
|
runs-on: ${{ matrix.os }}
|
||||||
# run_install needs a manifest to install against. Removing the
|
|
||||||
# repo's existing pnpm-lock.yaml avoids frozen-lockfile mismatch.
|
|
||||||
run: |
|
|
||||||
rm -f pnpm-lock.yaml
|
|
||||||
echo '{"name":"sw","private":true,"packageManager":"pnpm@11.1.0"}' > package.json
|
|
||||||
shell: bash
|
|
||||||
|
|
||||||
- name: Run the action
|
|
||||||
uses: ./
|
|
||||||
with:
|
|
||||||
version: 11.1.0
|
|
||||||
standalone: true
|
|
||||||
run_install: |
|
|
||||||
args: ['--no-runtime']
|
|
||||||
|
|
||||||
- name: 'Test: pnpm install completed under the self-updated pnpm'
|
|
||||||
# If the bug recurs, the previous step's run_install will have failed
|
|
||||||
# the job with "Unknown option: 'runtime'", so reaching this step
|
|
||||||
# implies success. Still verify the version on PATH matches request.
|
|
||||||
env:
|
|
||||||
REQUIRED: '11.1.0'
|
|
||||||
run: |
|
|
||||||
set -e
|
|
||||||
actual="$(pnpm --version)"
|
|
||||||
echo "pnpm --version: ${actual}"
|
|
||||||
if [ "${actual}" != "${REQUIRED}" ]; then
|
|
||||||
echo "Expected pnpm ${REQUIRED}, got ${actual}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
shell: bash
|
|
||||||
|
|
||||||
cache_store_path:
|
|
||||||
# Regression guard for #233. When package.json pins a pnpm major that
|
|
||||||
# differs from the bootstrap pnpm's major, the bootstrap reports its
|
|
||||||
# own STORE_VERSION from `pnpm store path` (the `store` command skips
|
|
||||||
# pnpm's auto-switch). The user's actual `pnpm install` runs under the
|
|
||||||
# pinned version and writes to a different STORE_VERSION, so the post
|
|
||||||
# step's saveCache then fails with "Path Validation Error". The fix is
|
|
||||||
# to self-update the bootstrap to the pinned version up front.
|
|
||||||
name: 'Cache store path matches install (#233): ${{ matrix.label }}'
|
|
||||||
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
include:
|
|
||||||
- label: 'packageManager pnpm@10.33.0'
|
|
||||||
manifest: '{"packageManager":"pnpm@10.33.0","dependencies":{"is-odd":"3.0.1"}}'
|
|
||||||
- label: 'devEngines exact pnpm@10.33.0'
|
|
||||||
manifest: '{"devEngines":{"packageManager":{"name":"pnpm","version":"10.33.0"}},"dependencies":{"is-odd":"3.0.1"}}'
|
|
||||||
- label: 'devEngines range >=10 <11'
|
|
||||||
manifest: '{"devEngines":{"packageManager":{"name":"pnpm","version":">=10 <11"}},"dependencies":{"is-odd":"3.0.1"}}'
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@34e114876b0b11c390a56381ad16ebd13914f8d5 # v4.3.1
|
|
||||||
|
|
||||||
- name: Set up package.json
|
|
||||||
run: echo '${{ matrix.manifest }}' > package.json
|
|
||||||
shell: bash
|
|
||||||
|
|
||||||
- id: pnpm
|
|
||||||
uses: ./
|
|
||||||
with:
|
|
||||||
cache: true
|
|
||||||
run_install: |
|
|
||||||
- args: [--no-frozen-lockfile]
|
|
||||||
|
|
||||||
- name: 'Test: store path computed by the action exists on disk'
|
|
||||||
run: |
|
|
||||||
set -e
|
|
||||||
actual="$(pnpm store path --silent)"
|
|
||||||
echo "pnpm store path: ${actual}"
|
|
||||||
if [ ! -d "${actual}" ]; then
|
|
||||||
echo "Expected store path to exist on disk; cache save would fail"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
shell: bash
|
|
||||||
|
|
||||||
run_install:
|
|
||||||
name: 'run_install (${{ matrix.run_install.name }})'
|
|
||||||
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
|
pnpm:
|
||||||
|
- 9.15.5
|
||||||
|
os:
|
||||||
|
- ubuntu-latest
|
||||||
|
- macos-latest
|
||||||
|
- windows-latest
|
||||||
run_install:
|
run_install:
|
||||||
- name: 'null'
|
- name: 'null'
|
||||||
value: 'null'
|
value: 'null'
|
||||||
@@ -318,14 +142,8 @@ jobs:
|
|||||||
version: 9.15.5
|
version: 9.15.5
|
||||||
run_install: ${{ matrix.run_install.value }}
|
run_install: ${{ matrix.run_install.value }}
|
||||||
|
|
||||||
- name: 'Test: pnpm works'
|
- name: 'Test: which'
|
||||||
run: |
|
run: which pnpm; which pnpx
|
||||||
set -e
|
|
||||||
which pnpm
|
- name: 'Test: version'
|
||||||
which pnpx
|
run: pnpm --version
|
||||||
actual="$(pnpm --version)"
|
|
||||||
if [ "${actual}" != "9.15.5" ]; then
|
|
||||||
echo "Expected 9.15.5, got ${actual}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
shell: bash
|
|
||||||
|
|||||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -9,4 +9,3 @@ temp
|
|||||||
tmp.*
|
tmp.*
|
||||||
temp.*
|
temp.*
|
||||||
.pnpm-store
|
.pnpm-store
|
||||||
.claude
|
|
||||||
|
|||||||
43
README.md
43
README.md
@@ -14,7 +14,7 @@ Version of pnpm to install.
|
|||||||
|
|
||||||
**Optional** when there is a [`packageManager` field in the `package.json`](https://nodejs.org/api/corepack.html).
|
**Optional** when there is a [`packageManager` field in the `package.json`](https://nodejs.org/api/corepack.html).
|
||||||
|
|
||||||
otherwise, this field is **required** It supports npm versioning scheme, it could be an exact version (such as `10.9.8`), or a version range (such as `10`, `10.x.x`, `10.9.x`, `^10.9.8`, `*`, etc.), or `latest`.
|
otherwise, this field is **required** It supports npm versioning scheme, it could be an exact version (such as `6.24.1`), or a version range (such as `6`, `6.x.x`, `6.24.x`, `^6.24.1`, `*`, etc.), or `latest`.
|
||||||
|
|
||||||
### `dest`
|
### `dest`
|
||||||
|
|
||||||
@@ -48,7 +48,7 @@ If `run_install` is a YAML string representation of either an object or an array
|
|||||||
|
|
||||||
### `cache_dependency_path`
|
### `cache_dependency_path`
|
||||||
|
|
||||||
**Optional** (_type:_ `string`, _default:_ `pnpm-lock.yaml`) File path to the pnpm lockfile, whose contents hash will be used as a cache key. Accepts multiple paths delimited by newlines.
|
**Optional** (_type:_ `string|string[]`, _default:_ `pnpm-lock.yaml`) File path to the pnpm lockfile, which contents hash will be used as a cache key.
|
||||||
|
|
||||||
### `package_json_file`
|
### `package_json_file`
|
||||||
|
|
||||||
@@ -86,7 +86,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: pnpm/action-setup@v6
|
- uses: pnpm/action-setup@v4
|
||||||
with:
|
with:
|
||||||
version: 10
|
version: 10
|
||||||
```
|
```
|
||||||
@@ -105,7 +105,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: pnpm/action-setup@v6
|
- uses: pnpm/action-setup@v4
|
||||||
```
|
```
|
||||||
|
|
||||||
### Install pnpm and a few npm packages
|
### Install pnpm and a few npm packages
|
||||||
@@ -120,9 +120,9 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v6
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- uses: pnpm/action-setup@v6
|
- uses: pnpm/action-setup@v4
|
||||||
with:
|
with:
|
||||||
version: 10
|
version: 10
|
||||||
run_install: |
|
run_install: |
|
||||||
@@ -144,9 +144,9 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v6
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- uses: pnpm/action-setup@v6
|
- uses: pnpm/action-setup@v4
|
||||||
name: Install pnpm
|
name: Install pnpm
|
||||||
with:
|
with:
|
||||||
version: 10
|
version: 10
|
||||||
@@ -158,33 +158,6 @@ jobs:
|
|||||||
|
|
||||||
**Note:** You don't need to run `pnpm store prune` at the end; post-action has already taken care of that.
|
**Note:** You don't need to run `pnpm store prune` at the end; post-action has already taken care of that.
|
||||||
|
|
||||||
### Cache dependencies from multiple lockfiles
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
on:
|
|
||||||
- push
|
|
||||||
- pull_request
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
cache-and-install-multiple:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v6
|
|
||||||
|
|
||||||
- uses: pnpm/action-setup@v6
|
|
||||||
with:
|
|
||||||
version: 10
|
|
||||||
cache: true
|
|
||||||
cache_dependency_path: |
|
|
||||||
one/pnpm-lock.yaml
|
|
||||||
two/pnpm-lock.yaml
|
|
||||||
run_install: |
|
|
||||||
- cwd: one
|
|
||||||
- cwd: two
|
|
||||||
```
|
|
||||||
|
|
||||||
## Notes
|
## Notes
|
||||||
|
|
||||||
This action does not setup Node.js for you, use [actions/setup-node](https://github.com/actions/setup-node) yourself.
|
This action does not setup Node.js for you, use [actions/setup-node](https://github.com/actions/setup-node) yourself.
|
||||||
|
|||||||
@@ -20,7 +20,7 @@ inputs:
|
|||||||
required: false
|
required: false
|
||||||
default: 'false'
|
default: 'false'
|
||||||
cache_dependency_path:
|
cache_dependency_path:
|
||||||
description: File path to the pnpm lockfile, whose contents hash will be used as a cache key. Accepts multiple paths delimited by newlines.
|
description: File path to the pnpm lockfile, which contents hash will be used as a cache key
|
||||||
required: false
|
required: false
|
||||||
default: 'pnpm-lock.yaml'
|
default: 'pnpm-lock.yaml'
|
||||||
package_json_file:
|
package_json_file:
|
||||||
|
|||||||
312
dist/index.js
vendored
312
dist/index.js
vendored
File diff suppressed because one or more lines are too long
@@ -3,8 +3,7 @@
|
|||||||
"scripts": {
|
"scripts": {
|
||||||
"build:bundle": "esbuild src/index.ts --bundle --platform=node --target=node24 --format=cjs --minify --outfile=dist/index.js --loader:.json=json",
|
"build:bundle": "esbuild src/index.ts --bundle --platform=node --target=node24 --format=cjs --minify --outfile=dist/index.js --loader:.json=json",
|
||||||
"build": "pnpm run build:bundle",
|
"build": "pnpm run build:bundle",
|
||||||
"start": "pnpm run build && sh ./run.sh",
|
"start": "pnpm run build && sh ./run.sh"
|
||||||
"update-bootstrap": "node scripts/update-bootstrap.mjs"
|
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@actions/cache": "^4.1.0",
|
"@actions/cache": "^4.1.0",
|
||||||
|
|||||||
@@ -1,47 +0,0 @@
|
|||||||
#!/usr/bin/env node
|
|
||||||
|
|
||||||
// Usage: node scripts/update-bootstrap.mjs [version]
|
|
||||||
// If version is omitted, fetches the latest next-11 tag from npm.
|
|
||||||
// Regenerates the bootstrap lockfiles used by action-setup to install pnpm via npm.
|
|
||||||
|
|
||||||
import { execSync } from 'child_process'
|
|
||||||
import { mkdtempSync, rmSync, readFileSync, writeFileSync } from 'fs'
|
|
||||||
import { join } from 'path'
|
|
||||||
import { tmpdir } from 'os'
|
|
||||||
|
|
||||||
const BOOTSTRAP_DIR = new URL('../src/install-pnpm/bootstrap/', import.meta.url).pathname
|
|
||||||
|
|
||||||
const version = process.argv[2] || resolveLatestVersion()
|
|
||||||
|
|
||||||
console.log(`Updating bootstrap lockfiles to pnpm@${version} ...`)
|
|
||||||
|
|
||||||
generateLock('pnpm-lock.json', { pnpm: version }, 'bootstrap-pnpm')
|
|
||||||
generateLock('exe-lock.json', { '@pnpm/exe': version }, 'bootstrap-exe')
|
|
||||||
|
|
||||||
console.log('Done!')
|
|
||||||
|
|
||||||
function resolveLatestVersion() {
|
|
||||||
const json = execSync('npm view @pnpm/exe dist-tags --json', { encoding: 'utf8' })
|
|
||||||
const tags = JSON.parse(json)
|
|
||||||
const version = tags['next-11'] || tags['latest']
|
|
||||||
if (!version) {
|
|
||||||
console.error('Could not determine latest pnpm version from npm dist-tags')
|
|
||||||
process.exit(1)
|
|
||||||
}
|
|
||||||
return version
|
|
||||||
}
|
|
||||||
|
|
||||||
function generateLock(filename, dependencies, name) {
|
|
||||||
const tmp = mkdtempSync(join(tmpdir(), 'pnpm-bootstrap-'))
|
|
||||||
try {
|
|
||||||
writeFileSync(join(tmp, 'package.json'), JSON.stringify({ private: true, dependencies }))
|
|
||||||
execSync('npm install --package-lock-only --ignore-scripts', { cwd: tmp, stdio: 'pipe' })
|
|
||||||
const lock = readFileSync(join(tmp, 'package-lock.json'), 'utf8')
|
|
||||||
const parsed = JSON.parse(lock)
|
|
||||||
parsed.name = name
|
|
||||||
writeFileSync(join(BOOTSTRAP_DIR, filename), JSON.stringify(parsed, null, 2) + '\n')
|
|
||||||
console.log(` ${filename} -> ${Object.values(dependencies)[0]}@${version}`)
|
|
||||||
} finally {
|
|
||||||
rmSync(tmp, { recursive: true, force: true })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
18
src/index.ts
18
src/index.ts
@@ -8,30 +8,28 @@ import pnpmInstall from './pnpm-install'
|
|||||||
import pruneStore from './pnpm-store-prune'
|
import pruneStore from './pnpm-store-prune'
|
||||||
|
|
||||||
async function main() {
|
async function main() {
|
||||||
|
const inputs = getInputs()
|
||||||
|
|
||||||
if (getState('is_post') === 'true') {
|
if (getState('is_post') === 'true') {
|
||||||
await runPost()
|
await runPost(inputs)
|
||||||
} else {
|
} else {
|
||||||
await runMain()
|
await runMain(inputs)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function runMain() {
|
async function runMain(inputs: Inputs) {
|
||||||
const inputs = getInputs()
|
|
||||||
saveState('inputs', inputs)
|
|
||||||
saveState('is_post', 'true')
|
saveState('is_post', 'true')
|
||||||
|
|
||||||
const binDest = await installPnpm(inputs)
|
await installPnpm(inputs)
|
||||||
if (binDest === undefined) return
|
|
||||||
console.log('Installation Completed!')
|
console.log('Installation Completed!')
|
||||||
setOutputs(inputs, binDest)
|
setOutputs(inputs)
|
||||||
|
|
||||||
await restoreCache(inputs)
|
await restoreCache(inputs)
|
||||||
|
|
||||||
pnpmInstall(inputs)
|
pnpmInstall(inputs)
|
||||||
}
|
}
|
||||||
|
|
||||||
async function runPost() {
|
async function runPost(inputs: Inputs) {
|
||||||
const inputs = JSON.parse(getState('inputs')) as Inputs
|
|
||||||
pruneStore(inputs)
|
pruneStore(inputs)
|
||||||
await saveCache(inputs)
|
await saveCache(inputs)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,42 +5,34 @@
|
|||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@pnpm/exe": "11.1.1"
|
"@pnpm/exe": "10.32.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@pnpm/exe": {
|
"node_modules/@pnpm/exe": {
|
||||||
"version": "11.1.1",
|
"version": "10.32.1",
|
||||||
"resolved": "https://registry.npmjs.org/@pnpm/exe/-/exe-11.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/@pnpm/exe/-/exe-10.32.1.tgz",
|
||||||
"integrity": "sha512-5mQnDW1NCBRRWA+cnGhQO+tIrfSfWm3/IyGxU88LnT+tzNW5UrwwKfjsnnYJToyAjIfdfEJtJKUxCvP+mhA+nQ==",
|
"integrity": "sha512-baEtwHeZwmZAdBuuDDL6tbdGg5KpxhPxr3QFfYTGXvY6ws+Z1bN0mQ7ZjcaXBSC1HuLpVXnZ6NsBiaZ2DMv4vg==",
|
||||||
"hasInstallScript": true,
|
"hasInstallScript": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
|
||||||
"@reflink/reflink": "0.1.19",
|
|
||||||
"detect-libc": "^2.0.3"
|
|
||||||
},
|
|
||||||
"bin": {
|
"bin": {
|
||||||
"pn": "pn",
|
"pnpm": "pnpm"
|
||||||
"pnpm": "pnpm",
|
|
||||||
"pnpx": "pnpx",
|
|
||||||
"pnx": "pnx"
|
|
||||||
},
|
},
|
||||||
"funding": {
|
"funding": {
|
||||||
"url": "https://opencollective.com/pnpm"
|
"url": "https://opencollective.com/pnpm"
|
||||||
},
|
},
|
||||||
"optionalDependencies": {
|
"optionalDependencies": {
|
||||||
"@pnpm/linux-arm64": "11.1.1",
|
"@pnpm/linux-arm64": "10.32.1",
|
||||||
"@pnpm/linux-x64": "11.1.1",
|
"@pnpm/linux-x64": "10.32.1",
|
||||||
"@pnpm/linuxstatic-arm64": "11.1.1",
|
"@pnpm/macos-arm64": "10.32.1",
|
||||||
"@pnpm/linuxstatic-x64": "11.1.1",
|
"@pnpm/macos-x64": "10.32.1",
|
||||||
"@pnpm/macos-arm64": "11.1.1",
|
"@pnpm/win-arm64": "10.32.1",
|
||||||
"@pnpm/win-arm64": "11.1.1",
|
"@pnpm/win-x64": "10.32.1"
|
||||||
"@pnpm/win-x64": "11.1.1"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@pnpm/linux-arm64": {
|
"node_modules/@pnpm/linux-arm64": {
|
||||||
"version": "11.1.1",
|
"version": "10.32.1",
|
||||||
"resolved": "https://registry.npmjs.org/@pnpm/linux-arm64/-/linux-arm64-11.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/@pnpm/linux-arm64/-/linux-arm64-10.32.1.tgz",
|
||||||
"integrity": "sha512-u9hs51XV0/gU5LLfNLoQsozGKIxNjxsh/0xPr+8Hny0M38psa4lBtwFvarL2bLToPIrtueQYi65LdlzRxITRyg==",
|
"integrity": "sha512-6uB0B+XvunQwHGzIMk2JCkl4Ur6BtM4XbJSwB/mgpWmXDoX/KTJmgx2lodcTjgJSGSySCHfIVuTR9sj/F2D4EA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -49,14 +41,17 @@
|
|||||||
"os": [
|
"os": [
|
||||||
"linux"
|
"linux"
|
||||||
],
|
],
|
||||||
|
"bin": {
|
||||||
|
"pnpm": "pnpm"
|
||||||
|
},
|
||||||
"funding": {
|
"funding": {
|
||||||
"url": "https://opencollective.com/pnpm"
|
"url": "https://opencollective.com/pnpm"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@pnpm/linux-x64": {
|
"node_modules/@pnpm/linux-x64": {
|
||||||
"version": "11.1.1",
|
"version": "10.32.1",
|
||||||
"resolved": "https://registry.npmjs.org/@pnpm/linux-x64/-/linux-x64-11.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/@pnpm/linux-x64/-/linux-x64-10.32.1.tgz",
|
||||||
"integrity": "sha512-yQO9i57oyJmIG22BjV7sqLUT2syKQohiku8yNZRgp7M6wsVkikpVLLVSpBifQnrI/P/roueKnWSUEESH1aPaoA==",
|
"integrity": "sha512-AM2tv23Fg7h+nV+adqA/SkZKUysSIEetHfBwYFl8ArgdgkqbGoQy0rAOdKYQBb920CqfexXfI8OA8kPCzRxYng==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -65,52 +60,17 @@
|
|||||||
"os": [
|
"os": [
|
||||||
"linux"
|
"linux"
|
||||||
],
|
],
|
||||||
"funding": {
|
"bin": {
|
||||||
"url": "https://opencollective.com/pnpm"
|
"pnpm": "pnpm"
|
||||||
}
|
},
|
||||||
},
|
|
||||||
"node_modules/@pnpm/linuxstatic-arm64": {
|
|
||||||
"version": "11.1.1",
|
|
||||||
"resolved": "https://registry.npmjs.org/@pnpm/linuxstatic-arm64/-/linuxstatic-arm64-11.1.1.tgz",
|
|
||||||
"integrity": "sha512-FUZB8L9Z8L5m88G0RTx5AsHFr5yUQPW+28zQdTNUWxiLwj11FW/fOLodYdcNYHdNJFepsZyqt3aRnpiqIdZb2g==",
|
|
||||||
"cpu": [
|
|
||||||
"arm64"
|
|
||||||
],
|
|
||||||
"libc": [
|
|
||||||
"musl"
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"optional": true,
|
|
||||||
"os": [
|
|
||||||
"linux"
|
|
||||||
],
|
|
||||||
"funding": {
|
|
||||||
"url": "https://opencollective.com/pnpm"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@pnpm/linuxstatic-x64": {
|
|
||||||
"version": "11.1.1",
|
|
||||||
"resolved": "https://registry.npmjs.org/@pnpm/linuxstatic-x64/-/linuxstatic-x64-11.1.1.tgz",
|
|
||||||
"integrity": "sha512-I/z56hfa1zM5F/Unup/1NrgsA+dcptsKQ2TjJLFz3wHKDx0RLrfF7DB0Rkpnr5IoAZ33v0GFZjlGhkOtc9VFGw==",
|
|
||||||
"cpu": [
|
|
||||||
"x64"
|
|
||||||
],
|
|
||||||
"libc": [
|
|
||||||
"musl"
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"optional": true,
|
|
||||||
"os": [
|
|
||||||
"linux"
|
|
||||||
],
|
|
||||||
"funding": {
|
"funding": {
|
||||||
"url": "https://opencollective.com/pnpm"
|
"url": "https://opencollective.com/pnpm"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@pnpm/macos-arm64": {
|
"node_modules/@pnpm/macos-arm64": {
|
||||||
"version": "11.1.1",
|
"version": "10.32.1",
|
||||||
"resolved": "https://registry.npmjs.org/@pnpm/macos-arm64/-/macos-arm64-11.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/@pnpm/macos-arm64/-/macos-arm64-10.32.1.tgz",
|
||||||
"integrity": "sha512-YQu6fC27F4jTIpXhF+4PdzOV7uSnVVG9KUxj5W+AFj0XFlUvBw+I1NsoPCY6uV1nccxWpIAZOTZtSj8+hWPb8w==",
|
"integrity": "sha512-Zr4JkhRbtGVsYgbuGZO0dq/6FPOn072Pdo0ubmqWtc14cUATKgAJD7efG03yqr3MLgtwFHgdtUzZ1WsaYAtUTA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -119,14 +79,36 @@
|
|||||||
"os": [
|
"os": [
|
||||||
"darwin"
|
"darwin"
|
||||||
],
|
],
|
||||||
|
"bin": {
|
||||||
|
"pnpm": "pnpm"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://opencollective.com/pnpm"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@pnpm/macos-x64": {
|
||||||
|
"version": "10.32.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/@pnpm/macos-x64/-/macos-x64-10.32.1.tgz",
|
||||||
|
"integrity": "sha512-Yk6q3oFDu//OniXJxfTSHo+aew1LX81FcbzJAtEkcCeTQ0SLbQT6J3QiOMNikp8n8IjNhsy+bn2bdkUxaw+akA==",
|
||||||
|
"cpu": [
|
||||||
|
"x64"
|
||||||
|
],
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"darwin"
|
||||||
|
],
|
||||||
|
"bin": {
|
||||||
|
"pnpm": "pnpm"
|
||||||
|
},
|
||||||
"funding": {
|
"funding": {
|
||||||
"url": "https://opencollective.com/pnpm"
|
"url": "https://opencollective.com/pnpm"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@pnpm/win-arm64": {
|
"node_modules/@pnpm/win-arm64": {
|
||||||
"version": "11.1.1",
|
"version": "10.32.1",
|
||||||
"resolved": "https://registry.npmjs.org/@pnpm/win-arm64/-/win-arm64-11.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/@pnpm/win-arm64/-/win-arm64-10.32.1.tgz",
|
||||||
"integrity": "sha512-2HvZut3IcKPxzIfOjBJ4677PaLIh57mWccL86O+q71QhO5emnQvht0CE19IoEyUIOEe1WjlN+Su/dD5k6CuGyg==",
|
"integrity": "sha512-P8rsP5IUetpYjr2iwggoswL2qUukYrJoToXWuMyo8immn58CsYxaXsHVQ1Oq1R3XMfmGGWTXLsiJuQ7H991MRg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -135,14 +117,17 @@
|
|||||||
"os": [
|
"os": [
|
||||||
"win32"
|
"win32"
|
||||||
],
|
],
|
||||||
|
"bin": {
|
||||||
|
"pnpm": "pnpm.exe"
|
||||||
|
},
|
||||||
"funding": {
|
"funding": {
|
||||||
"url": "https://opencollective.com/pnpm"
|
"url": "https://opencollective.com/pnpm"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@pnpm/win-x64": {
|
"node_modules/@pnpm/win-x64": {
|
||||||
"version": "11.1.1",
|
"version": "10.32.1",
|
||||||
"resolved": "https://registry.npmjs.org/@pnpm/win-x64/-/win-x64-11.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/@pnpm/win-x64/-/win-x64-10.32.1.tgz",
|
||||||
"integrity": "sha512-QXBIBErgPhGLovOVzTRIpHsejFKebyqlcF3fea/TfH87gkhN5yWH0WuTPRBoOWvpk6aNhjDW4RPUMx8RaPqxjw==",
|
"integrity": "sha512-i24GwbtBO8ojrhp8WWimX7NgZs0UKH1171oRt6qcRL+a+FxE0Eggv2y0KP7ZI7F3+LZMarwr3tnYsZryfciUOg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -151,177 +136,12 @@
|
|||||||
"os": [
|
"os": [
|
||||||
"win32"
|
"win32"
|
||||||
],
|
],
|
||||||
|
"bin": {
|
||||||
|
"pnpm": "pnpm.exe"
|
||||||
|
},
|
||||||
"funding": {
|
"funding": {
|
||||||
"url": "https://opencollective.com/pnpm"
|
"url": "https://opencollective.com/pnpm"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"node_modules/@reflink/reflink": {
|
|
||||||
"version": "0.1.19",
|
|
||||||
"resolved": "https://registry.npmjs.org/@reflink/reflink/-/reflink-0.1.19.tgz",
|
|
||||||
"integrity": "sha512-DmCG8GzysnCZ15bres3N5AHCmwBwYgp0As6xjhQ47rAUTUXxJiK+lLUxaGsX3hd/30qUpVElh05PbGuxRPgJwA==",
|
|
||||||
"license": "MIT",
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 10"
|
|
||||||
},
|
|
||||||
"optionalDependencies": {
|
|
||||||
"@reflink/reflink-darwin-arm64": "0.1.19",
|
|
||||||
"@reflink/reflink-darwin-x64": "0.1.19",
|
|
||||||
"@reflink/reflink-linux-arm64-gnu": "0.1.19",
|
|
||||||
"@reflink/reflink-linux-arm64-musl": "0.1.19",
|
|
||||||
"@reflink/reflink-linux-x64-gnu": "0.1.19",
|
|
||||||
"@reflink/reflink-linux-x64-musl": "0.1.19",
|
|
||||||
"@reflink/reflink-win32-arm64-msvc": "0.1.19",
|
|
||||||
"@reflink/reflink-win32-x64-msvc": "0.1.19"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@reflink/reflink-darwin-arm64": {
|
|
||||||
"version": "0.1.19",
|
|
||||||
"resolved": "https://registry.npmjs.org/@reflink/reflink-darwin-arm64/-/reflink-darwin-arm64-0.1.19.tgz",
|
|
||||||
"integrity": "sha512-ruy44Lpepdk1FqDz38vExBY/PVUsjxZA+chd9wozjUH9JjuDT/HEaQYA6wYN9mf041l0yLVar6BCZuWABJvHSA==",
|
|
||||||
"cpu": [
|
|
||||||
"arm64"
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"optional": true,
|
|
||||||
"os": [
|
|
||||||
"darwin"
|
|
||||||
],
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@reflink/reflink-darwin-x64": {
|
|
||||||
"version": "0.1.19",
|
|
||||||
"resolved": "https://registry.npmjs.org/@reflink/reflink-darwin-x64/-/reflink-darwin-x64-0.1.19.tgz",
|
|
||||||
"integrity": "sha512-By85MSWrMZa+c26TcnAy8SDk0sTUkYlNnwknSchkhHpGXOtjNDUOxJE9oByBnGbeuIE1PiQsxDG3Ud+IVV9yuA==",
|
|
||||||
"cpu": [
|
|
||||||
"x64"
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"optional": true,
|
|
||||||
"os": [
|
|
||||||
"darwin"
|
|
||||||
],
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@reflink/reflink-linux-arm64-gnu": {
|
|
||||||
"version": "0.1.19",
|
|
||||||
"resolved": "https://registry.npmjs.org/@reflink/reflink-linux-arm64-gnu/-/reflink-linux-arm64-gnu-0.1.19.tgz",
|
|
||||||
"integrity": "sha512-7P+er8+rP9iNeN+bfmccM4hTAaLP6PQJPKWSA4iSk2bNvo6KU6RyPgYeHxXmzNKzPVRcypZQTpFgstHam6maVg==",
|
|
||||||
"cpu": [
|
|
||||||
"arm64"
|
|
||||||
],
|
|
||||||
"libc": [
|
|
||||||
"glibc"
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"optional": true,
|
|
||||||
"os": [
|
|
||||||
"linux"
|
|
||||||
],
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@reflink/reflink-linux-arm64-musl": {
|
|
||||||
"version": "0.1.19",
|
|
||||||
"resolved": "https://registry.npmjs.org/@reflink/reflink-linux-arm64-musl/-/reflink-linux-arm64-musl-0.1.19.tgz",
|
|
||||||
"integrity": "sha512-37iO/Dp6m5DDaC2sf3zPtx/hl9FV3Xze4xoYidrxxS9bgP3S8ALroxRK6xBG/1TtfXKTvolvp+IjrUU6ujIGmA==",
|
|
||||||
"cpu": [
|
|
||||||
"arm64"
|
|
||||||
],
|
|
||||||
"libc": [
|
|
||||||
"musl"
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"optional": true,
|
|
||||||
"os": [
|
|
||||||
"linux"
|
|
||||||
],
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@reflink/reflink-linux-x64-gnu": {
|
|
||||||
"version": "0.1.19",
|
|
||||||
"resolved": "https://registry.npmjs.org/@reflink/reflink-linux-x64-gnu/-/reflink-linux-x64-gnu-0.1.19.tgz",
|
|
||||||
"integrity": "sha512-jbI8jvuYCaA3MVUdu8vLoLAFqC+iNMpiSuLbxlAgg7x3K5bsS8nOpTRnkLF7vISJ+rVR8W+7ThXlXlUQ93ulkw==",
|
|
||||||
"cpu": [
|
|
||||||
"x64"
|
|
||||||
],
|
|
||||||
"libc": [
|
|
||||||
"glibc"
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"optional": true,
|
|
||||||
"os": [
|
|
||||||
"linux"
|
|
||||||
],
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@reflink/reflink-linux-x64-musl": {
|
|
||||||
"version": "0.1.19",
|
|
||||||
"resolved": "https://registry.npmjs.org/@reflink/reflink-linux-x64-musl/-/reflink-linux-x64-musl-0.1.19.tgz",
|
|
||||||
"integrity": "sha512-e9FBWDe+lv7QKAwtKOt6A2W/fyy/aEEfr0g6j/hWzvQcrzHCsz07BNQYlNOjTfeytrtLU7k449H1PI95jA4OjQ==",
|
|
||||||
"cpu": [
|
|
||||||
"x64"
|
|
||||||
],
|
|
||||||
"libc": [
|
|
||||||
"musl"
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"optional": true,
|
|
||||||
"os": [
|
|
||||||
"linux"
|
|
||||||
],
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@reflink/reflink-win32-arm64-msvc": {
|
|
||||||
"version": "0.1.19",
|
|
||||||
"resolved": "https://registry.npmjs.org/@reflink/reflink-win32-arm64-msvc/-/reflink-win32-arm64-msvc-0.1.19.tgz",
|
|
||||||
"integrity": "sha512-09PxnVIQcd+UOn4WAW73WU6PXL7DwGS6wPlkMhMg2zlHHG65F3vHepOw06HFCq+N42qkaNAc8AKIabWvtk6cIQ==",
|
|
||||||
"cpu": [
|
|
||||||
"arm64"
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"optional": true,
|
|
||||||
"os": [
|
|
||||||
"win32"
|
|
||||||
],
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@reflink/reflink-win32-x64-msvc": {
|
|
||||||
"version": "0.1.19",
|
|
||||||
"resolved": "https://registry.npmjs.org/@reflink/reflink-win32-x64-msvc/-/reflink-win32-x64-msvc-0.1.19.tgz",
|
|
||||||
"integrity": "sha512-E//yT4ni2SyhwP8JRjVGWr3cbnhWDiPLgnQ66qqaanjjnMiu3O/2tjCPQXlcGc/DEYofpDc9fvhv6tALQsMV9w==",
|
|
||||||
"cpu": [
|
|
||||||
"x64"
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"optional": true,
|
|
||||||
"os": [
|
|
||||||
"win32"
|
|
||||||
],
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/detect-libc": {
|
|
||||||
"version": "2.1.2",
|
|
||||||
"resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz",
|
|
||||||
"integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==",
|
|
||||||
"license": "Apache-2.0",
|
|
||||||
"engines": {
|
|
||||||
"node": ">=8"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,22 +5,20 @@
|
|||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"pnpm": "11.1.1"
|
"pnpm": "latest"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/pnpm": {
|
"node_modules/pnpm": {
|
||||||
"version": "11.1.1",
|
"version": "10.32.1",
|
||||||
"resolved": "https://registry.npmjs.org/pnpm/-/pnpm-11.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/pnpm/-/pnpm-10.32.1.tgz",
|
||||||
"integrity": "sha512-0f319zxhe2T6GlaoHDyN/g6WbjOmAQqiVrUXrne+Idk+Ba/8DeGoOw5PKdVp9otEaujwaM1yR8C7PfD7TXvfmg==",
|
"integrity": "sha512-pwaTjw6JrBRWtlY+q07fHR+vM2jRGR/FxZeQ6W3JGORFarLmfWE94QQ9LoyB+HMD5rQNT/7KnfFe8a1Wc0jyvg==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"bin": {
|
"bin": {
|
||||||
"pn": "bin/pnpm.mjs",
|
"pnpm": "bin/pnpm.cjs",
|
||||||
"pnpm": "bin/pnpm.mjs",
|
"pnpx": "bin/pnpx.cjs"
|
||||||
"pnpx": "bin/pnpx.mjs",
|
|
||||||
"pnx": "bin/pnpx.mjs"
|
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=22.13"
|
"node": ">=18.12"
|
||||||
},
|
},
|
||||||
"funding": {
|
"funding": {
|
||||||
"url": "https://opencollective.com/pnpm"
|
"url": "https://opencollective.com/pnpm"
|
||||||
|
|||||||
@@ -4,15 +4,13 @@ import runSelfInstaller from './run'
|
|||||||
|
|
||||||
export { runSelfInstaller }
|
export { runSelfInstaller }
|
||||||
|
|
||||||
export async function install(inputs: Inputs): Promise<string | undefined> {
|
export async function install(inputs: Inputs) {
|
||||||
startGroup('Running self-installer...')
|
startGroup('Running self-installer...')
|
||||||
const { exitCode, binDest } = await runSelfInstaller(inputs)
|
const status = await runSelfInstaller(inputs)
|
||||||
endGroup()
|
endGroup()
|
||||||
if (exitCode) {
|
if (status) {
|
||||||
setFailed(`Something went wrong, self-installer exits with code ${exitCode}`)
|
return setFailed(`Something went wrong, self-installer exits with code ${status}`)
|
||||||
return undefined
|
|
||||||
}
|
}
|
||||||
return binDest
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default install
|
export default install
|
||||||
|
|||||||
@@ -12,18 +12,8 @@ import exeLock from './bootstrap/exe-lock.json'
|
|||||||
const BOOTSTRAP_PNPM_PACKAGE_JSON = JSON.stringify({ private: true, dependencies: { pnpm: pnpmLock.packages['node_modules/pnpm'].version } })
|
const BOOTSTRAP_PNPM_PACKAGE_JSON = JSON.stringify({ private: true, dependencies: { pnpm: pnpmLock.packages['node_modules/pnpm'].version } })
|
||||||
const BOOTSTRAP_EXE_PACKAGE_JSON = JSON.stringify({ private: true, dependencies: { '@pnpm/exe': exeLock.packages['node_modules/@pnpm/exe'].version } })
|
const BOOTSTRAP_EXE_PACKAGE_JSON = JSON.stringify({ private: true, dependencies: { '@pnpm/exe': exeLock.packages['node_modules/@pnpm/exe'].version } })
|
||||||
|
|
||||||
export interface SelfInstallerResult {
|
export async function runSelfInstaller(inputs: Inputs): Promise<number> {
|
||||||
exitCode: number
|
const { version, dest, packageJsonFile, standalone } = inputs
|
||||||
binDest: string
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function runSelfInstaller(inputs: Inputs): Promise<SelfInstallerResult> {
|
|
||||||
const { version, dest, packageJsonFile } = inputs
|
|
||||||
|
|
||||||
// pnpm v11 requires Node >= 22.13; use standalone (exe) bootstrap which
|
|
||||||
// bundles its own Node.js when the system Node is too old
|
|
||||||
const systemNode = await getSystemNodeVersion()
|
|
||||||
const standalone = inputs.standalone || systemNode.major < 22 || (systemNode.major === 22 && systemNode.minor < 13)
|
|
||||||
|
|
||||||
// Install bootstrap pnpm via npm (integrity verified by committed lockfile)
|
// Install bootstrap pnpm via npm (integrity verified by committed lockfile)
|
||||||
await rm(dest, { recursive: true, force: true })
|
await rm(dest, { recursive: true, force: true })
|
||||||
@@ -34,110 +24,73 @@ export async function runSelfInstaller(inputs: Inputs): Promise<SelfInstallerRes
|
|||||||
await writeFile(path.join(dest, 'package.json'), packageJson)
|
await writeFile(path.join(dest, 'package.json'), packageJson)
|
||||||
await writeFile(path.join(dest, 'package-lock.json'), JSON.stringify(lockfile))
|
await writeFile(path.join(dest, 'package-lock.json'), JSON.stringify(lockfile))
|
||||||
|
|
||||||
// Append the action's node directory to PATH so npm's
|
const npmExitCode = await runCommand('npm', ['ci'], { cwd: dest })
|
||||||
// `#!/usr/bin/env node` shebang resolves on runners (e.g. GHE
|
|
||||||
// self-hosted) where node isn't already on PATH. Append (not
|
|
||||||
// prepend) so a user-installed toolchain on PATH — e.g. from a
|
|
||||||
// prior `setup-node` step — keeps precedence; otherwise the
|
|
||||||
// runner-bundled node would shadow it and pair the user's npm
|
|
||||||
// with a mismatched node version. npm itself is resolved via
|
|
||||||
// PATH — on the GitHub Actions runner it is not co-located with
|
|
||||||
// `process.execPath`.
|
|
||||||
const nodeDir = path.dirname(process.execPath)
|
|
||||||
// On Windows, the PATH key casing varies; search case-insensitively.
|
|
||||||
const pathKey = Object.keys(process.env).find(k => k.toUpperCase() === 'PATH') ?? 'PATH'
|
|
||||||
const currentPath = process.env[pathKey]
|
|
||||||
const npmEnv = { ...process.env, [pathKey]: currentPath ? currentPath + path.delimiter + nodeDir : nodeDir }
|
|
||||||
const npmExitCode = await runCommand('npm', ['ci'], { cwd: dest, env: npmEnv })
|
|
||||||
if (npmExitCode !== 0) {
|
if (npmExitCode !== 0) {
|
||||||
return { exitCode: npmExitCode, binDest: path.join(dest, 'node_modules', '.bin') }
|
return npmExitCode
|
||||||
}
|
}
|
||||||
|
|
||||||
// On Windows with standalone mode, npm's .bin shims can't properly
|
const pnpmHome = path.join(dest, 'node_modules', '.bin')
|
||||||
// execute the extensionless @pnpm/exe native binaries. Add the
|
|
||||||
// @pnpm/exe directory directly to PATH so pnpm.exe is found natively.
|
|
||||||
const pnpmHome = standalone && process.platform === 'win32'
|
|
||||||
? path.join(dest, 'node_modules', '@pnpm', 'exe')
|
|
||||||
: path.join(dest, 'node_modules', '.bin')
|
|
||||||
// PNPM_HOME/bin is where `pnpm self-update` places the target version
|
|
||||||
// binary. It must have higher PATH precedence than pnpmHome (which
|
|
||||||
// contains the bootstrap binary) so the self-updated version is found
|
|
||||||
// first. The bootstrap pnpm is invoked via absolute path, not PATH,
|
|
||||||
// so this ordering does not affect the bootstrap step.
|
|
||||||
addPath(pnpmHome)
|
addPath(pnpmHome)
|
||||||
addPath(path.join(pnpmHome, 'bin'))
|
addPath(path.join(pnpmHome, 'bin'))
|
||||||
exportVariable('PNPM_HOME', pnpmHome)
|
exportVariable('PNPM_HOME', pnpmHome)
|
||||||
|
|
||||||
// Ensure pnpm bin link exists — npm ci sometimes doesn't create it
|
// Ensure pnpm bin link exists — npm ci sometimes doesn't create it
|
||||||
if (process.platform !== 'win32') {
|
const pnpmBinLink = path.join(pnpmHome, 'pnpm')
|
||||||
const pnpmBinLink = path.join(dest, 'node_modules', '.bin', 'pnpm')
|
if (!existsSync(pnpmBinLink)) {
|
||||||
if (!existsSync(pnpmBinLink)) {
|
await mkdir(pnpmHome, { recursive: true })
|
||||||
await mkdir(path.join(dest, 'node_modules', '.bin'), { recursive: true })
|
const target = standalone
|
||||||
const target = standalone
|
? path.join('..', '@pnpm', 'exe', 'pnpm')
|
||||||
? path.join('..', '@pnpm', 'exe', 'pnpm')
|
: path.join('..', 'pnpm', 'bin', 'pnpm.cjs')
|
||||||
: path.join('..', 'pnpm', 'bin', 'pnpm.mjs')
|
await symlink(target, pnpmBinLink)
|
||||||
await symlink(target, pnpmBinLink)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const bootstrapPnpm = standalone
|
const bootstrapPnpm = standalone
|
||||||
? path.join(dest, 'node_modules', '@pnpm', 'exe', process.platform === 'win32' ? 'pnpm.exe' : 'pnpm')
|
? path.join(dest, 'node_modules', '@pnpm', 'exe', 'pnpm')
|
||||||
: path.join(dest, 'node_modules', 'pnpm', 'bin', 'pnpm.mjs')
|
: path.join(dest, 'node_modules', 'pnpm', 'bin', 'pnpm.cjs')
|
||||||
|
|
||||||
// Self-update the bootstrap to the requested pnpm version. readTargetVersion
|
// Determine the target version
|
||||||
// either returns a value or throws, so this always runs.
|
|
||||||
const targetVersion = readTargetVersion({ version, packageJsonFile })
|
const targetVersion = readTargetVersion({ version, packageJsonFile })
|
||||||
const cmd = standalone ? bootstrapPnpm : process.execPath
|
|
||||||
const args = standalone ? ['self-update', targetVersion] : [bootstrapPnpm, 'self-update', targetVersion]
|
if (targetVersion) {
|
||||||
const exitCode = await runCommand(cmd, args, { cwd: dest })
|
const cmd = standalone ? bootstrapPnpm : process.execPath
|
||||||
if (exitCode !== 0) {
|
const args = standalone ? ['self-update', targetVersion] : [bootstrapPnpm, 'self-update', targetVersion]
|
||||||
return { exitCode, binDest: pnpmHome }
|
const exitCode = await runCommand(cmd, args, { cwd: dest })
|
||||||
|
if (exitCode !== 0) {
|
||||||
|
return exitCode
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// self-update writes the target pnpm/pnpx into PNPM_HOME/bin, leaving
|
|
||||||
// the bootstrap symlinks in pnpmHome pointing at the old version. Use
|
return 0
|
||||||
// PNPM_HOME/bin so consumers of the bin_dest output (e.g.
|
|
||||||
// `${steps.pnpm.outputs.bin_dest}/pnpm`) invoke the requested version.
|
|
||||||
//
|
|
||||||
// When the requested version resolves to the bootstrap version, self-update
|
|
||||||
// is a no-op and PNPM_HOME/bin is not created — fall back to pnpmHome,
|
|
||||||
// whose symlinks already point at the right version.
|
|
||||||
const updatedBinDir = path.join(pnpmHome, 'bin')
|
|
||||||
return { exitCode: 0, binDest: existsSync(updatedBinDir) ? updatedBinDir : pnpmHome }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function readTargetVersion(opts: {
|
function readTargetVersion(opts: {
|
||||||
readonly version?: string | undefined
|
readonly version?: string | undefined
|
||||||
readonly packageJsonFile: string
|
readonly packageJsonFile: string
|
||||||
}): string {
|
}): string | undefined {
|
||||||
const { version, packageJsonFile } = opts
|
const { version, packageJsonFile } = opts
|
||||||
const { GITHUB_WORKSPACE } = process.env
|
const { GITHUB_WORKSPACE } = process.env
|
||||||
|
|
||||||
let packageManager: string | undefined
|
let packageManager: unknown
|
||||||
let devEngines: { packageManager?: { name?: string; version?: string } } | undefined
|
|
||||||
|
|
||||||
if (GITHUB_WORKSPACE) {
|
if (GITHUB_WORKSPACE) {
|
||||||
try {
|
try {
|
||||||
const content = readFileSync(path.join(GITHUB_WORKSPACE, packageJsonFile), 'utf8');
|
const content = readFileSync(path.join(GITHUB_WORKSPACE, packageJsonFile), 'utf8');
|
||||||
const manifest = packageJsonFile.endsWith(".yaml")
|
({ packageManager } = packageJsonFile.endsWith(".yaml")
|
||||||
? parseYaml(content, { merge: true })
|
? parseYaml(content, { merge: true })
|
||||||
: JSON.parse(content)
|
: JSON.parse(content)
|
||||||
packageManager = manifest.packageManager
|
)
|
||||||
devEngines = manifest.devEngines
|
|
||||||
} catch (error: unknown) {
|
} catch (error: unknown) {
|
||||||
// Swallow error if package.json doesn't exist in root
|
// Swallow error if package.json doesn't exist in root
|
||||||
if (!util.types.isNativeError(error) || !('code' in error) || error.code !== 'ENOENT') throw error
|
if (!util.types.isNativeError(error) || !('code' in error) || error.code !== 'ENOENT') throw error
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// packageManager is always exact `pnpm@<version>[+<integrity>]` per spec.
|
|
||||||
// Strip the integrity hash for self-update.
|
|
||||||
const packageManagerVersion =
|
|
||||||
typeof packageManager === 'string' && packageManager.startsWith('pnpm@')
|
|
||||||
? packageManager.slice('pnpm@'.length).split('+')[0]
|
|
||||||
: undefined
|
|
||||||
|
|
||||||
if (version) {
|
if (version) {
|
||||||
if (packageManagerVersion && packageManagerVersion !== version) {
|
if (
|
||||||
|
typeof packageManager === 'string' &&
|
||||||
|
packageManager.startsWith('pnpm@') &&
|
||||||
|
packageManager.replace('pnpm@', '') !== version
|
||||||
|
) {
|
||||||
throw new Error(`Multiple versions of pnpm specified:
|
throw new Error(`Multiple versions of pnpm specified:
|
||||||
- version ${version} in the GitHub Action config with the key "version"
|
- version ${version} in the GitHub Action config with the key "version"
|
||||||
- version ${packageManager} in the package.json with the key "packageManager"
|
- version ${packageManager} in the package.json with the key "packageManager"
|
||||||
@@ -147,22 +100,9 @@ Remove one of these versions to avoid version mismatch errors like ERR_PNPM_BAD_
|
|||||||
return version
|
return version
|
||||||
}
|
}
|
||||||
|
|
||||||
// Self-update the bootstrap pnpm to the version pinned in package.json so
|
if (typeof packageManager === 'string' && packageManager.startsWith('pnpm@')) {
|
||||||
// PATH-resolved `pnpm` (and the bin_dest output) reflect the target
|
// pnpm will handle version management via packageManager field
|
||||||
// version. Without this, `pnpm store path` runs as the bootstrap and
|
return undefined
|
||||||
// reports a different STORE_VERSION than the one the user's actual
|
|
||||||
// install writes to — breaking cache: true and actions/setup-node's
|
|
||||||
// `cache: pnpm` on cold caches (issue #233).
|
|
||||||
//
|
|
||||||
// devEngines.packageManager takes priority over packageManager, matching
|
|
||||||
// pnpm's getWantedPackageManager logic. `pnpm self-update` accepts both
|
|
||||||
// exact versions and semver ranges, so we pass either through directly.
|
|
||||||
if (devEngines?.packageManager?.name === 'pnpm' && devEngines.packageManager.version) {
|
|
||||||
return devEngines.packageManager.version
|
|
||||||
}
|
|
||||||
|
|
||||||
if (packageManagerVersion) {
|
|
||||||
return packageManagerVersion
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!GITHUB_WORKSPACE) {
|
if (!GITHUB_WORKSPACE) {
|
||||||
@@ -175,28 +115,13 @@ Otherwise, please specify the pnpm version in the action configuration.`)
|
|||||||
throw new Error(`No pnpm version is specified.
|
throw new Error(`No pnpm version is specified.
|
||||||
Please specify it by one of the following ways:
|
Please specify it by one of the following ways:
|
||||||
- in the GitHub Action config with the key "version"
|
- in the GitHub Action config with the key "version"
|
||||||
- in the package.json with the key "packageManager"
|
- in the package.json with the key "packageManager"`)
|
||||||
- in the package.json with the key "devEngines.packageManager"`)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function getSystemNodeVersion(): Promise<{ major: number; minor: number }> {
|
function runCommand(cmd: string, args: string[], opts: { cwd: string }): Promise<number> {
|
||||||
return new Promise((resolve) => {
|
|
||||||
const cp = spawn('node', ['--version'], { stdio: ['pipe', 'pipe', 'pipe'], shell: process.platform === 'win32' })
|
|
||||||
let output = ''
|
|
||||||
cp.stdout.on('data', (data: Buffer) => { output += data.toString() })
|
|
||||||
cp.on('close', () => {
|
|
||||||
const match = output.match(/^v(\d+)\.(\d+)/)
|
|
||||||
resolve(match ? { major: parseInt(match[1], 10), minor: parseInt(match[2], 10) } : { major: 0, minor: 0 })
|
|
||||||
})
|
|
||||||
cp.on('error', () => resolve({ major: 0, minor: 0 }))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
function runCommand(cmd: string, args: string[], opts: { cwd: string; env?: Record<string, string | undefined> }): Promise<number> {
|
|
||||||
return new Promise<number>((resolve, reject) => {
|
return new Promise<number>((resolve, reject) => {
|
||||||
const cp = spawn(cmd, args, {
|
const cp = spawn(cmd, args, {
|
||||||
cwd: opts.cwd,
|
cwd: opts.cwd,
|
||||||
env: opts.env,
|
|
||||||
stdio: ['pipe', 'inherit', 'inherit'],
|
stdio: ['pipe', 'inherit', 'inherit'],
|
||||||
shell: process.platform === 'win32',
|
shell: process.platform === 'win32',
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
import { setOutput } from '@actions/core'
|
import { setOutput, addPath } from '@actions/core'
|
||||||
import { Inputs } from '../inputs'
|
import { Inputs } from '../inputs'
|
||||||
|
import { getBinDest } from '../utils'
|
||||||
|
|
||||||
export function setOutputs(inputs: Inputs, binDest: string) {
|
export function setOutputs(inputs: Inputs) {
|
||||||
// NOTE: addPath is already called in installPnpm — do not call it again
|
const binDest = getBinDest(inputs)
|
||||||
// here, as a second addPath would shadow the correct entry on Windows.
|
addPath(binDest)
|
||||||
setOutput('dest', inputs.dest)
|
setOutput('dest', inputs.dest)
|
||||||
setOutput('bin_dest', binDest)
|
setOutput('bin_dest', binDest)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,11 @@
|
|||||||
import { setFailed, startGroup, endGroup } from '@actions/core'
|
import { setFailed, startGroup, endGroup } from '@actions/core'
|
||||||
import { spawnSync } from 'child_process'
|
import { spawnSync } from 'child_process'
|
||||||
import { Inputs } from '../inputs'
|
import { Inputs } from '../inputs'
|
||||||
|
import { patchPnpmEnv } from '../utils'
|
||||||
|
|
||||||
export function runPnpmInstall(inputs: Inputs) {
|
export function runPnpmInstall(inputs: Inputs) {
|
||||||
|
const env = patchPnpmEnv(inputs)
|
||||||
|
|
||||||
for (const options of inputs.runInstall) {
|
for (const options of inputs.runInstall) {
|
||||||
const args = ['install']
|
const args = ['install']
|
||||||
if (options.recursive) args.unshift('recursive')
|
if (options.recursive) args.unshift('recursive')
|
||||||
@@ -11,16 +14,11 @@ export function runPnpmInstall(inputs: Inputs) {
|
|||||||
const cmdStr = ['pnpm', ...args].join(' ')
|
const cmdStr = ['pnpm', ...args].join(' ')
|
||||||
startGroup(`Running ${cmdStr}...`)
|
startGroup(`Running ${cmdStr}...`)
|
||||||
|
|
||||||
// spawnSync inherits process.env, which already has $PNPM_HOME/bin and
|
|
||||||
// $PNPM_HOME prepended via addPath() in install-pnpm. Do NOT pass a
|
|
||||||
// hand-patched env that adds node_modules/.bin to the front — on
|
|
||||||
// Windows standalone, .bin/pnpm.cmd is an npm shim pointing at the
|
|
||||||
// BOOTSTRAP pnpm, which would shadow the self-updated one and break
|
|
||||||
// newer-pnpm-only behavior.
|
|
||||||
const { error, status } = spawnSync('pnpm', args, {
|
const { error, status } = spawnSync('pnpm', args, {
|
||||||
stdio: 'inherit',
|
stdio: 'inherit',
|
||||||
cwd: options.cwd,
|
cwd: options.cwd,
|
||||||
shell: true,
|
shell: true,
|
||||||
|
env,
|
||||||
})
|
})
|
||||||
|
|
||||||
endGroup()
|
endGroup()
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import { warning, startGroup, endGroup } from '@actions/core'
|
import { warning, startGroup, endGroup } from '@actions/core'
|
||||||
import { spawnSync } from 'child_process'
|
import { spawnSync } from 'child_process'
|
||||||
import { Inputs } from '../inputs'
|
import { Inputs } from '../inputs'
|
||||||
|
import { patchPnpmEnv } from '../utils'
|
||||||
|
|
||||||
export function pruneStore(inputs: Inputs) {
|
export function pruneStore(inputs: Inputs) {
|
||||||
if (inputs.runInstall.length === 0) {
|
if (inputs.runInstall.length === 0) {
|
||||||
@@ -9,11 +10,10 @@ export function pruneStore(inputs: Inputs) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
startGroup('Running pnpm store prune...')
|
startGroup('Running pnpm store prune...')
|
||||||
// spawnSync inherits process.env (which has the right PATH from addPath
|
|
||||||
// in install-pnpm). See pnpm-install/index.ts for the rationale.
|
|
||||||
const { error, status } = spawnSync('pnpm', ['store', 'prune'], {
|
const { error, status } = spawnSync('pnpm', ['store', 'prune'], {
|
||||||
stdio: 'inherit',
|
stdio: 'inherit',
|
||||||
shell: true,
|
shell: true,
|
||||||
|
env: patchPnpmEnv(inputs),
|
||||||
})
|
})
|
||||||
endGroup()
|
endGroup()
|
||||||
|
|
||||||
|
|||||||
10
src/utils/index.ts
Normal file
10
src/utils/index.ts
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
import path from 'path'
|
||||||
|
import process from 'process'
|
||||||
|
import { Inputs } from '../inputs'
|
||||||
|
|
||||||
|
export const getBinDest = (inputs: Inputs): string => path.join(inputs.dest, 'node_modules', '.bin')
|
||||||
|
|
||||||
|
export const patchPnpmEnv = (inputs: Inputs): NodeJS.ProcessEnv => ({
|
||||||
|
...process.env,
|
||||||
|
PATH: path.join(getBinDest(inputs), 'bin') + path.delimiter + getBinDest(inputs) + path.delimiter + process.env.PATH,
|
||||||
|
})
|
||||||
Reference in New Issue
Block a user