cordis/client/webapp/personal-db.ts
2021-11-30 20:45:06 -06:00

535 lines
18 KiB
TypeScript

import * as crypto from 'crypto';
import ConcurrentQueue from "../../concurrent-queue/concurrent-queue";
import * as sqlite from 'sqlite';
import * as sqlite3 from 'sqlite3';
import { Channel, GuildMetadataWithIds, Member, Message, Resource, SocketConfig } from "./data-types";
export default class PersonalDB {
private transactions = new ConcurrentQueue(1);
private constructor(
private readonly db: sqlite.Database
) {}
public static async create(filePath: string): Promise<PersonalDB> {
return new PersonalDB(
await sqlite.open({ driver: sqlite3.Database, filename: filePath })
);
}
async open(): Promise<void> {
await this.db.open();
}
async close(): Promise<void> {
await this.db.close();
}
async beginTransaction(): Promise<void> {
await this.db.run('BEGIN TRANSACTION');
}
async rollbackTransaction(): Promise<void> {
await this.db.run('ROLLBACK');
}
async commitTransaction(): Promise<void> {
await this.db.run('COMMIT');
}
async queueTransaction(func: (() => Promise<void>)): Promise<void> {
await this.transactions.push(async () => {
try {
await this.beginTransaction();
await func();
await this.commitTransaction();
} catch (e) {
await this.rollbackTransaction();
throw e;
}
});
}
async init(): Promise<void> {
await this.queueTransaction(async () => {
// NOTE: Guild ID is not shared between instances of corDis
await this.db.run(`
CREATE TABLE IF NOT EXISTS guilds (
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT
, name TEXT
, icon_resource_id TEXT
, member_id TEXT
)
`);
await this.db.run(`
CREATE TABLE IF NOT EXISTS guild_sockets (
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT
, guild_id INTEGER NOT NULL
, url TEXT NOT NULL
, cert TEXT NOT NULL
, public_key TEXT NOT NULL
, private_key TEXT NOT NULL
, FOREIGN KEY (guild_id) REFERENCES guilds(id)
)
`);
await this.db.run(`
CREATE TABLE IF NOT EXISTS members (
id TEXT NOT NULL
, guild_id INTEGER NOT NULL REFERENCES guilds(id)
, display_name TEXT NOT NULL
, status TEXT NOT NULL
, avatar_resource_id TEXT NOT NULL
, role_name TEXT
, role_color TEXT
, role_priority INTEGER
, privileges TEXT
, CONSTRAINT members_id_guild_id_con UNIQUE (id, guild_id)
)
`);
await this.db.run(`
CREATE TABLE IF NOT EXISTS channels (
id TEXT NOT NULL
, guild_id INTEGER NOT NULL REFERENCES guilds(id)
, "index" INTEGER NOT NULL
, name TEXT NOT NULL
, flavor_text TEXT
, CONSTRAINT channels_id_guild_id_con UNIQUE (id, guild_id)
)
`);
await this.db.run(`
CREATE TABLE IF NOT EXISTS resources (
id TEXT NOT NULL
, guild_id INTEGER NOT NULL REFERENCES guilds(id)
, hash BLOB NOT NULL
, data BLOB NOT NULL
, data_size INTEGER NOT NULL
, last_used INTEGER NOT NULL
, CONSTRAINT resources_id_guild_id_con UNIQUE (id, guild_id)
)
`);
await this.db.run('CREATE INDEX IF NOT EXISTS resources_data_size_idx ON resources (data_size)');
// note: no foreign key on resource_id since we may not have cached the resource yet
await this.db.run(`
CREATE TABLE IF NOT EXISTS messages (
id TEXT NOT NULL
, guild_id INTEGER NOT NULL REFERENCES guilds(id)
, channel_id TEXT NOT NULL REFERENCES channels(id)
, member_id TEXT NOT NULL REFERENCES members(id)
, sent_dtg INTEGER NOT NULL
, text TEXT
, resource_id TEXT
, resource_name TEXT
, resource_width INTEGER
, resource_height INTEGER
, resource_preview_id TEXT
, CONSTRAINT messages_id_guild_id_con UNIQUE (id, guild_id)
)
`);
await this.db.run('CREATE INDEX IF NOT EXISTS messages_id_idx ON messages (id)');
await this.db.run('CREATE INDEX IF NOT EXISTS messages_sent_dtg_idx ON messages (sent_dtg)');
});
}
// dangerous!
async reset(): Promise<void> {
await this.queueTransaction(async () => {
await this.db.run('DROP TABLE IF EXISTS guilds');
await this.db.run('DROP TABLE IF EXISTS guild_sockets');
await this.db.run('DROP TABLE IF EXISTS members');
await this.db.run('DROP TABLE IF EXISTS channels');
await this.db.run('DROP TABLE IF EXISTS resources');
await this.db.run('DROP TABLE IF EXISTS messages');
});
}
// Guilds
async addGuild(name: string, iconResourceId: string, memberId: string): Promise<number> {
let result = await this.db.run(
`INSERT INTO guilds (name, icon_resource_id, member_id) VALUES (:name, :icon_resource_id, :member_id)`,
{ ':name': name, ':icon_resource_id': iconResourceId, ':member_id': memberId }
);
if (result.changes !== 1) throw new Error('unable to add guild');
if (result.lastID === undefined) throw new Error('unable to get guild last id');
return result.lastID as number;
}
async removeGuild(guildId: number): Promise<void> {
let result = await this.db.run(
`DELETE FROM guilds WHERE id=:id`,
{ ':id': guildId }
);
if (result?.changes !== 1) throw new Error('unable to remove guild');
}
async updateGuildName(guildId: number, newName: string): Promise<void> {
let result = await this.db.run(
`UPDATE guilds SET name=:name WHERE id=:guild_id`,
{ ':guild_id': guildId, ':name': newName }
);
if (result?.changes !== 1) throw new Error('unable to update guild name');
}
async updateGuildIcon(guildId: number, newIconResourceId: string): Promise<void> {
let result = await this.db.run(
`UPDATE guilds SET icon_resource_id=:icon_resource_id WHERE id=:guild_id`,
{ ':guild_id': guildId, ':icon_resource_id': newIconResourceId }
);
if (result?.changes !== 1) throw new Error('unable to update guild icon');
}
async fetchGuild(guildId: number): Promise<GuildMetadataWithIds> {
let result = await this.db.get(
`SELECT * FROM guilds WHERE id=:id`,
{ ':id': guildId }
);
if (!result) throw new Error('unable to fetch guild');
return GuildMetadataWithIds.fromDBData(result);
}
async fetchGuilds(): Promise<GuildMetadataWithIds[]> {
let result = await this.db.all('SELECT * FROM guilds');
return result.map(dataGuild => GuildMetadataWithIds.fromDBData(dataGuild));
}
// Guild Sockets
async addGuildSocket(guildId: number, url: string, cert: string, publicKey: crypto.KeyObject, privateKey: crypto.KeyObject): Promise<number> {
let publicKeyPem = publicKey.export({ type: 'spki', format: 'pem' });
let privateKeyPem = privateKey.export({ type: 'pkcs8', format: 'pem' });
let result = await this.db.run(
`INSERT INTO guild_sockets (guild_id, url, cert, public_key, private_key) VALUES (:guild_id, :url, :cert, :public_key, :private_key)`,
{ ':guild_id': guildId, ':url': url, ':cert': cert, ':public_key': publicKeyPem, ':private_key': privateKeyPem }
);
if (result.changes !== 1) throw new Error('unable to add guild');
return result.lastID as number;
}
async removeGuildSocket(guildId: number, guildSocketId: number): Promise<void> {
let result = await this.db.run(
`DELETE FROM guild_sockets WHERE id=:guild_socket_id AND guild_id=:guild_id`,
{ ':guild_id': guildId, ':guild_socket_id': guildSocketId }
);
if (result?.changes !== 1) throw new Error('unable to remove guild');
}
async removeGuildSockets(guildId: number): Promise<void> {
let result = await this.db.run(
`DELETE FROM guild_sockets WHERE guild_id=:guild_id`,
{ ':guild_id': guildId }
);
if (result?.changes !== 1) throw new Error('unable to remove guild');
}
async fetchGuildSockets(guildId: number): Promise<SocketConfig[]> {
let result = await this.db.all(
`SELECT * FROM guild_sockets WHERE guild_id=:guild_id`,
{ ':guild_id': guildId }
);
return result.map(dataGuildSocket => SocketConfig.fromDBData(dataGuildSocket));
}
async fetchGuildSocket(guildId: number, guildSocketId: number): Promise<SocketConfig> {
let result = await this.db.get(
`SELECT * FROM guild_sockets WHERE id=:guild_socket_id AND guild_id=:guild_id`,
{ ':guild_socket_id': guildSocketId, ':guild_id': guildId }
);
if (!result) throw new Error('unable to fetch specific guild socket');
return SocketConfig.fromDBData(result);
}
// Members
async addMembers(guildId: number, members: Member[]): Promise<void> {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`INSERT INTO members (
id, guild_id, display_name, status, avatar_resource_id
, role_name, role_color, role_priority, privileges
) VALUES (
:id, :guild_id, :display_name, :status, :avatar_resource_id
, :role_name, :role_color, :role_priority, :privileges
)`
);
for (let member of members) {
await stmt.run({
':id': member.id, ':guild_id': guildId, ':display_name': member.displayName, ':status': member.status, ':avatar_resource_id': member.avatarResourceId
, ':role_name': member.roleName, ':role_color': member.roleColor, ':role_priority': member.rolePriority, ':privileges': member.privileges.join(',')
});
}
await stmt.finalize();
});
}
async updateMembers(guildId: number, newMembers: Member[]): Promise<void> {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`UPDATE members SET
display_name=:display_name, status=:status, avatar_resource_id=:avatar_resource_id
, role_name=:role_name, role_color=:role_color, role_priority=:role_priority, privileges=:privileges
WHERE
id=:id AND guild_id=:guild_id`
);
for (let member of newMembers) {
await stmt.run({
':id': member.id, ':guild_id': guildId, ':display_name': member.displayName, ':status': member.status, ':avatar_resource_id': member.avatarResourceId
, ':role_name': member.roleName, ':role_color': member.roleColor, ':role_priority': member.rolePriority, ':privileges': member.privileges.join(',')
});
}
await stmt.finalize();
})
}
async deleteMembers(guildId: number, deletedMembers: Member[]): Promise<void> {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`DELETE FROM members WHERE id=:id AND guild_id=:guild_id`
);
for (let member of deletedMembers) {
await stmt.run({ ':id': member.id, ':guild_id': guildId });
}
await stmt.finalize();
})
}
async fetchMembers(guildId: number): Promise<Member[] | null> {
let result = await this.db.all('SELECT * FROM members WHERE guild_id=:guild_id', { ':guild_id': guildId });
if (result.length === 0) return null;
return result.map(dataMember => Member.fromDBData(dataMember));
}
async clearAllMembersStatus(guildId: number): Promise<void> {
await this.db.run(`UPDATE members SET status='unknown' WHERE guild_id=:guild_id`, { ':guild_id': guildId });
}
// Channels
async addChannels(guildId: number, channels: Channel[]) {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`INSERT INTO channels (
id, guild_id, "index", name, flavor_text
) VALUES (
:id, :guild_id, :index, :name, :flavor_text
)`
);
for (let channel of channels) {
await stmt.run({
':id': channel.id, ':guild_id': guildId, ':index': channel.index, ':name': channel.name, ':flavor_text': channel.flavorText
});
}
await stmt.finalize();
});
}
async updateChannels(guildId: number, newChannels: Channel[]): Promise<void> {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`UPDATE channels SET
"index"=:index, name=:name, flavor_text=:flavor_text
WHERE
id=:id AND guild_id=:guild_id`
);
for (let channel of newChannels) {
await stmt.run({
':id': channel.id, ':guild_id': guildId, ':index': channel.index, ':name': channel.name, ':flavor_text': channel.flavorText
});
}
await stmt.finalize();
})
}
async deleteChannels(guildId: number, deletedChannels: Channel[]): Promise<void> {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`DELETE FROM channels WHERE id=:id AND guild_id=:guild_id`
);
for (let channel of deletedChannels) {
await stmt.run({ ':id': channel.id, ':guild_id': guildId });
}
await stmt.finalize();
})
}
async fetchChannels(guildId: number): Promise<Channel[] | null> {
let result = await this.db.all('SELECT * FROM channels WHERE guild_id=:guild_id', { ':guild_id': guildId });
if (result.length === 0) return null;
return result.map(dataMember => Channel.fromDBData(dataMember));
}
// Resources
async addResources(guildId: number, resources: Resource[]) {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`INSERT INTO resources (
id, guild_id, hash, data, data_size, last_used
) VALUES (
:id, :guild_id, :hash, :data, :data_size, :last_used
)`
);
for (let resource of resources) {
await stmt.run({
':id': resource.id, ':guild_id': guildId, ':hash': resource.hash,
':data': resource.data, ':data_size': resource.data.length, ':last_used': Date.now()
});
}
await stmt.finalize();
});
}
async updateResources(guildId: number, newResources: Resource[]): Promise<void> {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`UPDATE resources SET
hash=:hash, data=:data, data_size=:data_size, last_used=:last_used
WHERE
id=:id AND guild_id=:guild_id`
);
for (let resource of newResources) {
await stmt.run({
':id': resource.id, ':guild_id': guildId, ':hash': resource.hash,
':data': resource.data, ':data_size': resource.data.length, ':last_used': Date.now()
});
}
await stmt.finalize();
})
}
async deleteResources(guildId: number, deletedResources: Resource[]): Promise<void> {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`DELETE FROM resources WHERE id=:id AND guild_id=:guild_id`
);
for (let resource of deletedResources) {
await stmt.run({ ':id': resource.id, ':guild_id': guildId });
}
await stmt.finalize();
})
}
async fetchResource(guildId: number, resourceId: string): Promise<Resource | null> {
let result = await this.db.get(
`SELECT * FROM resources WHERE id=:id AND guild_id=:guild_id`,
{ ':id': resourceId, ':guild_id': guildId }
);
if (!result) return null;
return Resource.fromDBData(result);
}
// Messages
async addMessages(guildId: number, messages: Message[]) {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`INSERT INTO messages (
id, guild_id, channel_id, member_id, sent_dtg, text
, resource_id, resource_name, resource_width, resource_height, resource_preview_id
) VALUES (
:id, :guild_id, :channel_id, :member_id, :sent_dtg, :text
, :resource_id, :resource_name, :resource_width, :resource_height, :resource_preview_id
)`
);
for (let message of messages) {
await stmt.run({
':id': message.id, ':guild_id': guildId,
':channel_id': message.channel.id, ':member_id': message.member.id,
':sent_dtg': message.sent.getTime(), ':text': message.text,
':resource_id': message.resourceId, ':resource_name': message.resourceName,
':resource_width': message.resourceWidth, ':resource_height': message.resourceHeight,
':resource_preview_id': message.resourcePreviewId
});
}
await stmt.finalize();
});
}
async updateMessages(guildId: number, newMessages: Message[]): Promise<void> {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`UPDATE resources SET
hash=:hash, data=:data, data_size=:data_size, last_used=:last_used
WHERE
id=:id AND guild_id=:guild_id`
);
for (let message of newMessages) {
await stmt.run({
':id': message.id, ':guild_id': guildId,
':channel_id': message.channel.id, ':member_id': message.member.id,
':sent_dtg': message.sent.getTime(), ':text': message.text,
':resource_id': message.resourceId, ':resource_name': message.resourceName,
':resource_width': message.resourceWidth, ':resource_height': message.resourceHeight,
':resource_preview_id': message.resourcePreviewId
});
}
await stmt.finalize();
})
}
async deleteMessages(guildId: number, deletedMessages: Message[]): Promise<void> {
await this.queueTransaction(async () => {
let stmt = await this.db.prepare(
`DELETE FROM resources WHERE id=:id AND guild_id=:guild_id`
);
for (let message of deletedMessages) {
await stmt.run({ ':id': message.id, ':guild_id': guildId });
}
await stmt.finalize();
})
}
async fetchMessagesRecent(guildId: number, channelId: string, number: number): Promise<Message[] | null> {
let messages = await this.db.all(`
SELECT * FROM (
SELECT *
FROM "messages"
WHERE "guild_id"=:guild_id AND "channel_id"=:channel_id
ORDER BY "sent_dtg" DESC
LIMIT :number
) AS "r" ORDER BY "r"."sent_dtg" ASC
`, { ':guild_id': guildId, ':channel_id': channelId, ':number': number });
if (messages.length === 0) return null;
return messages.map(dataMessage => Message.fromDBData(dataMessage));
}
async fetchMessagesBefore(guildId: number, channelId: string, messageId: string, number: number): Promise<Message[] | null> {
let messages = await this.db.all(`
SELECT * FROM (
SELECT *
FROM "messages"
WHERE
"guild_id"=:guild_id
AND "channel_id"=:channel_id
AND "sent_dtg" < (SELECT "sent_dtg" FROM "messages" WHERE "id"=:message_id)
ORDER BY "sent_dtg" DESC
LIMIT :number
) AS "r" ORDER BY "r"."sent_dtg" ASC
`, { ':guild_id': guildId, ':channel_id': channelId, ':message_id': messageId, ':number': number });
if (messages.length === 0) return null;
return messages.map(dataMessage => Message.fromDBData(dataMessage));
}
async fetchMessagesAfter(guildId: number, channelId: string, messageId: string, number: number): Promise<Message[] | null> {
let messages = await this.db.all(`
SELECT *
FROM "messages"
WHERE
"guild_id"=:guild_id
AND "channel_id"=:channel_id
AND "sent_dtg" > (SELECT "sent_dtg" FROM "messages" WHERE "id"=:message_id)
ORDER BY "sent_dtg" ASC
LIMIT :number
`, { ':guild_id': guildId, ':channel_id': channelId, ':message_id': messageId, ':number': number });
if (messages.length === 0) return null;
return messages.map(dataMessage => Message.fromDBData(dataMessage));
}
}