mirror of
https://github.com/mark3labs/kit.git
synced 2026-06-14 03:30:26 +00:00
ci: add lint job, npm publish pipeline, and update goreleaser config
- CI: add golangci-lint job, target master branch, use Go 1.26 - Release: add workflow_dispatch for npm-only releases, Discord notifications, and npm-publish job for @mark3labs/kit - Goreleaser: upgrade to v2 schema, lowercase archive names, structured changelog groups, release header with install instructions - npm: add package scaffolding with postinstall binary downloader
This commit is contained in:
@@ -0,0 +1,3 @@
|
||||
# Downloaded binaries (only wrapper script should be committed)
|
||||
bin/kit-bin
|
||||
bin/kit.exe
|
||||
Executable
+100
@@ -0,0 +1,100 @@
|
||||
#!/usr/bin/env node
|
||||
|
||||
const { spawnSync } = require("child_process");
|
||||
const fs = require("fs");
|
||||
const path = require("path");
|
||||
const https = require("https");
|
||||
|
||||
const REPO = "mark3labs/kit";
|
||||
const BINARY = "kit";
|
||||
|
||||
const binDir = __dirname;
|
||||
const isWindows = process.platform === "win32";
|
||||
const binaryName = isWindows ? "kit.exe" : "kit-bin";
|
||||
const binaryPath = path.join(binDir, binaryName);
|
||||
|
||||
// Platform/arch mapping
|
||||
const PLATFORM_MAP = { darwin: "darwin", linux: "linux", win32: "windows" };
|
||||
const ARCH_MAP = { x64: "amd64", arm64: "arm64" };
|
||||
|
||||
function download(url, dest) {
|
||||
return new Promise((resolve, reject) => {
|
||||
const follow = (url, redirects = 0) => {
|
||||
if (redirects > 10) return reject(new Error("Too many redirects"));
|
||||
https.get(url, (res) => {
|
||||
if (res.statusCode >= 300 && res.statusCode < 400 && res.headers.location) {
|
||||
return follow(res.headers.location, redirects + 1);
|
||||
}
|
||||
if (res.statusCode !== 200) return reject(new Error(`HTTP ${res.statusCode}`));
|
||||
const file = fs.createWriteStream(dest);
|
||||
res.pipe(file);
|
||||
file.on("finish", () => { file.close(); resolve(); });
|
||||
file.on("error", (err) => { fs.unlinkSync(dest); reject(err); });
|
||||
}).on("error", reject);
|
||||
};
|
||||
follow(url);
|
||||
});
|
||||
}
|
||||
|
||||
async function install() {
|
||||
const platform = PLATFORM_MAP[process.platform];
|
||||
const arch = ARCH_MAP[process.arch];
|
||||
if (!platform || !arch) {
|
||||
console.error(`Unsupported platform: ${process.platform}/${process.arch}`);
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
const packageJson = require("../package.json");
|
||||
const version = packageJson.version;
|
||||
const ext = platform === "windows" ? "zip" : "tar.gz";
|
||||
const filename = `${BINARY}_${version}_${platform}_${arch}.${ext}`;
|
||||
const url = `https://github.com/${REPO}/releases/download/v${version}/${filename}`;
|
||||
const archivePath = path.join(binDir, filename);
|
||||
// Extract to temp dir to avoid overwriting this JS wrapper script
|
||||
const tempDir = path.join(binDir, ".tmp-extract");
|
||||
|
||||
console.log(`Installing ${BINARY} v${version} (${platform}/${arch})...`);
|
||||
|
||||
await download(url, archivePath);
|
||||
|
||||
// Create temp extraction directory
|
||||
fs.mkdirSync(tempDir, { recursive: true });
|
||||
|
||||
if (platform === "windows") {
|
||||
spawnSync("powershell", ["-Command", `Expand-Archive -Path "${archivePath}" -DestinationPath "${tempDir}" -Force`]);
|
||||
} else {
|
||||
spawnSync("tar", ["-xzf", archivePath, "-C", tempDir]);
|
||||
}
|
||||
|
||||
// Move extracted binary to final location (from temp dir to avoid overwriting wrapper)
|
||||
const extractedPath = path.join(tempDir, platform === "windows" ? `${BINARY}.exe` : BINARY);
|
||||
if (fs.existsSync(extractedPath)) {
|
||||
fs.renameSync(extractedPath, binaryPath);
|
||||
}
|
||||
|
||||
if (platform !== "windows") fs.chmodSync(binaryPath, 0o755);
|
||||
fs.unlinkSync(archivePath);
|
||||
fs.rmSync(tempDir, { recursive: true, force: true });
|
||||
console.log(`Installed ${BINARY} successfully`);
|
||||
}
|
||||
|
||||
async function main() {
|
||||
// Install binary if missing
|
||||
if (!fs.existsSync(binaryPath)) {
|
||||
await install();
|
||||
}
|
||||
|
||||
const result = spawnSync(binaryPath, process.argv.slice(2), {
|
||||
stdio: "inherit",
|
||||
shell: false,
|
||||
});
|
||||
|
||||
if (result.error) {
|
||||
console.error(`Failed to run kit: ${result.error.message}`);
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
process.exit(result.status ?? 1);
|
||||
}
|
||||
|
||||
main();
|
||||
+153
@@ -0,0 +1,153 @@
|
||||
#!/usr/bin/env node
|
||||
|
||||
const { execSync, spawnSync } = require("child_process");
|
||||
const fs = require("fs");
|
||||
const path = require("path");
|
||||
const https = require("https");
|
||||
const { createWriteStream, mkdirSync, chmodSync, unlinkSync } = fs;
|
||||
|
||||
const REPO = "mark3labs/kit";
|
||||
const BINARY = "kit";
|
||||
|
||||
// Get version from package.json
|
||||
const packageJson = require("./package.json");
|
||||
const VERSION = packageJson.version;
|
||||
|
||||
// Platform mapping
|
||||
const PLATFORM_MAP = {
|
||||
darwin: "darwin",
|
||||
linux: "linux",
|
||||
win32: "windows",
|
||||
};
|
||||
|
||||
// Arch mapping
|
||||
const ARCH_MAP = {
|
||||
x64: "amd64",
|
||||
arm64: "arm64",
|
||||
};
|
||||
|
||||
function getPlatform() {
|
||||
const platform = PLATFORM_MAP[process.platform];
|
||||
if (!platform) {
|
||||
throw new Error(`Unsupported platform: ${process.platform}`);
|
||||
}
|
||||
return platform;
|
||||
}
|
||||
|
||||
function getArch() {
|
||||
const arch = ARCH_MAP[process.arch];
|
||||
if (!arch) {
|
||||
throw new Error(`Unsupported architecture: ${process.arch}`);
|
||||
}
|
||||
return arch;
|
||||
}
|
||||
|
||||
function getExtension(platform) {
|
||||
return platform === "windows" ? "zip" : "tar.gz";
|
||||
}
|
||||
|
||||
function getBinaryName(platform) {
|
||||
// Use different name to avoid conflict with JS wrapper on Unix
|
||||
return platform === "windows" ? `${BINARY}.exe` : `${BINARY}-bin`;
|
||||
}
|
||||
|
||||
function download(url, dest) {
|
||||
return new Promise((resolve, reject) => {
|
||||
const follow = (url, redirects = 0) => {
|
||||
if (redirects > 10) {
|
||||
reject(new Error("Too many redirects"));
|
||||
return;
|
||||
}
|
||||
|
||||
https
|
||||
.get(url, (response) => {
|
||||
if (response.statusCode >= 300 && response.statusCode < 400 && response.headers.location) {
|
||||
follow(response.headers.location, redirects + 1);
|
||||
return;
|
||||
}
|
||||
|
||||
if (response.statusCode !== 200) {
|
||||
reject(new Error(`Failed to download: ${response.statusCode}`));
|
||||
return;
|
||||
}
|
||||
|
||||
const file = createWriteStream(dest);
|
||||
response.pipe(file);
|
||||
file.on("finish", () => {
|
||||
file.close();
|
||||
resolve();
|
||||
});
|
||||
file.on("error", (err) => {
|
||||
unlinkSync(dest);
|
||||
reject(err);
|
||||
});
|
||||
})
|
||||
.on("error", reject);
|
||||
};
|
||||
|
||||
follow(url);
|
||||
});
|
||||
}
|
||||
|
||||
function extract(archivePath, destDir, platform) {
|
||||
if (platform === "windows") {
|
||||
// Use PowerShell to extract zip on Windows
|
||||
spawnSync("powershell", [
|
||||
"-Command",
|
||||
`Expand-Archive -Path "${archivePath}" -DestinationPath "${destDir}" -Force`,
|
||||
]);
|
||||
} else {
|
||||
// Use tar for Unix systems
|
||||
spawnSync("tar", ["-xzf", archivePath, "-C", destDir]);
|
||||
}
|
||||
}
|
||||
|
||||
async function main() {
|
||||
const platform = getPlatform();
|
||||
const arch = getArch();
|
||||
const ext = getExtension(platform);
|
||||
const binaryName = getBinaryName(platform);
|
||||
|
||||
const filename = `${BINARY}_${VERSION}_${platform}_${arch}.${ext}`;
|
||||
const url = `https://github.com/${REPO}/releases/download/v${VERSION}/${filename}`;
|
||||
|
||||
const binDir = path.join(__dirname, "bin");
|
||||
const archivePath = path.join(__dirname, filename);
|
||||
const binaryPath = path.join(binDir, binaryName);
|
||||
|
||||
console.log(`Installing ${BINARY} v${VERSION} (${platform}/${arch})...`);
|
||||
console.log(`Downloading from ${url}...`);
|
||||
|
||||
try {
|
||||
// Ensure bin directory exists
|
||||
mkdirSync(binDir, { recursive: true });
|
||||
|
||||
// Download archive
|
||||
await download(url, archivePath);
|
||||
|
||||
// Extract
|
||||
extract(archivePath, binDir, platform);
|
||||
|
||||
// Rename binary (archive contains "kit", we want "kit-bin" on Unix)
|
||||
const extractedName = platform === "windows" ? `${BINARY}.exe` : BINARY;
|
||||
const extractedPath = path.join(binDir, extractedName);
|
||||
if (extractedPath !== binaryPath && fs.existsSync(extractedPath)) {
|
||||
fs.renameSync(extractedPath, binaryPath);
|
||||
}
|
||||
|
||||
// Make executable on Unix
|
||||
if (platform !== "windows") {
|
||||
chmodSync(binaryPath, 0o755);
|
||||
}
|
||||
|
||||
// Clean up archive
|
||||
unlinkSync(archivePath);
|
||||
|
||||
console.log(`Successfully installed ${BINARY} to ${binaryPath}`);
|
||||
} catch (err) {
|
||||
console.error(`Failed to install ${BINARY}: ${err.message}`);
|
||||
process.exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
main();
|
||||
@@ -0,0 +1,41 @@
|
||||
{
|
||||
"name": "@mark3labs/kit",
|
||||
"version": "0.0.0",
|
||||
"description": "AI-powered CLI tool",
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git+https://github.com/mark3labs/kit.git"
|
||||
},
|
||||
"homepage": "https://github.com/mark3labs/kit",
|
||||
"bugs": {
|
||||
"url": "https://github.com/mark3labs/kit/issues"
|
||||
},
|
||||
"license": "MIT",
|
||||
"bin": {
|
||||
"kit": "bin/kit"
|
||||
},
|
||||
"scripts": {
|
||||
"postinstall": "node install.js"
|
||||
},
|
||||
"files": [
|
||||
"bin",
|
||||
"install.js"
|
||||
],
|
||||
"engines": {
|
||||
"node": ">=16"
|
||||
},
|
||||
"os": [
|
||||
"darwin",
|
||||
"linux",
|
||||
"win32"
|
||||
],
|
||||
"cpu": [
|
||||
"x64",
|
||||
"arm64"
|
||||
],
|
||||
"keywords": [
|
||||
"cli",
|
||||
"kit",
|
||||
"ai"
|
||||
]
|
||||
}
|
||||
Reference in New Issue
Block a user