repoweaver
Version:
A GitHub App that skillfully weaves multiple templates together to create and update repositories with intelligent merge strategies
180 lines (156 loc) • 6.49 kB
text/typescript
import { Command } from 'commander';
import * as path from 'path';
import { Bootstrapper } from './bootstrapper';
import { ConfigLoader } from './config-loader';
import { BootstrapOptions, TemplateRepository, WeaverConfig } from './types';
const program = new Command();
program.name('repoweaver').description('Skillfully weave multiple templates together to create and update repositories').version('1.0.0');
// Add init command to create sample configuration files
program
.command('init')
.description('Initialize a RepoWeaver project with sample configuration files')
.option('--config-only', 'Only create configuration file')
.option('--ignore-only', 'Only create ignore file')
.action(async (options) => {
try {
const configLoader = new ConfigLoader();
if (options.ignoreOnly) {
await configLoader.createSampleIgnore();
} else if (options.configOnly) {
await configLoader.createSampleConfig();
} else {
await configLoader.createSampleConfig();
await configLoader.createSampleIgnore();
}
console.log('✅ RepoWeaver project initialized!');
console.log('📝 Edit weaver.json to configure your templates');
console.log('🚫 Edit .weaverignore to exclude files from processing');
} catch (error) {
console.error('❌ Initialization failed:', error);
process.exit(1);
}
});
program
.command('bootstrap')
.description('Create a new repository from template(s)')
.argument('<name>', 'Repository name')
.argument('<path>', 'Target path for the new repository')
.option('-t, --template <url>', 'Template repository URL(s)', collect, [])
.option('-b, --branch <branch>', 'Template branch (default: main)')
.option('-s, --subdir <path>', 'Use subdirectory from template')
.option('--git', 'Initialize git repository', false)
.option('--remote <url>', 'Add git remote origin')
.option('--exclude <pattern>', 'Exclude patterns', collect, [])
.option('--merge-strategy <strategy>', 'Merge strategy: overwrite|merge|skip', 'merge')
.action(async (name, targetPath, options) => {
try {
// Load configuration from files if they exist
const configLoader = new ConfigLoader();
const config = await configLoader.loadConfiguration();
// Command line options override configuration file
const templates: TemplateRepository[] =
options.template.length > 0
? options.template.map((url: string) => ({
url,
name: extractRepoName(url),
branch: options.branch,
}))
: config.templates;
if (templates.length === 0) {
console.error('Error: At least one template is required');
console.error('Either provide --template options or create a weaver.json configuration file');
process.exit(1);
}
// Merge configuration with command line options
const bootstrapOptions: BootstrapOptions = {
targetPath,
templates,
repositoryName: name || config.name || path.basename(targetPath),
initGit: options.git !== undefined ? options.git : config.initGit,
addRemote: options.remote || config.addRemote,
excludePatterns: options.exclude.length > 0 ? options.exclude : config.excludePatterns,
mergeStrategy: options.mergeStrategy || config.mergeStrategy,
};
const bootstrapper = new Bootstrapper();
const result = await bootstrapper.bootstrap(bootstrapOptions);
if (result.success) {
console.log(`✅ Successfully bootstrapped repository at: ${result.repositoryPath}`);
console.log(`📁 Total files processed: ${result.totalFilesProcessed}`);
result.templateResults.forEach((tr) => {
console.log(` - ${tr.template.name}: ${tr.filesProcessed} files`);
});
} else {
console.error('❌ Bootstrap failed:');
result.errors.forEach((error) => console.error(` - ${error}`));
process.exit(1);
}
} catch (error) {
console.error(`❌ Bootstrap failed: ${error}`);
process.exit(1);
}
});
program
.command('update')
.description('Update an existing repository with template(s)')
.argument('<path>', 'Path to the existing repository')
.option('-t, --template <url>', 'Template repository URL(s)', collect, [])
.option('-b, --branch <branch>', 'Template branch (default: main)')
.option('-s, --subdir <path>', 'Use subdirectory from template')
.option('--exclude <pattern>', 'Exclude patterns', collect, [])
.option('--merge-strategy <strategy>', 'Merge strategy: overwrite|merge|skip', 'merge')
.action(async (targetPath, options) => {
try {
// Load configuration from files if they exist
const configLoader = new ConfigLoader(targetPath);
const config = await configLoader.loadConfiguration();
// Command line options override configuration file
const templates: TemplateRepository[] =
options.template.length > 0
? options.template.map((url: string) => ({
url,
name: extractRepoName(url),
branch: options.branch,
}))
: config.templates;
if (templates.length === 0) {
console.error('Error: At least one template is required');
console.error('Either provide --template options or create a weaver.json configuration file');
process.exit(1);
}
// Merge configuration with command line options
const bootstrapOptions: BootstrapOptions = {
targetPath,
templates,
repositoryName: config.name || path.basename(targetPath),
excludePatterns: options.exclude.length > 0 ? options.exclude : config.excludePatterns,
mergeStrategy: options.mergeStrategy || config.mergeStrategy,
};
const bootstrapper = new Bootstrapper();
const result = await bootstrapper.updateRepository(bootstrapOptions);
if (result.success) {
console.log(`✅ Successfully updated repository at: ${result.repositoryPath}`);
console.log(`📁 Total files processed: ${result.totalFilesProcessed}`);
result.templateResults.forEach((tr) => {
console.log(` - ${tr.template.name}: ${tr.filesProcessed} files`);
});
} else {
console.error('❌ Update failed:');
result.errors.forEach((error) => console.error(` - ${error}`));
process.exit(1);
}
} catch (error) {
console.error(`❌ Update failed: ${error}`);
process.exit(1);
}
});
function collect(value: string, previous: string[]): string[] {
return previous.concat([value]);
}
function extractRepoName(url: string): string {
const match = url.match(/\/([^\/]+?)(?:\.git)?$/);
return match ? match[1] : 'unknown';
}
if (require.main === module) {
program.parse();
}