t-convex-nextjs-saas/bin/init-template.mjs

102 lines
3.1 KiB
JavaScript
Raw Permalink Normal View History

#!/usr/bin/env node
import fs from 'fs';
import path from 'path';
import { execSync } from 'child_process';
import { fileURLToPath } from 'url';
const EXCLUDE = [
'.git',
'node_modules',
'.next',
'.sisyphus',
'.memsearch',
'.playwright-mcp',
'pnpm-lock.yaml',
'bin',
'features.md',
'.env.local',
'README.md',
'tmuxi.template.yml',
];
function copyRecursive(src, dest, replacements) {
const stat = fs.statSync(src);
if (stat.isDirectory()) {
if (EXCLUDE.includes(path.basename(src))) return;
fs.mkdirSync(dest, { recursive: true });
for (const entry of fs.readdirSync(src)) {
copyRecursive(path.join(src, entry), path.join(dest, entry), replacements);
}
} else {
if (EXCLUDE.includes(path.basename(src))) return;
let content = fs.readFileSync(src, 'utf-8');
for (const [key, value] of Object.entries(replacements)) {
content = content.split(key).join(value);
}
fs.writeFileSync(dest, content);
}
}
function main() {
const projectName = process.argv[2];
if (!projectName) {
console.error('Usage: node bin/init-template.mjs <project-name>');
process.exit(1);
}
const targetDir = path.resolve(process.cwd(), projectName);
if (fs.existsSync(targetDir)) {
console.error(`Directory already exists: ${targetDir}`);
process.exit(1);
}
const __dirname = path.dirname(fileURLToPath(import.meta.url));
const templateDir = path.resolve(__dirname, '..');
const replacements = {
'convex-next-saas': projectName,
'https://convex-backend.mentat.ovh': 'https://your-convex-backend.example.com',
'https://backend-site-olnjg91x5ervt6j6owwgnlha.mentat.ovh': 'https://your-convex-site.example.com',
'self-hosted-convex|01eea0ecf04bed0f70b73021564229f7f08eecff003f7294e5f9279faa4c19ffd5c501b0ac': 'self-hosted-convex|YOUR_ADMIN_KEY',
'http://localhost:3000': 'http://localhost:3000',
'SaaS Template': projectName,
'Create SaaS in a day!': `${projectName} — built with convex-next-saas`,
'https://convex-backend.mentat.ovh:3210': 'https://your-convex-backend.example.com:3210',
'https://backend-site-xxx.mentat.ovh:3211': 'https://your-convex-site.example.com:3211',
'/home/nxtkofi/.config/tmuxinator/': '~/.config/tmuxinator/',
'~/vaults/mentat/': '~/workspace/',
};
// Safety: exclude target dir name to prevent recursion if run inside template
EXCLUDE.push(projectName);
const srcDir = path.join(targetDir, 'src');
const docsDir = path.join(targetDir, 'docs');
console.log(`Creating ${projectName}...`);
// Copy template files into <name>/src/
copyRecursive(templateDir, srcDir, replacements);
// Create empty docs/readme.md for the new project
fs.mkdirSync(docsDir, { recursive: true });
fs.writeFileSync(
path.join(docsDir, 'readme.md'),
`# ${projectName}\n\nProject documentation goes here.\n`
);
console.log('Initializing git...');
execSync('git init', { cwd: targetDir, stdio: 'inherit' });
console.log('Installing dependencies...');
execSync('pnpm install', { cwd: srcDir, stdio: 'inherit' });
console.log(`\nDone!`);
console.log(` cd ${projectName}/src && pnpm dev --webpack`);
}
main();