Files
antigravity-skills-reference/tools/bin/install.js
2026-03-15 08:39:22 +01:00

308 lines
8.4 KiB
JavaScript
Executable File

#!/usr/bin/env node
const { spawnSync } = require("child_process");
const path = require("path");
const fs = require("fs");
const os = require("os");
const { resolveSafeRealPath } = require("../lib/symlink-safety");
const REPO = "https://github.com/sickn33/antigravity-awesome-skills.git";
const HOME = process.env.HOME || process.env.USERPROFILE || "";
function resolveDir(p) {
if (!p) return null;
const s = p.replace(/^~($|\/)/, HOME + "$1");
return path.resolve(s);
}
function parseArgs() {
const a = process.argv.slice(2);
let pathArg = null;
let versionArg = null;
let tagArg = null;
let cursor = false,
claude = false,
gemini = false,
codex = false,
antigravity = false,
kiro = false;
for (let i = 0; i < a.length; i++) {
if (a[i] === "--help" || a[i] === "-h") return { help: true };
if (a[i] === "--path" && a[i + 1]) {
pathArg = a[++i];
continue;
}
if (a[i] === "--version" && a[i + 1]) {
versionArg = a[++i];
continue;
}
if (a[i] === "--tag" && a[i + 1]) {
tagArg = a[++i];
continue;
}
if (a[i] === "--cursor") {
cursor = true;
continue;
}
if (a[i] === "--claude") {
claude = true;
continue;
}
if (a[i] === "--gemini") {
gemini = true;
continue;
}
if (a[i] === "--codex") {
codex = true;
continue;
}
if (a[i] === "--antigravity") {
antigravity = true;
continue;
}
if (a[i] === "--kiro") {
kiro = true;
continue;
}
if (a[i] === "install") continue;
}
return {
pathArg,
versionArg,
tagArg,
cursor,
claude,
gemini,
codex,
antigravity,
kiro,
};
}
function getTargets(opts) {
const targets = [];
if (opts.pathArg) {
return [{ name: "Custom", path: resolveDir(opts.pathArg) }];
}
if (opts.cursor) {
targets.push({ name: "Cursor", path: path.join(HOME, ".cursor", "skills") });
}
if (opts.claude) {
targets.push({ name: "Claude Code", path: path.join(HOME, ".claude", "skills") });
}
if (opts.gemini) {
targets.push({ name: "Gemini CLI", path: path.join(HOME, ".gemini", "skills") });
}
if (opts.codex) {
const codexHome = process.env.CODEX_HOME;
const codexPath = codexHome
? path.join(codexHome, "skills")
: path.join(HOME, ".codex", "skills");
targets.push({ name: "Codex CLI", path: codexPath });
}
if (opts.kiro) {
targets.push({ name: "Kiro", path: path.join(HOME, ".kiro", "skills") });
}
if (opts.antigravity) {
targets.push({ name: "Antigravity", path: path.join(HOME, ".gemini", "antigravity", "skills") });
}
if (targets.length === 0) {
targets.push({ name: "Antigravity", path: path.join(HOME, ".gemini", "antigravity", "skills") });
}
return targets;
}
function printHelp() {
console.log(`
antigravity-awesome-skills — installer
npx antigravity-awesome-skills [install] [options]
Clones the skills repo into your agent's skills directory.
Options:
--cursor Install to ~/.cursor/skills (Cursor)
--claude Install to ~/.claude/skills (Claude Code)
--gemini Install to ~/.gemini/skills (Gemini CLI)
--codex Install to ~/.codex/skills (Codex CLI)
--kiro Install to ~/.kiro/skills (Kiro CLI)
--antigravity Install to ~/.gemini/antigravity/skills (Antigravity)
--path <dir> Install to <dir> (default: ~/.gemini/antigravity/skills)
--version <ver> After clone, checkout tag v<ver> (e.g. 4.6.0 -> v4.6.0)
--tag <tag> After clone, checkout this tag (e.g. v4.6.0)
Examples:
npx antigravity-awesome-skills
npx antigravity-awesome-skills --cursor
npx antigravity-awesome-skills --kiro
npx antigravity-awesome-skills --antigravity
npx antigravity-awesome-skills --version 4.6.0
npx antigravity-awesome-skills --path ./my-skills
npx antigravity-awesome-skills --claude --codex Install to multiple targets
`);
}
function copyRecursiveSync(src, dest, rootDir = src, skipGit = true) {
const stats = fs.lstatSync(src);
const resolvedSource = stats.isSymbolicLink()
? resolveSafeRealPath(rootDir, src)
: src;
if (!resolvedSource) {
console.warn(` Skipping symlink outside cloned skills root: ${src}`);
return;
}
const resolvedStats = fs.statSync(resolvedSource);
if (resolvedStats.isDirectory()) {
if (!fs.existsSync(dest)) {
fs.mkdirSync(dest, { recursive: true });
}
fs.readdirSync(resolvedSource).forEach((child) => {
if (skipGit && child === ".git") return;
copyRecursiveSync(path.join(resolvedSource, child), path.join(dest, child), rootDir, skipGit);
});
} else {
fs.copyFileSync(resolvedSource, dest);
}
}
/** Copy contents of repo's skills/ into target so each skill is target/skill-name/ (for Claude Code etc.). */
function installSkillsIntoTarget(tempDir, target) {
const repoSkills = path.join(tempDir, "skills");
if (!fs.existsSync(repoSkills)) {
console.error("Cloned repo has no skills/ directory.");
process.exit(1);
}
fs.readdirSync(repoSkills).forEach((name) => {
const src = path.join(repoSkills, name);
const dest = path.join(target, name);
copyRecursiveSync(src, dest, repoSkills);
});
const repoDocs = path.join(tempDir, "docs");
if (fs.existsSync(repoDocs)) {
const docsDest = path.join(target, "docs");
if (!fs.existsSync(docsDest)) fs.mkdirSync(docsDest, { recursive: true });
copyRecursiveSync(repoDocs, docsDest, repoDocs);
}
}
function run(cmd, args, opts = {}) {
const r = spawnSync(cmd, args, { stdio: "inherit", ...opts });
if (r.status !== 0) process.exit(r.status == null ? 1 : r.status);
}
function installForTarget(tempDir, target) {
if (fs.existsSync(target.path)) {
const gitDir = path.join(target.path, ".git");
if (fs.existsSync(gitDir)) {
console.log(` Migrating from full-repo install to skills-only layout…`);
const entries = fs.readdirSync(target.path);
for (const name of entries) {
const full = path.join(target.path, name);
const stat = fs.lstatSync(full);
if (stat.isDirectory() && !stat.isSymbolicLink()) {
if (fs.rmSync) {
fs.rmSync(full, { recursive: true, force: true });
} else {
fs.rmdirSync(full, { recursive: true });
}
} else {
fs.unlinkSync(full);
}
}
} else {
console.log(` Updating existing install at ${target.path}`);
}
} else {
const parent = path.dirname(target.path);
if (!fs.existsSync(parent)) {
try {
fs.mkdirSync(parent, { recursive: true });
} catch (e) {
console.error(` Cannot create parent directory: ${parent}`, e.message);
process.exit(1);
}
}
fs.mkdirSync(target.path, { recursive: true });
}
installSkillsIntoTarget(tempDir, target.path);
console.log(` ✓ Installed to ${target.path}`);
}
function main() {
const opts = parseArgs();
const { tagArg, versionArg } = opts;
if (opts.help) {
printHelp();
return;
}
const targets = getTargets(opts);
if (!targets.length || !HOME) {
console.error(
"Could not resolve home directory. Use --path <absolute-path>.",
);
process.exit(1);
}
const tempDir = fs.mkdtempSync(path.join(os.tmpdir(), "ag-skills-"));
const originalCwd = process.cwd();
try {
console.log("Cloning repository…");
run("git", ["clone", REPO, tempDir]);
const ref =
tagArg ||
(versionArg
? versionArg.startsWith("v")
? versionArg
: `v${versionArg}`
: null);
if (ref) {
console.log(`Checking out ${ref}`);
process.chdir(tempDir);
run("git", ["checkout", ref]);
process.chdir(originalCwd);
}
console.log(`\nInstalling for ${targets.length} target(s):`);
for (const target of targets) {
console.log(`\n${target.name}:`);
installForTarget(tempDir, target);
}
console.log(
"\nPick a bundle in docs/users/bundles.md and use @skill-name in your AI assistant.",
);
} finally {
try {
if (fs.existsSync(tempDir)) {
if (fs.rmSync) {
fs.rmSync(tempDir, { recursive: true, force: true });
} else {
fs.rmdirSync(tempDir, { recursive: true });
}
}
} catch (e) {
// ignore cleanup errors
}
}
}
if (require.main === module) {
main();
}
module.exports = {
copyRecursiveSync,
installSkillsIntoTarget,
installForTarget,
main,
};