mirror of
https://github.com/pnpm/action-setup.git
synced 2026-03-01 07:51:02 +08:00
fix: check if pnpm already installed (#176)
This commit is contained in:
parent
1e1c8eafbd
commit
ab72e3e12e
2
dist/index.js
vendored
2
dist/index.js
vendored
File diff suppressed because one or more lines are too long
@ -1,16 +1,27 @@
|
||||
import { setFailed, startGroup, endGroup } from '@actions/core'
|
||||
import { Inputs } from '../inputs'
|
||||
import runSelfInstaller from './run'
|
||||
import { setFailed, startGroup, endGroup } from "@actions/core";
|
||||
import { Inputs } from "../inputs";
|
||||
import runSelfInstaller from "./run";
|
||||
import { exec } from "child_process";
|
||||
import { promisify } from "util";
|
||||
|
||||
export { runSelfInstaller }
|
||||
const execAsync = promisify(exec);
|
||||
|
||||
export async function install(inputs: Inputs) {
|
||||
startGroup('Running self-installer...')
|
||||
const status = await runSelfInstaller(inputs)
|
||||
endGroup()
|
||||
try {
|
||||
await execAsync("pnpm --version");
|
||||
// pnpm already installed
|
||||
return;
|
||||
} catch {
|
||||
// pnpm not installed, continue
|
||||
}
|
||||
|
||||
startGroup("Running self-installer...");
|
||||
const status = await runSelfInstaller(inputs);
|
||||
endGroup();
|
||||
|
||||
if (status) {
|
||||
return setFailed(`Something went wrong, self-installer exits with code ${status}`)
|
||||
setFailed(`Something went wrong, self-installer exits with code ${status}`);
|
||||
}
|
||||
}
|
||||
|
||||
export default install
|
||||
export default install;
|
||||
|
||||
@ -1,115 +1,134 @@
|
||||
import { addPath, exportVariable } from '@actions/core'
|
||||
import { spawn } from 'child_process'
|
||||
import { rm, writeFile, mkdir, copyFile } from 'fs/promises'
|
||||
import { readFileSync } from 'fs'
|
||||
import path from 'path'
|
||||
import { execPath } from 'process'
|
||||
import util from 'util'
|
||||
import { Inputs } from '../inputs'
|
||||
import { parse as parseYaml } from 'yaml'
|
||||
import { addPath, exportVariable } from "@actions/core";
|
||||
import { spawn } from "child_process";
|
||||
import { rm, writeFile, mkdir, copyFile } from "fs/promises";
|
||||
import { readFileSync } from "fs";
|
||||
import path from "path";
|
||||
import { execPath } from "process";
|
||||
import util from "util";
|
||||
import { Inputs } from "../inputs";
|
||||
import { parse as parseYaml } from "yaml";
|
||||
|
||||
export async function runSelfInstaller(inputs: Inputs): Promise<number> {
|
||||
const { version, dest, packageJsonFile, standalone } = inputs
|
||||
const { GITHUB_WORKSPACE } = process.env
|
||||
const { version, dest, packageJsonFile, standalone } = inputs;
|
||||
const { GITHUB_WORKSPACE } = process.env;
|
||||
|
||||
// prepare self install
|
||||
await rm(dest, { recursive: true, force: true })
|
||||
await rm(dest, { recursive: true, force: true });
|
||||
// create dest directory after removal
|
||||
await mkdir(dest, { recursive: true })
|
||||
const pkgJson = path.join(dest, 'package.json')
|
||||
await mkdir(dest, { recursive: true });
|
||||
const pkgJson = path.join(dest, "package.json");
|
||||
// we have ensured the dest directory exists, we can write the file directly
|
||||
await writeFile(pkgJson, JSON.stringify({ private: true }))
|
||||
await writeFile(pkgJson, JSON.stringify({ private: true }));
|
||||
|
||||
// copy .npmrc if it exists to install from custom registry
|
||||
if (GITHUB_WORKSPACE) {
|
||||
try {
|
||||
await copyFile(path.join(GITHUB_WORKSPACE, '.npmrc'), path.join(dest, '.npmrc'))
|
||||
await copyFile(
|
||||
path.join(GITHUB_WORKSPACE, ".npmrc"),
|
||||
path.join(dest, ".npmrc"),
|
||||
);
|
||||
} catch (error) {
|
||||
// Swallow error if .npmrc doesn't exist
|
||||
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;
|
||||
}
|
||||
}
|
||||
|
||||
// prepare target pnpm
|
||||
const target = await readTarget({ version, packageJsonFile, standalone })
|
||||
const cp = spawn(execPath, [path.join(__dirname, 'pnpm.cjs'), 'install', target, '--no-lockfile'], {
|
||||
cwd: dest,
|
||||
stdio: ['pipe', 'inherit', 'inherit'],
|
||||
})
|
||||
const target = await readTarget({ version, packageJsonFile, standalone });
|
||||
const cp = spawn(
|
||||
execPath,
|
||||
[path.join(__dirname, "pnpm.cjs"), "install", target, "--no-lockfile"],
|
||||
{
|
||||
cwd: dest,
|
||||
stdio: ["pipe", "inherit", "inherit"],
|
||||
},
|
||||
);
|
||||
|
||||
const exitCode = await new Promise<number>((resolve, reject) => {
|
||||
cp.on('error', reject)
|
||||
cp.on('close', resolve)
|
||||
})
|
||||
cp.on("error", reject);
|
||||
cp.on("close", resolve);
|
||||
});
|
||||
if (exitCode === 0) {
|
||||
const pnpmHome = path.join(dest, 'node_modules/.bin')
|
||||
addPath(pnpmHome)
|
||||
exportVariable('PNPM_HOME', pnpmHome)
|
||||
const pnpmHome = path.join(dest, "node_modules/.bin");
|
||||
addPath(pnpmHome);
|
||||
exportVariable("PNPM_HOME", pnpmHome);
|
||||
}
|
||||
return exitCode
|
||||
return exitCode;
|
||||
}
|
||||
|
||||
async function readTarget(opts: {
|
||||
readonly version?: string | undefined
|
||||
readonly packageJsonFile: string
|
||||
readonly standalone: boolean
|
||||
readonly version?: string | undefined;
|
||||
readonly packageJsonFile: string;
|
||||
readonly standalone: boolean;
|
||||
}) {
|
||||
const { version, packageJsonFile, standalone } = opts
|
||||
const { GITHUB_WORKSPACE } = process.env
|
||||
const { version, packageJsonFile, standalone } = opts;
|
||||
const { GITHUB_WORKSPACE } = process.env;
|
||||
|
||||
let packageManager
|
||||
let packageManager;
|
||||
|
||||
if (GITHUB_WORKSPACE) {
|
||||
try {
|
||||
const content = readFileSync(path.join(GITHUB_WORKSPACE, packageJsonFile), 'utf8');
|
||||
const content = readFileSync(
|
||||
path.join(GITHUB_WORKSPACE, packageJsonFile),
|
||||
"utf8",
|
||||
);
|
||||
({ packageManager } = packageJsonFile.endsWith(".yaml")
|
||||
? parseYaml(content, { merge: true })
|
||||
: JSON.parse(content)
|
||||
)
|
||||
: JSON.parse(content));
|
||||
} catch (error: unknown) {
|
||||
// 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;
|
||||
}
|
||||
}
|
||||
|
||||
if (version) {
|
||||
if (
|
||||
typeof packageManager === 'string' &&
|
||||
packageManager.startsWith('pnpm@') &&
|
||||
packageManager.replace('pnpm@', '') !== version
|
||||
typeof packageManager === "string" &&
|
||||
packageManager.startsWith("pnpm@") &&
|
||||
packageManager.replace("pnpm@", "") !== version
|
||||
) {
|
||||
throw new Error(`Multiple versions of pnpm specified:
|
||||
- version ${version} in the GitHub Action config with the key "version"
|
||||
- version ${packageManager} in the package.json with the key "packageManager"
|
||||
Remove one of these versions to avoid version mismatch errors like ERR_PNPM_BAD_PM_VERSION`)
|
||||
Remove one of these versions to avoid version mismatch errors like ERR_PNPM_BAD_PM_VERSION`);
|
||||
}
|
||||
|
||||
return `${ standalone ? '@pnpm/exe' : 'pnpm' }@${version}`
|
||||
return `${standalone ? "@pnpm/exe" : "pnpm"}@${version}`;
|
||||
}
|
||||
|
||||
if (!GITHUB_WORKSPACE) {
|
||||
throw new Error(`No workspace is found.
|
||||
If you've intended to let pnpm/action-setup read preferred pnpm version from the "packageManager" field in the package.json file,
|
||||
please run the actions/checkout before pnpm/action-setup.
|
||||
Otherwise, please specify the pnpm version in the action configuration.`)
|
||||
Otherwise, please specify the pnpm version in the action configuration.`);
|
||||
}
|
||||
|
||||
if (typeof packageManager !== 'string') {
|
||||
if (typeof packageManager !== "string") {
|
||||
throw new Error(`No pnpm version is specified.
|
||||
Please specify it by one of the following ways:
|
||||
- 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"`);
|
||||
}
|
||||
|
||||
if (!packageManager.startsWith('pnpm@')) {
|
||||
throw new Error('Invalid packageManager field in package.json')
|
||||
if (!packageManager.startsWith("pnpm@")) {
|
||||
throw new Error("Invalid packageManager field in package.json");
|
||||
}
|
||||
|
||||
if (standalone) {
|
||||
return packageManager.replace('pnpm@', '@pnpm/exe@')
|
||||
return packageManager.replace("pnpm@", "@pnpm/exe@");
|
||||
}
|
||||
|
||||
return packageManager
|
||||
return packageManager;
|
||||
}
|
||||
|
||||
export default runSelfInstaller
|
||||
export default runSelfInstaller;
|
||||
|
||||
Loading…
Reference in New Issue
Block a user