-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathbfg.ts
122 lines (103 loc) · 3.5 KB
/
bfg.ts
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
/**
* BFG Database Cleaner
*
* This script completely wipes a Neo4j database for testing purposes.
* It removes ALL data, constraints, indexes, and migration metadata.
*
* Usage: npx ts-node bfg.ts [--force] [--uri=neo4j://localhost:7687] [--user=neo4j] [--password=password]
*/
import neo4j, {Driver, Session} from 'neo4j-driver'
const args = process.argv.slice(2)
const params = {
database: 'neo4j',
force: args.includes('--force'),
password: 'password',
uri: 'neo4j://localhost:7687',
user: 'neo4j',
}
for (const arg of args) {
if (arg.startsWith('--uri=')) {
params.uri = arg.slice(6)
} else if (arg.startsWith('--user=')) {
params.user = arg.slice(7)
} else if (arg.startsWith('--password=')) {
params.password = arg.slice(11)
} else if (arg.startsWith('--database=')) {
params.database = arg.slice(11)
}
}
function log(message: string, type: 'error' | 'info' | 'warn' = 'info'): void {
const timestamp = new Date().toISOString()
const prefix = type === 'error' ? '❌' : type === 'warn' ? '⚠️' : 'ℹ️'
console[type](`${timestamp} ${prefix} ${message}`)
}
async function run(): Promise<void> {
if (!params.force) {
console.warn('⚠️ WARNING: This will completely erase your database!')
console.warn('⚠️ Run with --force to skip this warning.')
process.exit(1)
}
log(`Connecting to Neo4j at ${params.uri}`)
const driver: Driver = neo4j.driver(params.uri, neo4j.auth.basic(params.user, params.password))
let session: Session | null = null
try {
session = driver.session({database: params.database})
log('Testing database connection...')
await session.run('RETURN 1')
log('Connection successful')
log('Dropping all constraints and indexes...')
const dropAllConstraints = async () => {
const constraints = await session!.run('SHOW CONSTRAINTS')
for (const record of constraints.records) {
const name = record.get('name')
if (name) {
try {
await session!.run(`DROP CONSTRAINT ${name}`)
log(`Dropped constraint: ${name}`)
} catch (error) {
log(`Failed to drop constraint ${name}: ${error}`, 'warn')
}
}
}
}
const dropAllIndexes = async () => {
const indexes = await session!.run('SHOW INDEXES')
for (const record of indexes.records) {
const name = record.get('name')
if (name) {
try {
await session!.run(`DROP INDEX ${name}`)
log(`Dropped index: ${name}`)
} catch (error) {
log(`Failed to drop index ${name}: ${error}`, 'warn')
}
}
}
}
await dropAllConstraints()
await dropAllIndexes()
log('All constraints and indexes removed')
log('Deleting all data...')
await session.run('MATCH (n) DETACH DELETE n')
log('All nodes and relationships removed')
const countResult = await session.run('MATCH (n) RETURN count(n) as count')
const nodeCount = countResult.records[0].get('count').toNumber()
if (nodeCount === 0) {
log('Database successfully wiped clean!')
} else {
log(`Database still contains ${nodeCount} nodes. Something went wrong.`, 'error')
}
} catch (error) {
log(`Error: ${error instanceof Error ? error.message : String(error)}`, 'error')
process.exit(1)
} finally {
if (session) {
await session.close()
}
await driver.close()
}
}
run().catch((error) => {
log(`Unhandled error: ${error.message}`, 'error')
process.exit(1)
})