mirror of
https://github.com/czlonkowski/n8n-mcp.git
synced 2026-02-06 21:43:07 +00:00
feat: add template metadata generation and smart discovery
- Implement OpenAI batch API integration for metadata generation - Add search_templates_by_metadata tool with advanced filtering - Enhance list_templates to include descriptions and optional metadata - Generate metadata for 2,534 templates (97.5% coverage) - Update README with Template Tools section and enhanced Claude setup - Add comprehensive documentation for metadata system Enables intelligent template discovery through: - Complexity levels (simple/medium/complex) - Setup time estimates (5-480 minutes) - Target audience filtering (developers/marketers/analysts) - Required services detection - Category and use case classification Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
@@ -3,13 +3,34 @@ import { createDatabaseAdapter } from '../database/database-adapter';
|
||||
import { TemplateService } from '../templates/template-service';
|
||||
import * as fs from 'fs';
|
||||
import * as path from 'path';
|
||||
import * as zlib from 'zlib';
|
||||
import * as dotenv from 'dotenv';
|
||||
import type { MetadataRequest } from '../templates/metadata-generator';
|
||||
|
||||
// Load environment variables
|
||||
dotenv.config();
|
||||
|
||||
async function fetchTemplates(mode: 'rebuild' | 'update' = 'rebuild', generateMetadata: boolean = false) {
|
||||
async function fetchTemplates(mode: 'rebuild' | 'update' = 'rebuild', generateMetadata: boolean = false, metadataOnly: boolean = false) {
|
||||
// If metadata-only mode, skip template fetching entirely
|
||||
if (metadataOnly) {
|
||||
console.log('🤖 Metadata-only mode: Generating metadata for existing templates...\n');
|
||||
|
||||
if (!process.env.OPENAI_API_KEY) {
|
||||
console.error('❌ OPENAI_API_KEY not set in environment');
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
const db = await createDatabaseAdapter('./data/nodes.db');
|
||||
const service = new TemplateService(db);
|
||||
|
||||
await generateTemplateMetadata(db, service);
|
||||
|
||||
if ('close' in db && typeof db.close === 'function') {
|
||||
db.close();
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
const modeEmoji = mode === 'rebuild' ? '🔄' : '⬆️';
|
||||
const modeText = mode === 'rebuild' ? 'Rebuilding' : 'Updating';
|
||||
console.log(`${modeEmoji} ${modeText} n8n workflow templates...\n`);
|
||||
@@ -27,66 +48,48 @@ async function fetchTemplates(mode: 'rebuild' | 'update' = 'rebuild', generateMe
|
||||
// Initialize database
|
||||
const db = await createDatabaseAdapter('./data/nodes.db');
|
||||
|
||||
// Only drop tables in rebuild mode
|
||||
// Handle database schema based on mode
|
||||
if (mode === 'rebuild') {
|
||||
try {
|
||||
// Drop existing tables in rebuild mode
|
||||
db.exec('DROP TABLE IF EXISTS templates');
|
||||
db.exec('DROP TABLE IF EXISTS templates_fts');
|
||||
console.log('🗑️ Dropped existing templates tables (rebuild mode)\n');
|
||||
|
||||
// Apply fresh schema
|
||||
const schema = fs.readFileSync(path.join(__dirname, '../../src/database/schema.sql'), 'utf8');
|
||||
db.exec(schema);
|
||||
console.log('📋 Applied database schema\n');
|
||||
} catch (error) {
|
||||
// Ignore errors if tables don't exist
|
||||
console.error('❌ Error setting up database schema:', error);
|
||||
throw error;
|
||||
}
|
||||
} else {
|
||||
console.log('📊 Update mode: Keeping existing templates\n');
|
||||
}
|
||||
|
||||
// Apply schema with updated constraint
|
||||
const schema = fs.readFileSync(path.join(__dirname, '../../src/database/schema.sql'), 'utf8');
|
||||
db.exec(schema);
|
||||
|
||||
// Pre-create FTS5 tables if supported
|
||||
const hasFTS5 = db.checkFTS5Support();
|
||||
if (hasFTS5) {
|
||||
console.log('🔍 Creating FTS5 tables for template search...');
|
||||
console.log('📊 Update mode: Keeping existing templates and schema\n');
|
||||
|
||||
// In update mode, only ensure new columns exist (for migration)
|
||||
try {
|
||||
// Create FTS5 virtual table
|
||||
db.exec(`
|
||||
CREATE VIRTUAL TABLE IF NOT EXISTS templates_fts USING fts5(
|
||||
name, description, content=templates
|
||||
);
|
||||
`);
|
||||
// Check if metadata columns exist, add them if not (migration support)
|
||||
const columns = db.prepare("PRAGMA table_info(templates)").all() as any[];
|
||||
const hasMetadataColumn = columns.some((col: any) => col.name === 'metadata_json');
|
||||
|
||||
// Create triggers to keep FTS5 in sync
|
||||
db.exec(`
|
||||
CREATE TRIGGER IF NOT EXISTS templates_ai AFTER INSERT ON templates BEGIN
|
||||
INSERT INTO templates_fts(rowid, name, description)
|
||||
VALUES (new.id, new.name, new.description);
|
||||
END;
|
||||
`);
|
||||
|
||||
db.exec(`
|
||||
CREATE TRIGGER IF NOT EXISTS templates_au AFTER UPDATE ON templates BEGIN
|
||||
UPDATE templates_fts SET name = new.name, description = new.description
|
||||
WHERE rowid = new.id;
|
||||
END;
|
||||
`);
|
||||
|
||||
db.exec(`
|
||||
CREATE TRIGGER IF NOT EXISTS templates_ad AFTER DELETE ON templates BEGIN
|
||||
DELETE FROM templates_fts WHERE rowid = old.id;
|
||||
END;
|
||||
`);
|
||||
|
||||
console.log('✅ FTS5 tables created successfully\n');
|
||||
if (!hasMetadataColumn) {
|
||||
console.log('📋 Adding metadata columns to existing schema...');
|
||||
db.exec(`
|
||||
ALTER TABLE templates ADD COLUMN metadata_json TEXT;
|
||||
ALTER TABLE templates ADD COLUMN metadata_generated_at DATETIME;
|
||||
`);
|
||||
console.log('✅ Metadata columns added\n');
|
||||
}
|
||||
} catch (error) {
|
||||
console.log('⚠️ Failed to create FTS5 tables:', error);
|
||||
console.log(' Template search will use LIKE fallback\n');
|
||||
// Columns might already exist, that's fine
|
||||
console.log('📋 Schema is up to date\n');
|
||||
}
|
||||
} else {
|
||||
console.log('ℹ️ FTS5 not supported in this SQLite build');
|
||||
console.log(' Template search will use LIKE queries\n');
|
||||
}
|
||||
|
||||
// FTS5 initialization is handled by TemplateRepository
|
||||
// No need to duplicate the logic here
|
||||
|
||||
// Create service
|
||||
const service = new TemplateService(db);
|
||||
|
||||
@@ -104,7 +107,7 @@ async function fetchTemplates(mode: 'rebuild' | 'update' = 'rebuild', generateMe
|
||||
const progress = total > 0 ? Math.round((current / total) * 100) : 0;
|
||||
lastMessage = `📊 ${message}: ${current}/${total} (${progress}%)`;
|
||||
process.stdout.write(lastMessage);
|
||||
}, mode);
|
||||
}, mode); // Pass the mode parameter!
|
||||
|
||||
console.log('\n'); // New line after progress
|
||||
|
||||
@@ -148,8 +151,11 @@ async function generateTemplateMetadata(db: any, service: TemplateService) {
|
||||
const { BatchProcessor } = await import('../templates/batch-processor');
|
||||
const repository = (service as any).repository;
|
||||
|
||||
// Get templates without metadata
|
||||
const templatesWithoutMetadata = repository.getTemplatesWithoutMetadata(500);
|
||||
// Get templates without metadata (0 = no limit)
|
||||
const limit = parseInt(process.env.METADATA_LIMIT || '0');
|
||||
const templatesWithoutMetadata = limit > 0
|
||||
? repository.getTemplatesWithoutMetadata(limit)
|
||||
: repository.getTemplatesWithoutMetadata(999999); // Get all
|
||||
|
||||
if (templatesWithoutMetadata.length === 0) {
|
||||
console.log('✅ All templates already have metadata');
|
||||
@@ -159,23 +165,44 @@ async function generateTemplateMetadata(db: any, service: TemplateService) {
|
||||
console.log(`Found ${templatesWithoutMetadata.length} templates without metadata`);
|
||||
|
||||
// Create batch processor
|
||||
const batchSize = parseInt(process.env.OPENAI_BATCH_SIZE || '50');
|
||||
console.log(`Processing in batches of ${batchSize} templates each`);
|
||||
|
||||
// Warn if batch size is very large
|
||||
if (batchSize > 100) {
|
||||
console.log(`⚠️ Large batch size (${batchSize}) may take longer to process`);
|
||||
console.log(` Consider using OPENAI_BATCH_SIZE=50 for faster results`);
|
||||
}
|
||||
|
||||
const processor = new BatchProcessor({
|
||||
apiKey: process.env.OPENAI_API_KEY!,
|
||||
model: process.env.OPENAI_MODEL || 'gpt-4o-mini',
|
||||
batchSize: parseInt(process.env.OPENAI_BATCH_SIZE || '100'),
|
||||
batchSize: batchSize,
|
||||
outputDir: './temp/batch'
|
||||
});
|
||||
|
||||
// Prepare metadata requests
|
||||
const requests: MetadataRequest[] = templatesWithoutMetadata.map((t: any) => ({
|
||||
templateId: t.id,
|
||||
name: t.name,
|
||||
description: t.description,
|
||||
nodes: JSON.parse(t.nodes_used),
|
||||
workflow: t.workflow_json_compressed
|
||||
? JSON.parse(Buffer.from(t.workflow_json_compressed, 'base64').toString())
|
||||
: (t.workflow_json ? JSON.parse(t.workflow_json) : undefined)
|
||||
}));
|
||||
const requests: MetadataRequest[] = templatesWithoutMetadata.map((t: any) => {
|
||||
let workflow = undefined;
|
||||
try {
|
||||
if (t.workflow_json_compressed) {
|
||||
const decompressed = zlib.gunzipSync(Buffer.from(t.workflow_json_compressed, 'base64'));
|
||||
workflow = JSON.parse(decompressed.toString());
|
||||
} else if (t.workflow_json) {
|
||||
workflow = JSON.parse(t.workflow_json);
|
||||
}
|
||||
} catch (error) {
|
||||
console.warn(`Failed to parse workflow for template ${t.id}:`, error);
|
||||
}
|
||||
|
||||
return {
|
||||
templateId: t.id,
|
||||
name: t.name,
|
||||
description: t.description,
|
||||
nodes: JSON.parse(t.nodes_used),
|
||||
workflow
|
||||
};
|
||||
});
|
||||
|
||||
// Process in batches
|
||||
const results = await processor.processTemplates(requests, (message, current, total) => {
|
||||
@@ -210,11 +237,12 @@ async function generateTemplateMetadata(db: any, service: TemplateService) {
|
||||
}
|
||||
|
||||
// Parse command line arguments
|
||||
function parseArgs(): { mode: 'rebuild' | 'update', generateMetadata: boolean } {
|
||||
function parseArgs(): { mode: 'rebuild' | 'update', generateMetadata: boolean, metadataOnly: boolean } {
|
||||
const args = process.argv.slice(2);
|
||||
|
||||
let mode: 'rebuild' | 'update' = 'rebuild';
|
||||
let generateMetadata = false;
|
||||
let metadataOnly = false;
|
||||
|
||||
// Check for --mode flag
|
||||
const modeIndex = args.findIndex(arg => arg.startsWith('--mode'));
|
||||
@@ -237,25 +265,31 @@ function parseArgs(): { mode: 'rebuild' | 'update', generateMetadata: boolean }
|
||||
generateMetadata = true;
|
||||
}
|
||||
|
||||
// Check for --metadata-only flag
|
||||
if (args.includes('--metadata-only')) {
|
||||
metadataOnly = true;
|
||||
}
|
||||
|
||||
// Show help if requested
|
||||
if (args.includes('--help') || args.includes('-h')) {
|
||||
console.log('Usage: npm run fetch:templates [options]\n');
|
||||
console.log('Options:');
|
||||
console.log(' --mode=rebuild|update Rebuild from scratch or update existing (default: rebuild)');
|
||||
console.log(' --update Shorthand for --mode=update');
|
||||
console.log(' --generate-metadata Generate AI metadata for templates (requires OPENAI_API_KEY)');
|
||||
console.log(' --generate-metadata Generate AI metadata after fetching templates');
|
||||
console.log(' --metadata Shorthand for --generate-metadata');
|
||||
console.log(' --metadata-only Only generate metadata, skip template fetching');
|
||||
console.log(' --help, -h Show this help message');
|
||||
process.exit(0);
|
||||
}
|
||||
|
||||
return { mode, generateMetadata };
|
||||
return { mode, generateMetadata, metadataOnly };
|
||||
}
|
||||
|
||||
// Run if called directly
|
||||
if (require.main === module) {
|
||||
const { mode, generateMetadata } = parseArgs();
|
||||
fetchTemplates(mode, generateMetadata).catch(console.error);
|
||||
const { mode, generateMetadata, metadataOnly } = parseArgs();
|
||||
fetchTemplates(mode, generateMetadata, metadataOnly).catch(console.error);
|
||||
}
|
||||
|
||||
export { fetchTemplates };
|
||||
Reference in New Issue
Block a user