Spaces:
Sleeping
Sleeping
File size: 6,679 Bytes
da2e594 | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 | /**
* Integration tests for empty database scenarios
* Ensures we detect and handle empty database situations that caused production failures
*/
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
import { createDatabaseAdapter } from '../../../src/database/database-adapter';
import { NodeRepository } from '../../../src/database/node-repository';
import * as fs from 'fs';
import * as path from 'path';
import * as os from 'os';
describe('Empty Database Detection Tests', () => {
let tempDbPath: string;
let db: any;
let repository: NodeRepository;
beforeEach(async () => {
// Create a temporary database file
tempDbPath = path.join(os.tmpdir(), `test-empty-${Date.now()}.db`);
db = await createDatabaseAdapter(tempDbPath);
// Initialize schema
const schemaPath = path.join(__dirname, '../../../src/database/schema.sql');
const schema = fs.readFileSync(schemaPath, 'utf-8');
db.exec(schema);
repository = new NodeRepository(db);
});
afterEach(() => {
if (db) {
db.close();
}
// Clean up temp file
if (fs.existsSync(tempDbPath)) {
fs.unlinkSync(tempDbPath);
}
});
describe('Empty Nodes Table Detection', () => {
it('should detect empty nodes table', () => {
const count = db.prepare('SELECT COUNT(*) as count FROM nodes').get();
expect(count.count).toBe(0);
});
it('should detect empty FTS5 index', () => {
const count = db.prepare('SELECT COUNT(*) as count FROM nodes_fts').get();
expect(count.count).toBe(0);
});
it('should return empty results for critical node searches', () => {
const criticalSearches = ['webhook', 'merge', 'split', 'code', 'http'];
for (const search of criticalSearches) {
const results = db.prepare(`
SELECT node_type FROM nodes_fts
WHERE nodes_fts MATCH ?
`).all(search);
expect(results).toHaveLength(0);
}
});
it('should fail validation with empty database', () => {
const validation = validateEmptyDatabase(repository);
expect(validation.passed).toBe(false);
expect(validation.issues.length).toBeGreaterThan(0);
expect(validation.issues[0]).toMatch(/CRITICAL.*no nodes found/i);
});
});
describe('LIKE Fallback with Empty Database', () => {
it('should return empty results for LIKE searches', () => {
const results = db.prepare(`
SELECT node_type FROM nodes
WHERE node_type LIKE ? OR display_name LIKE ? OR description LIKE ?
`).all('%webhook%', '%webhook%', '%webhook%');
expect(results).toHaveLength(0);
});
it('should return empty results for multi-word LIKE searches', () => {
const results = db.prepare(`
SELECT node_type FROM nodes
WHERE (node_type LIKE ? OR display_name LIKE ? OR description LIKE ?)
OR (node_type LIKE ? OR display_name LIKE ? OR description LIKE ?)
`).all('%split%', '%split%', '%split%', '%batch%', '%batch%', '%batch%');
expect(results).toHaveLength(0);
});
});
describe('Repository Methods with Empty Database', () => {
it('should return null for getNode() with empty database', () => {
const node = repository.getNode('nodes-base.webhook');
expect(node).toBeNull();
});
it('should return empty array for searchNodes() with empty database', () => {
const results = repository.searchNodes('webhook');
expect(results).toHaveLength(0);
});
it('should return empty array for getAITools() with empty database', () => {
const tools = repository.getAITools();
expect(tools).toHaveLength(0);
});
it('should return 0 for getNodeCount() with empty database', () => {
const count = repository.getNodeCount();
expect(count).toBe(0);
});
});
describe('Validation Messages for Empty Database', () => {
it('should provide clear error message for empty database', () => {
const validation = validateEmptyDatabase(repository);
const criticalError = validation.issues.find(issue =>
issue.includes('CRITICAL') && issue.includes('empty')
);
expect(criticalError).toBeDefined();
expect(criticalError).toContain('no nodes found');
});
it('should suggest rebuild command in error message', () => {
const validation = validateEmptyDatabase(repository);
const errorWithSuggestion = validation.issues.find(issue =>
issue.toLowerCase().includes('rebuild')
);
// This expectation documents that we should add rebuild suggestions
// Currently validation doesn't include this, but it should
if (!errorWithSuggestion) {
console.warn('TODO: Add rebuild suggestion to validation error messages');
}
});
});
describe('Empty Template Data', () => {
it('should detect empty templates table', () => {
const count = db.prepare('SELECT COUNT(*) as count FROM templates').get();
expect(count.count).toBe(0);
});
it('should handle missing template data gracefully', () => {
const templates = db.prepare('SELECT * FROM templates LIMIT 10').all();
expect(templates).toHaveLength(0);
});
});
});
/**
* Validation function matching rebuild.ts logic
*/
function validateEmptyDatabase(repository: NodeRepository): { passed: boolean; issues: string[] } {
const issues: string[] = [];
try {
const db = (repository as any).db;
// Check if database has any nodes
const nodeCount = db.prepare('SELECT COUNT(*) as count FROM nodes').get() as { count: number };
if (nodeCount.count === 0) {
issues.push('CRITICAL: Database is empty - no nodes found! Rebuild failed or was interrupted.');
return { passed: false, issues };
}
// Check minimum expected node count
if (nodeCount.count < 500) {
issues.push(`WARNING: Only ${nodeCount.count} nodes found - expected at least 500 (both n8n packages)`);
}
// Check FTS5 table
const ftsTableCheck = db.prepare(`
SELECT name FROM sqlite_master
WHERE type='table' AND name='nodes_fts'
`).get();
if (!ftsTableCheck) {
issues.push('CRITICAL: FTS5 table (nodes_fts) does not exist - searches will fail or be very slow');
} else {
const ftsCount = db.prepare('SELECT COUNT(*) as count FROM nodes_fts').get() as { count: number };
if (ftsCount.count === 0) {
issues.push('CRITICAL: FTS5 index is empty - searches will return zero results');
}
}
} catch (error) {
issues.push(`Validation error: ${(error as Error).message}`);
}
return {
passed: issues.length === 0,
issues
};
}
|