47 Commits

Author SHA1 Message Date
3236b209b5 Merge branch 'v2' 2021-03-27 16:56:27 +07:00
2124926520 Update README.md 2021-03-23 12:53:37 +07:00
ba9826e81c Update test workflow 2021-03-23 12:53:25 +07:00
7c4472dbcf Forgot to specify working directory 2021-03-23 12:48:54 +07:00
0db7fb9961 Add .pnpm-store to .gitignore 2021-03-23 12:48:07 +07:00
aefcd1e623 Use pnpm.js to install pnpm 2021-03-23 12:42:43 +07:00
b7b9d6344b Instruct renovate to ignore ajv updates 2021-03-23 11:47:55 +07:00
7e61ea5847 Add "$schema" to renovate.json 2021-03-23 11:46:56 +07:00
3a2c7247e1 Fix js-yaml usage 2021-03-23 11:43:49 +07:00
2a105263a5 Update dependencies 2021-03-23 11:43:36 +07:00
9facd4db8e Merge pull request #9 from hildjj/cache-docs
Adds documentation hints for caching
2021-02-18 08:31:04 +07:00
6c1466d327 Make some tweaks to example code snippets 2021-02-18 08:30:07 +07:00
b26427e53e Adds documentation hints for caching. Fixes #8. 2021-02-17 09:45:25 -07:00
086f5bd3b6 Update README.md
close https://github.com/pnpm/action-setup/issues/6
2020-10-19 14:10:51 +07:00
bdb2a5ee76 Enable Dependabot 2020-10-05 22:16:14 +07:00
8e1abe543f Update dist/index.js 2020-10-05 21:51:43 +07:00
af6247d08a Fix https://github.com/pnpm/action-setup/issues/5 2020-10-05 21:45:45 +07:00
f87c8a916e Fix TypeScript error 2020-10-05 21:45:20 +07:00
d6790970e0 Update dependencies 2020-10-05 21:44:44 +07:00
fa62771e12 Add workflow_dispatch event 2020-08-27 08:25:30 +07:00
935101478d v1.2.0 2020-05-10 12:17:17 +07:00
4abca36d2a Merge pull request #2 from pnpm/install-command
Support install command
2020-05-10 12:16:23 +07:00
56013f801f Add more examples 2020-05-10 12:14:27 +07:00
fb99aeb8e3 Describe run_install in more details 2020-05-10 12:08:45 +07:00
b78eaea668 Describe default values 2020-05-10 12:00:23 +07:00
83681c63a7 Make logs foldable 2020-05-09 21:50:47 +07:00
6eb237a86d Only warn on failure of post action 2020-05-09 21:43:19 +07:00
b1febf84ed Fix pruneStore 2020-05-09 21:41:25 +07:00
2546768411 Add matrix.os to name 2020-05-09 21:35:09 +07:00
8cdddb18c5 Try fixing 2020-05-09 21:33:18 +07:00
4457a83971 Test with run_install 2020-05-09 21:26:17 +07:00
c8fc1974e1 Run pnpm store prune post action 2020-05-09 21:15:50 +07:00
291e58ad85 Enable post action 2020-05-09 21:02:32 +07:00
1790ca7f76 Add pnpm install 2020-05-09 20:24:52 +07:00
9a1617cf46 Rename install to install-pnpm 2020-05-09 20:03:45 +07:00
6fe65dc1af Move run-install to its own module 2020-05-09 20:01:25 +07:00
91d3d73121 Correct validation error message 2020-05-09 19:52:48 +07:00
7a5d08caa7 Update run.sh 2020-05-09 19:46:36 +07:00
e373fffa0a Use ajv to validate schema 2020-05-09 19:45:31 +07:00
bb24f595c2 Add schema generator 2020-05-09 19:24:07 +07:00
d44b8c5e53 Add run_install input 2020-05-09 19:13:46 +07:00
4b13327683 Re-add dist/index.js 2020-05-09 18:27:15 +07:00
49ba4cbc60 Delete dist/index.js 2020-05-09 18:26:16 +07:00
9649109f2e Fix .gitattributes 2020-05-09 08:13:09 +07:00
087311f996 refactor: Remove then 2020-05-08 21:55:03 +07:00
9979c3d928 Use node-fetch 2020-05-08 21:34:25 +07:00
738fb9213f Fix README.md 2020-05-08 21:23:23 +07:00
26 changed files with 1086 additions and 712 deletions

2
.gitattributes vendored
View File

@ -1,2 +1,2 @@
* text=auto * text=auto
dist/index.js -text /dist/index.js binary

10
.github/dependabot.yml vendored Normal file
View File

@ -0,0 +1,10 @@
version: 2
updates:
- package-ecosystem: github-actions
directory: "/"
schedule:
interval: weekly
open-pull-requests-limit: 10
labels:
- dependabot
- github-actions

View File

@ -3,6 +3,7 @@ name: Test Action
on: on:
- push - push
- pull_request - pull_request
- workflow_dispatch
jobs: jobs:
test_default_inputs: test_default_inputs:
@ -57,11 +58,65 @@ jobs:
with: with:
version: 4.11.1 version: 4.11.1
dest: ~/test/pnpm dest: ~/test/pnpm
bin_dest: ~/test/pnpm/.bin
registry: http://registry.yarnpkg.com/
- name: 'Test: which' - name: 'Test: which'
run: which pnpm && which pnpx run: which pnpm && which pnpx
- name: 'Test: install' - name: 'Test: install'
run: pnpm install run: pnpm install
test_run_install:
name: 'Test with run_install (${{ matrix.run_install.name }}, ${{ matrix.os }})'
runs-on: ${{ matrix.os }}
strategy:
fail-fast: false
matrix:
pnpm:
- 4.11.1
os:
- ubuntu-latest
- macos-latest
- windows-latest
run_install:
- name: 'null'
value: 'null'
- name: 'empty object'
value: '{}'
- name: 'recursive'
value: |
recursive: true
- name: 'global'
value: |
args:
- --global
- --global-dir=./pnpm-global
- npm
- yarn
- pnpm
- name: 'array'
value: |
- {}
- recursive: true
- args:
- --global
- --global-dir=./pnpm-global
- npm
- yarn
- pnpm
steps:
- uses: actions/checkout@v2
- name: Run the action
uses: ./
with:
version: 4.11.1
run_install: ${{ matrix.run_install.value }}
- name: 'Test: which'
run: which pnpm; which pnpx
- name: 'Test: install'
run: pnpm install

1
.gitignore vendored
View File

@ -8,3 +8,4 @@ temp
*.temp *.temp
tmp.* tmp.*
temp.* temp.*
.pnpm-store

View File

@ -12,13 +12,27 @@ Install PNPM package manager.
**Optional** Where to store PNPM files. **Optional** Where to store PNPM files.
### `bin_dest` ### `run_install`
**Optional** Where to store executables (`pnpm` and `pnpx` commands). **Optional** (_default:_ `null`) If specified, run `pnpm install`.
### `registry` If `run_install` is either `null` or `false`, pnpm will not install any npm package.
**Optional** Registry to download PNPM from. If `run_install` is `true`, pnpm will install dependencies recursively.
If `run_install` is a YAML string representation of either an object or an array, pnpm will execute every install commands.
#### `run_install.recursive`
**Optional** (_type:_ `boolean`, _default:_ `false`) Whether to use `pnpm recursive install`.
#### `run_install.cwd`
**Optional** (_type:_ `string`) Working directory when run `pnpm [recursive] install`.
#### `run_install.args`
**Optional** (_type:_ `string[]`) Additional arguments after `pnpm [recursive] install`, e.g. `[--frozen-lockfile, --strict-peer-dependencies]`.
## Outputs ## Outputs
@ -28,10 +42,28 @@ Expanded path of inputs#dest.
### `bin_dest` ### `bin_dest`
Expanded path of inputs@bin_dest. Location of `pnpm` and `pnpx` command.
## Usage example ## Usage example
### Just install PNPM
```yaml
on:
- push
- pull_request
jobs:
runs-on: ubuntu-latest
steps:
- uses: pnpm/action-setup@v1.2.1
with:
version: 5.17.2
```
### Install PNPM and a few NPM packages
```yaml ```yaml
on: on:
- push - push
@ -43,14 +75,47 @@ jobs:
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- uses: pnpm/action-setup@v1.1.0 - uses: pnpm/action-setup@v1.2.1
with: with:
version: 4.11.1 version: 5.17.2
run_install: |
- name: Install dependencies - recursive: true
run: pnpm install args: [--frozen-lockfile, --strict-peer-dependencies]
- args: [--global, gulp, prettier, typescript]
``` ```
### Use cache to reduce installation time
```yaml
on:
- push
- pull_request
jobs:
runs-on: ubuntu-latest
steps:
build:
- uses: actions/checkout@v2
- name: Cache pnpm modules
uses: actions/cache@v2
env:
cache-name: cache-pnpm-modules
with:
path: ~/.pnpm-store
key: ${{ runner.os }}-build-${{ env.cache-name }}-${{ matrix.node-version }}-${{ hashFiles('**/package.json') }}
restore-keys: |
${{ runner.os }}-build-${{ env.cache-name }}-${{ matrix.node-version }}-
- uses: pnpm/action-setup@v1.2.1
with:
version: 5.17.2
run_install: true
```
**Note:** You don't need to run `pnpm store prune` at the end; post-action has already taken care of that.
## 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.

View File

@ -11,14 +11,11 @@ inputs:
description: Where to store PNPM files description: Where to store PNPM files
required: false required: false
default: ~/setup-pnpm default: ~/setup-pnpm
bin_dest: run_install:
description: Where to store executables (pnpm and pnpx commands) description: If specified, run `pnpm install`
required: false required: false
default: ~/setup-pnpm/.bin default: 'null'
registry:
description: Registry to download PNPM from
required: false
default: https://registry.npmjs.com
runs: runs:
using: node12 using: node12
main: dist/index.js main: dist/index.js
post: dist/index.js

2
dist/index.js vendored

File diff suppressed because one or more lines are too long

View File

@ -1,20 +1,27 @@
{ {
"private": true, "private": true,
"scripts": { "scripts": {
"build:schemas": "ts-schema-autogen generate",
"build:ncc": "ncc build --minify --no-source-map-register --no-cache dist/tsc/index.js --out dist/", "build:ncc": "ncc build --minify --no-source-map-register --no-cache dist/tsc/index.js --out dist/",
"build": "tsc && pnpm run build:ncc", "build": "pnpm run build:schemas && tsc && pnpm run build:ncc",
"start": "pnpm run build && sh ./run.sh" "start": "pnpm run build && sh ./run.sh"
}, },
"dependencies": { "dependencies": {
"download": "^8.0.0", "node-fetch": "^2.6.1",
"expand-tilde": "^2.0.2", "expand-tilde": "^2.0.2",
"@actions/core": "^1.2.4", "js-yaml": "^4.0.0",
"@types/download": "^6.2.4", "ajv": "^6.12.5",
"fs-extra": "^9.1.0",
"@actions/core": "^1.2.6",
"@types/expand-tilde": "^2.0.0", "@types/expand-tilde": "^2.0.0",
"@types/node": "^13.13.5" "@types/node-fetch": "^2.5.8",
"@types/js-yaml": "^4.0.0",
"@types/fs-extra": "^9.0.8",
"@types/node": "^14.14.35"
}, },
"devDependencies": { "devDependencies": {
"typescript": "^3.8.3", "typescript": "^4.2.3",
"@zeit/ncc": "^0.22.1" "@ts-schema-autogen/cli": "^0.1.2",
"@vercel/ncc": "^0.27.0"
} }
} }

1323
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@ -1,6 +1,10 @@
{ {
"$schema": "https://docs.renovatebot.com/renovate-schema.json",
"pinVersions": false, "pinVersions": false,
"extends": [ "extends": [
"config:base" "config:base"
],
"ignoreDeps": [
"ajv"
] ]
} }

3
run.sh
View File

@ -3,6 +3,5 @@
export HOME="$(pwd)" export HOME="$(pwd)"
export INPUT_VERSION=4.11.1 export INPUT_VERSION=4.11.1
export INPUT_DEST='~/pnpm.temp' export INPUT_DEST='~/pnpm.temp'
export INPUT_BIN_DEST='~/pnpm.temp/.bin' export INPUT_RUN_INSTALL=null
export INPUT_REGISTRY=https://registry.npmjs.com
exec node dist/index.js exec node dist/index.js

View File

@ -1,14 +1,19 @@
import { setFailed } from '@actions/core' import { setFailed, saveState, getState } from '@actions/core'
import getInputs from './inputs' import getInputs from './inputs'
import setOutputs from './outputs' import setOutputs from './outputs'
import install from './install' import installPnpm from './install-pnpm'
import pnpmInstall from './pnpm-install'
import pruneStore from './pnpm-store-prune'
async function main() { async function main() {
const inputs = getInputs() const inputs = getInputs()
await install(inputs).then(() => { const isPost = getState('is_post')
console.log('Installation Completed!') if (isPost === 'true') return pruneStore(inputs)
setOutputs(inputs) saveState('is_post', 'true')
}) await installPnpm(inputs)
console.log('Installation Completed!')
setOutputs(inputs)
pnpmInstall(inputs)
} }
main().catch(error => { main().catch(error => {

View File

@ -1,11 +1,11 @@
import { getInput, InputOptions } from '@actions/core' import { getInput, InputOptions } from '@actions/core'
import expandTilde from 'expand-tilde' import expandTilde from 'expand-tilde'
import { RunInstall, parseRunInstall } from './run-install'
export interface Inputs { export interface Inputs {
readonly version: string readonly version: string
readonly dest: string readonly dest: string
readonly binDest: string readonly runInstall: RunInstall[]
readonly registry: string
} }
const options: InputOptions = { const options: InputOptions = {
@ -17,8 +17,7 @@ const parseInputPath = (name: string) => expandTilde(getInput(name, options))
export const getInputs = (): Inputs => ({ export const getInputs = (): Inputs => ({
version: getInput('version', options), version: getInput('version', options),
dest: parseInputPath('dest'), dest: parseInputPath('dest'),
binDest: parseInputPath('bin_dest'), runInstall: parseRunInstall('run_install'),
registry: getInput('registry', options),
}) })
export default getInputs export default getInputs

View File

@ -0,0 +1,21 @@
{
"$schema": "https://raw.githubusercontent.com/ksxnodeapps/ts-schema-autogen/master/packages/schemas/config.schema.json",
"instruction": {
"compilerOptions": {
"strict": true,
"target": "ES2018",
"lib": [
"ES2018",
"ES2019",
"ES2020",
"ESNext"
],
"moduleResolution": "Node",
"esModuleInterop": true,
"resolveJsonModule": true
},
"input": "run-install.ts",
"symbol": "RunInstallInput",
"output": "run-install-input.schema.json"
}
}

View File

@ -0,0 +1,39 @@
{
"anyOf": [
{
"$ref": "#/definitions/RunInstall"
},
{
"type": "array",
"items": {
"$ref": "#/definitions/RunInstall"
}
},
{
"type": [
"null",
"boolean"
]
}
],
"definitions": {
"RunInstall": {
"type": "object",
"properties": {
"recursive": {
"type": "boolean"
},
"cwd": {
"type": "string"
},
"args": {
"type": "array",
"items": {
"type": "string"
}
}
}
}
},
"$schema": "http://json-schema.org/draft-07/schema#"
}

39
src/inputs/run-install.ts Normal file
View File

@ -0,0 +1,39 @@
import process from 'process'
import { load } from 'js-yaml'
import Ajv from 'ajv'
import { getInput, error, InputOptions } from '@actions/core'
import runInstallSchema from './run-install-input.schema.json'
export interface RunInstall {
readonly recursive?: boolean
readonly cwd?: string
readonly args?: readonly string[]
}
export type RunInstallInput =
| null
| boolean
| RunInstall
| RunInstall[]
const options: InputOptions = {
required: true,
}
export function parseRunInstall(name: string): RunInstall[] {
const result: RunInstallInput = load(getInput(name, options)) as any
const ajv = new Ajv({
allErrors: true,
})
const validate = ajv.compile(runInstallSchema)
if (!validate(result)) {
for (const errorItem of validate.errors!) {
error(`with.run_install${errorItem.dataPath}: ${errorItem.message}`)
}
return process.exit(1)
}
if (!result) return []
if (result === true) return [{ recursive: true }]
if (Array.isArray(result)) return result
return [result]
}

View File

@ -1,11 +1,13 @@
import { setFailed } from '@actions/core' import { setFailed, startGroup, endGroup } from '@actions/core'
import { Inputs } from '../inputs' import { Inputs } from '../inputs'
import runSelfInstaller from './run' import runSelfInstaller from './run'
export { runSelfInstaller } export { runSelfInstaller }
export async function install(inputs: Inputs) { export async function install(inputs: Inputs) {
startGroup('Running self-installer...')
const status = await runSelfInstaller(inputs) const status = await runSelfInstaller(inputs)
endGroup()
if (status) { if (status) {
return setFailed(`Something does wrong, self-installer exits with code ${status}`) return setFailed(`Something does wrong, self-installer exits with code ${status}`)
} }

31
src/install-pnpm/run.ts Normal file
View File

@ -0,0 +1,31 @@
import { spawn } from 'child_process'
import { execPath } from 'process'
import { join } from 'path'
import { remove, ensureFile, writeFile } from 'fs-extra'
import fetch from 'node-fetch'
import { Inputs } from '../inputs'
export async function runSelfInstaller(inputs: Inputs): Promise<number> {
const { version, dest } = inputs
const target = version ? `pnpm@${version}` : 'pnpm'
const pkgJson = join(dest, 'package.json')
await remove(dest)
await ensureFile(pkgJson)
await writeFile(pkgJson, JSON.stringify({ private: true }))
const cp = spawn(execPath, ['-', 'install', target, '--no-lockfile'], {
cwd: dest,
stdio: ['pipe', 'inherit', 'inherit'],
})
const response = await fetch('https://pnpm.js.org/pnpm.js')
response.body.pipe(cp.stdin)
return new Promise((resolve, reject) => {
cp.on('error', reject)
cp.on('close', resolve)
})
}
export default runSelfInstaller

View File

@ -1,25 +0,0 @@
import { spawn } from 'child_process'
import { execPath } from 'process'
import { downloadSelfInstaller } from '../self-installer'
import { Inputs } from '../inputs'
export function runSelfInstaller(inputs: Inputs): Promise<number> {
const cp = spawn(execPath, {
env: {
PNPM_VERSION: inputs.version,
PNPM_DEST: inputs.dest,
PNPM_BIN_DEST: inputs.binDest,
PNPM_REGISTRY: inputs.registry,
},
stdio: ['pipe', 'inherit', 'inherit'],
})
downloadSelfInstaller().pipe(cp.stdin)
return new Promise((resolve, reject) => {
cp.on('error', reject)
cp.on('close', resolve)
})
}
export default runSelfInstaller

View File

@ -1,10 +1,12 @@
import { setOutput, addPath } 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) { export function setOutputs(inputs: Inputs) {
addPath(inputs.binDest) const binDest = getBinDest(inputs)
addPath(binDest)
setOutput('dest', inputs.dest) setOutput('dest', inputs.dest)
setOutput('bin_dest', inputs.binDest) setOutput('bin_dest', binDest)
} }
export default setOutputs export default setOutputs

38
src/pnpm-install/index.ts Normal file
View File

@ -0,0 +1,38 @@
import { spawnSync } from 'child_process'
import { setFailed, startGroup, endGroup } from '@actions/core'
import { Inputs } from '../inputs'
import { patchPnpmEnv } from '../utils'
export function runPnpmInstall(inputs: Inputs) {
const env = patchPnpmEnv(inputs)
for (const options of inputs.runInstall) {
const args = ['install']
if (options.recursive) args.unshift('recursive')
if (options.args) args.push(...options.args)
const cmdStr = ['pnpm', ...args].join(' ')
startGroup(`Running ${cmdStr}...`)
const { error, status } = spawnSync('pnpm', args, {
stdio: 'inherit',
cwd: options.cwd,
shell: true,
env,
})
endGroup()
if (error) {
setFailed(error)
continue
}
if (status) {
setFailed(`Command ${cmdStr} (cwd: ${options.cwd}) exits with status ${status}`)
continue
}
}
}
export default runPnpmInstall

View File

@ -0,0 +1,31 @@
import { spawnSync } from 'child_process'
import { warning, startGroup, endGroup } from '@actions/core'
import { Inputs } from '../inputs'
import { patchPnpmEnv } from '../utils'
export function pruneStore(inputs: Inputs) {
if (inputs.runInstall.length === 0) {
console.log('Pruning is unnecessary.')
return
}
startGroup('Running pnpm store prune...')
const { error, status } = spawnSync('pnpm', ['store', 'prune'], {
stdio: 'inherit',
shell: true,
env: patchPnpmEnv(inputs)
})
endGroup()
if (error) {
warning(error)
return
}
if (status) {
warning(`command pnpm store prune exits with code ${status}`)
return
}
}
export default pruneStore

View File

@ -1,4 +0,0 @@
import download from 'download'
import url from './url'
export const downloadSelfInstaller = () => download(url)
export default downloadSelfInstaller

View File

@ -1,2 +0,0 @@
export * from './url'
export * from './download'

View File

@ -1,3 +0,0 @@
export const ref = '301414cec74a2b6b63c95b42f2ad1790ccb980ed'
export const url = `https://raw.githubusercontent.com/pnpm/self-installer/${ref}/install.js`
export default url

10
src/utils/index.ts Normal file
View File

@ -0,0 +1,10 @@
import process from 'process'
import path from 'path'
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: getBinDest(inputs) + path.delimiter + process.env.PATH
})