mirror of
https://github.com/AikidoSec/safe-chain.git
synced 2026-05-26 12:10:49 +00:00
232 lines
5.9 KiB
JavaScript
232 lines
5.9 KiB
JavaScript
import chalk from "chalk";
|
|
import { ui } from "../environment/userInteraction.js";
|
|
import { knownAikidoTools, getPackageManagerList } from "./helpers.js";
|
|
import fs from "fs";
|
|
import os from "os";
|
|
import path from "path";
|
|
import { fileURLToPath } from "url";
|
|
|
|
/**
|
|
* Loops over the detected shells and calls the setup function for each.
|
|
*/
|
|
export async function setupCi() {
|
|
ui.writeInformation(
|
|
chalk.bold("Setting up shell aliases.") +
|
|
` This will wrap safe-chain around ${getPackageManagerList()}.`
|
|
);
|
|
ui.emptyLine();
|
|
|
|
const shimsDir = path.join(os.homedir(), ".safe-chain", "shims");
|
|
// Create the shims directory if it doesn't exist
|
|
if (!fs.existsSync(shimsDir)) {
|
|
fs.mkdirSync(shimsDir, { recursive: true });
|
|
}
|
|
|
|
createShims(shimsDir);
|
|
ui.writeInformation(`Created shims in ${shimsDir}`);
|
|
modifyPathForCi(shimsDir);
|
|
ui.writeInformation(`Added shims directory to PATH for CI environments.`);
|
|
}
|
|
|
|
/**
|
|
* @param {string} shimsDir
|
|
*
|
|
* @returns {void}
|
|
*/
|
|
function createUnixShims(shimsDir) {
|
|
// Read the template file
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = path.dirname(__filename);
|
|
const templatePath = path.resolve(
|
|
__dirname,
|
|
"path-wrappers",
|
|
"templates",
|
|
"unix-wrapper.template.sh"
|
|
);
|
|
|
|
if (!fs.existsSync(templatePath)) {
|
|
ui.writeError(`Template file not found: ${templatePath}`);
|
|
return;
|
|
}
|
|
|
|
const template = fs.readFileSync(templatePath, "utf-8");
|
|
|
|
// Create a shim for each tool
|
|
let created = 0;
|
|
for (const toolInfo of knownAikidoTools) {
|
|
const shimContent = template
|
|
.replaceAll("{{PACKAGE_MANAGER}}", toolInfo.tool)
|
|
.replaceAll("{{AIKIDO_COMMAND}}", toolInfo.aikidoCommand);
|
|
|
|
const shimPath = path.join(shimsDir, toolInfo.tool);
|
|
fs.writeFileSync(shimPath, shimContent, "utf-8");
|
|
|
|
// Make the shim executable on Unix systems
|
|
fs.chmodSync(shimPath, 0o755);
|
|
created++;
|
|
}
|
|
|
|
// Also create python and python3 shims to support `python[3] -m pip[3]` in CI
|
|
createUnixPythonShims(shimsDir);
|
|
|
|
ui.writeInformation(
|
|
`Created ${created} Unix shim(s) in ${shimsDir}`
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @param {string} shimsDir
|
|
*/
|
|
function createUnixPythonShims(shimsDir) {
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = path.dirname(__filename);
|
|
|
|
const entries = [
|
|
{
|
|
name: "python",
|
|
template: path.resolve(
|
|
__dirname,
|
|
"path-wrappers",
|
|
"templates",
|
|
"unix-python-wrapper.template.sh"
|
|
),
|
|
},
|
|
{
|
|
name: "python3",
|
|
template: path.resolve(
|
|
__dirname,
|
|
"path-wrappers",
|
|
"templates",
|
|
"unix-python-wrapper.template.sh"
|
|
),
|
|
},
|
|
];
|
|
|
|
for (const entry of entries) {
|
|
if (!fs.existsSync(entry.template)) {
|
|
ui.writeError(`Template file not found: ${entry.template}`);
|
|
continue;
|
|
}
|
|
const shimContent = fs.readFileSync(entry.template, "utf-8");
|
|
const shimPath = `${shimsDir}/${entry.name}`;
|
|
fs.writeFileSync(shimPath, shimContent, "utf-8");
|
|
fs.chmodSync(shimPath, 0o755);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} shimsDir
|
|
*
|
|
* @returns {void}
|
|
*/
|
|
function createWindowsShims(shimsDir) {
|
|
// Read the template file
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = path.dirname(__filename);
|
|
const templatePath = path.resolve(
|
|
__dirname,
|
|
"path-wrappers",
|
|
"templates",
|
|
"windows-wrapper.template.cmd"
|
|
);
|
|
|
|
if (!fs.existsSync(templatePath)) {
|
|
ui.writeError(`Windows template file not found: ${templatePath}`);
|
|
return;
|
|
}
|
|
|
|
const template = fs.readFileSync(templatePath, "utf-8");
|
|
|
|
// Create a shim for each tool
|
|
let created = 0;
|
|
for (const toolInfo of knownAikidoTools) {
|
|
const shimContent = template
|
|
.replaceAll("{{PACKAGE_MANAGER}}", toolInfo.tool)
|
|
.replaceAll("{{AIKIDO_COMMAND}}", toolInfo.aikidoCommand);
|
|
|
|
const shimPath = `${shimsDir}/${toolInfo.tool}.cmd`;
|
|
fs.writeFileSync(shimPath, shimContent, "utf-8");
|
|
created++;
|
|
}
|
|
|
|
// Also create python and python3 shims for Windows to support `python[3] -m pip[3]` in CI
|
|
createWindowsPythonShims(shimsDir);
|
|
|
|
ui.writeInformation(
|
|
`Created ${created} Windows shim(s) in ${shimsDir}`
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @param {string} shimsDir
|
|
*/
|
|
function createWindowsPythonShims(shimsDir) {
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = path.dirname(__filename);
|
|
|
|
const entries = [
|
|
{
|
|
name: "python.cmd",
|
|
template: path.resolve(
|
|
__dirname,
|
|
"path-wrappers",
|
|
"templates",
|
|
"windows-python-wrapper.template.cmd"
|
|
),
|
|
},
|
|
{
|
|
name: "python3.cmd",
|
|
template: path.resolve(
|
|
__dirname,
|
|
"path-wrappers",
|
|
"templates",
|
|
"windows-python-wrapper.template.cmd"
|
|
),
|
|
},
|
|
];
|
|
|
|
for (const entry of entries) {
|
|
if (!fs.existsSync(entry.template)) {
|
|
ui.writeError(`Windows template file not found: ${entry.template}`);
|
|
continue;
|
|
}
|
|
const shimContent = fs.readFileSync(entry.template, "utf-8");
|
|
const shimPath = `${shimsDir}/${entry.name}`;
|
|
fs.writeFileSync(shimPath, shimContent, "utf-8");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} shimsDir
|
|
*
|
|
* @returns {void}
|
|
*/
|
|
function createShims(shimsDir) {
|
|
if (os.platform() === "win32") {
|
|
createWindowsShims(shimsDir);
|
|
} else {
|
|
createUnixShims(shimsDir);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} shimsDir
|
|
*
|
|
* @returns {void}
|
|
*/
|
|
function modifyPathForCi(shimsDir) {
|
|
if (process.env.GITHUB_PATH) {
|
|
// In GitHub Actions, append the shims directory to GITHUB_PATH
|
|
fs.appendFileSync(process.env.GITHUB_PATH, shimsDir + os.EOL, "utf-8");
|
|
ui.writeInformation(
|
|
`Added shims directory to GITHUB_PATH for GitHub Actions.`
|
|
);
|
|
}
|
|
|
|
if (process.env.TF_BUILD) {
|
|
// In Azure Pipelines, prepending the path is done via a logging command:
|
|
// ##vso[task.prependpath]/path/to/add
|
|
// Logging this to stdout will cause the Azure Pipelines agent to pick it up
|
|
ui.writeInformation("##vso[task.prependpath]" + shimsDir);
|
|
}
|
|
}
|