-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathdb.js
More file actions
184 lines (162 loc) · 6.34 KB
/
db.js
File metadata and controls
184 lines (162 loc) · 6.34 KB
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
import Database from "better-sqlite3";
import path from "path";
import { fileURLToPath } from "url";
const __dirname = path.dirname(fileURLToPath(import.meta.url));
const dataDir = process.env.DATA_DIR || __dirname;
const dbPath = path.join(dataDir, "data.sqlite");
const db = new Database(dbPath);
// WAL mode for concurrent reads + performance tuning
db.pragma("journal_mode = WAL");
db.pragma("busy_timeout = 5000");
db.pragma("synchronous = NORMAL"); // Faster writes, still safe with WAL
db.pragma("cache_size = -64000"); // 64MB cache
db.pragma("temp_store = MEMORY"); // Temp tables in RAM
db.pragma("mmap_size = 268435456"); // 256MB memory-mapped I/O
// Create tables
db.exec(`
CREATE TABLE IF NOT EXISTS sessions (
id TEXT PRIMARY KEY,
created_at TEXT NOT NULL,
last_seen TEXT NOT NULL
);
CREATE TABLE IF NOT EXISTS pastes (
id TEXT PRIMARY KEY,
session_id TEXT NOT NULL,
title TEXT NOT NULL,
markdown TEXT NOT NULL,
created_at TEXT NOT NULL,
updated_at TEXT NOT NULL,
shared_at TEXT,
FOREIGN KEY (session_id) REFERENCES sessions(id)
);
CREATE TABLE IF NOT EXISTS images (
id TEXT PRIMARY KEY,
paste_id TEXT NOT NULL,
filename TEXT NOT NULL,
mime_type TEXT NOT NULL,
size_bytes INTEGER NOT NULL,
created_at TEXT NOT NULL,
FOREIGN KEY (paste_id) REFERENCES pastes(id)
);
CREATE TABLE IF NOT EXISTS marketing_events (
id TEXT PRIMARY KEY,
session_id TEXT,
created_at TEXT NOT NULL,
event_type TEXT NOT NULL,
surface TEXT NOT NULL,
path TEXT NOT NULL,
target TEXT,
referrer_host TEXT,
referrer_url TEXT,
utm_source TEXT,
utm_medium TEXT,
utm_campaign TEXT,
FOREIGN KEY (session_id) REFERENCES sessions(id)
);
`);
// Migration: Add 'shared' column if it doesn't exist
const checkShared = db.prepare("SELECT COUNT(*) as count FROM pragma_table_info('pastes') WHERE name='shared'");
const hasShared = checkShared.get().count > 0;
if (!hasShared) {
db.exec("ALTER TABLE pastes ADD COLUMN shared INTEGER NOT NULL DEFAULT 0");
console.log("✅ Migration: Added 'shared' column to pastes table");
}
const checkSharedAt = db.prepare("SELECT COUNT(*) as count FROM pragma_table_info('pastes') WHERE name='shared_at'");
const hasSharedAt = checkSharedAt.get().count > 0;
if (!hasSharedAt) {
db.exec("ALTER TABLE pastes ADD COLUMN shared_at TEXT");
db.exec("UPDATE pastes SET shared_at = created_at WHERE shared = 1 AND shared_at IS NULL");
console.log("✅ Migration: Added 'shared_at' column to pastes table");
}
// Migration: Create images table if it doesn't exist
const checkImages = db.prepare("SELECT COUNT(*) as count FROM sqlite_master WHERE type='table' AND name='images'");
const hasImages = checkImages.get().count > 0;
if (!hasImages) {
db.exec(`
CREATE TABLE images (
id TEXT PRIMARY KEY,
paste_id TEXT NOT NULL,
filename TEXT NOT NULL,
mime_type TEXT NOT NULL,
size_bytes INTEGER NOT NULL,
created_at TEXT NOT NULL,
FOREIGN KEY (paste_id) REFERENCES pastes(id)
);
CREATE INDEX idx_images_paste_id ON images(paste_id);
`);
console.log("✅ Migration: Created images table");
}
// Migration: Create collab tables
const checkCollabSettings = db.prepare("SELECT COUNT(*) as count FROM sqlite_master WHERE type='table' AND name='collab_settings'");
const hasCollabSettings = checkCollabSettings.get().count > 0;
if (!hasCollabSettings) {
db.exec(`
CREATE TABLE collab_settings (
paste_id TEXT PRIMARY KEY,
password_hash TEXT,
can_read INTEGER NOT NULL DEFAULT 1,
can_write INTEGER NOT NULL DEFAULT 1,
created_at TEXT NOT NULL,
updated_at TEXT NOT NULL,
FOREIGN KEY (paste_id) REFERENCES pastes(id)
);
CREATE TABLE collab_members (
id TEXT PRIMARY KEY,
paste_id TEXT NOT NULL,
session_id TEXT,
fantasy_name TEXT NOT NULL,
avatar_color TEXT NOT NULL,
last_seen TEXT NOT NULL,
created_at TEXT NOT NULL,
FOREIGN KEY (paste_id) REFERENCES pastes(id),
FOREIGN KEY (session_id) REFERENCES sessions(id)
);
CREATE TABLE collab_snapshots (
id TEXT PRIMARY KEY,
paste_id TEXT NOT NULL,
markdown TEXT NOT NULL,
created_at TEXT NOT NULL,
created_by_member_id TEXT,
FOREIGN KEY (paste_id) REFERENCES pastes(id),
FOREIGN KEY (created_by_member_id) REFERENCES collab_members(id)
);
CREATE TABLE collab_chat_threads (
id TEXT PRIMARY KEY,
paste_id TEXT NOT NULL,
title TEXT NOT NULL,
created_at TEXT NOT NULL,
created_by_member_id TEXT,
FOREIGN KEY (paste_id) REFERENCES pastes(id),
FOREIGN KEY (created_by_member_id) REFERENCES collab_members(id)
);
CREATE TABLE collab_chat_messages (
id TEXT PRIMARY KEY,
thread_id TEXT NOT NULL,
member_id TEXT NOT NULL,
message TEXT NOT NULL,
created_at TEXT NOT NULL,
FOREIGN KEY (thread_id) REFERENCES collab_chat_threads(id),
FOREIGN KEY (member_id) REFERENCES collab_members(id)
);
CREATE INDEX idx_collab_members_paste ON collab_members(paste_id);
CREATE INDEX idx_collab_members_session ON collab_members(session_id);
CREATE INDEX idx_collab_snapshots_paste ON collab_snapshots(paste_id, created_at DESC);
CREATE INDEX idx_collab_chat_threads_paste ON collab_chat_threads(paste_id, created_at DESC);
CREATE INDEX idx_collab_chat_messages_thread ON collab_chat_messages(thread_id, created_at ASC);
`);
console.log("✅ Migration: Created collab tables");
}
// Create indexes (now safe because 'shared' column exists)
db.exec(`
CREATE INDEX IF NOT EXISTS idx_pastes_session_id ON pastes(session_id);
CREATE INDEX IF NOT EXISTS idx_pastes_session_updated ON pastes(session_id, updated_at DESC);
CREATE INDEX IF NOT EXISTS idx_pastes_session_created ON pastes(session_id, created_at DESC);
CREATE INDEX IF NOT EXISTS idx_pastes_shared ON pastes(shared, id);
CREATE INDEX IF NOT EXISTS idx_pastes_shared_at ON pastes(shared, shared_at);
CREATE INDEX IF NOT EXISTS idx_sessions_last_seen ON sessions(last_seen);
CREATE INDEX IF NOT EXISTS idx_images_paste_id ON images(paste_id);
CREATE INDEX IF NOT EXISTS idx_marketing_events_created_at ON marketing_events(created_at DESC);
CREATE INDEX IF NOT EXISTS idx_marketing_events_surface_type ON marketing_events(surface, event_type, created_at DESC);
CREATE INDEX IF NOT EXISTS idx_marketing_events_session ON marketing_events(session_id, created_at DESC);
`);
export default db;