2018-03-28 13:51:05 +00:00
|
|
|
import {EventMessage, Message} from '../chat/common';
|
|
|
|
import core from '../chat/core';
|
|
|
|
import {Conversation, Logs as Logging, Settings} from '../chat/interfaces';
|
|
|
|
|
|
|
|
type StoredConversation = {id: number, key: string, name: string};
|
|
|
|
type StoredMessage = {
|
2018-04-08 00:22:32 +00:00
|
|
|
id: number, conversation: number, type: Conversation.Message.Type, sender: string, text: string, time: Date, day: number | string
|
2018-03-28 13:51:05 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
async function promisifyRequest<T>(req: IDBRequest): Promise<T> {
|
|
|
|
return new Promise<T>((resolve, reject) => {
|
|
|
|
req.onsuccess = () => resolve(<T>req.result);
|
|
|
|
req.onerror = reject;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-04-08 00:22:32 +00:00
|
|
|
async function iterate<S, T>(request: IDBRequest, map: (stored: S) => T, count: number = -1): Promise<T[]> {
|
2018-03-28 13:51:05 +00:00
|
|
|
const array: T[] = [];
|
2018-04-08 00:22:32 +00:00
|
|
|
return new Promise<T[]>((resolve, reject) => {
|
2018-03-28 13:51:05 +00:00
|
|
|
request.onsuccess = function(): void {
|
|
|
|
const c = <IDBCursorWithValue | undefined>this.result;
|
2018-04-08 00:22:32 +00:00
|
|
|
if(!c || count !== -1 && array.length >= count) return resolve(array); //tslint:disable-line:strict-boolean-expressions
|
2018-03-28 13:51:05 +00:00
|
|
|
array.push(map(<S>c.value));
|
|
|
|
c.continue();
|
|
|
|
};
|
|
|
|
request.onerror = reject;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const dayMs = 86400000;
|
2018-04-08 00:22:32 +00:00
|
|
|
const charactersKey = 'fchat.characters';
|
|
|
|
let hasComposite = true;
|
|
|
|
let getComposite: (conv: number, day: number) => string | number[] = (conv, day) => [conv, day];
|
|
|
|
const decode = (str: string) => (str.charCodeAt(0) << 16) + str.charCodeAt(1);
|
|
|
|
try {
|
|
|
|
IDBKeyRange.only([]);
|
|
|
|
} catch {
|
|
|
|
hasComposite = false;
|
|
|
|
const encode = (num: number) => String.fromCharCode((num >> 16) % 65536) + String.fromCharCode(num % 65536);
|
|
|
|
getComposite = (conv, day) => `${encode(conv)}${encode(day)}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
type Index = {[key: string]: StoredConversation | undefined};
|
|
|
|
|
|
|
|
async function openDatabase(character: string): Promise<IDBDatabase> {
|
|
|
|
const request = window.indexedDB.open(`logs-${character}`);
|
|
|
|
request.onupgradeneeded = () => {
|
|
|
|
const db = <IDBDatabase>request.result;
|
|
|
|
const logsStore = db.createObjectStore('logs', {keyPath: 'id', autoIncrement: true});
|
|
|
|
logsStore.createIndex('conversation', 'conversation');
|
|
|
|
logsStore.createIndex('conversation-day', hasComposite ? ['conversation', 'day'] : 'day');
|
|
|
|
db.createObjectStore('conversations', {keyPath: 'id', autoIncrement: true});
|
|
|
|
};
|
|
|
|
return promisifyRequest<IDBDatabase>(request);
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getIndex(db: IDBDatabase): Promise<Index> {
|
|
|
|
const trans = db.transaction(['conversations']);
|
|
|
|
const index: Index = {};
|
|
|
|
await iterate(trans.objectStore('conversations').openCursor(), (x: StoredConversation) => index[x.key] = x);
|
|
|
|
return index;
|
|
|
|
}
|
2018-03-28 13:51:05 +00:00
|
|
|
|
|
|
|
export class Logs implements Logging {
|
2018-04-08 00:22:32 +00:00
|
|
|
index?: Index;
|
|
|
|
loadedDb?: IDBDatabase;
|
|
|
|
loadedCharacter?: string;
|
|
|
|
loadedIndex?: Index;
|
2018-03-28 13:51:05 +00:00
|
|
|
db!: IDBDatabase;
|
|
|
|
|
|
|
|
constructor() {
|
|
|
|
core.connection.onEvent('connecting', async() => {
|
2018-04-08 00:22:32 +00:00
|
|
|
const characters = (await this.getAvailableCharacters());
|
|
|
|
if(characters.indexOf(core.connection.character) === -1)
|
|
|
|
window.localStorage.setItem(charactersKey, JSON.stringify(characters.concat(core.connection.character)));
|
|
|
|
this.db = await openDatabase(core.connection.character);
|
|
|
|
this.index = await getIndex(this.db);
|
2018-03-28 13:51:05 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async logMessage(conversation: Conversation, message: Conversation.Message): Promise<void> {
|
2018-04-08 00:22:32 +00:00
|
|
|
let conv = this.index![conversation.key];
|
2018-03-28 13:51:05 +00:00
|
|
|
if(conv === undefined) {
|
2018-04-08 00:22:32 +00:00
|
|
|
const cTrans = this.db.transaction(['conversations'], 'readwrite');
|
|
|
|
const convId = await promisifyRequest<number>(cTrans.objectStore('conversations').add(
|
2018-03-28 13:51:05 +00:00
|
|
|
{key: conversation.key, name: conversation.name}));
|
2018-04-08 00:22:32 +00:00
|
|
|
this.index![conversation.key] = conv = {id: convId, key: conversation.key, name: conversation.name};
|
2018-03-28 13:51:05 +00:00
|
|
|
}
|
2018-04-08 00:22:32 +00:00
|
|
|
const lTrans = this.db.transaction(['logs'], 'readwrite');
|
2018-03-28 13:51:05 +00:00
|
|
|
const sender = message.type === Conversation.Message.Type.Event ? undefined : message.sender.name;
|
|
|
|
const day = Math.floor(message.time.getTime() / dayMs - message.time.getTimezoneOffset() / 1440);
|
2018-04-08 00:22:32 +00:00
|
|
|
const dayValue = hasComposite ? day : getComposite(conv.id, day);
|
|
|
|
await promisifyRequest<number>(lTrans.objectStore('logs').put(
|
|
|
|
{conversation: conv.id, type: message.type, sender, text: message.text, time: message.time, day: dayValue}));
|
2018-03-28 13:51:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async getBacklog(conversation: Conversation): Promise<ReadonlyArray<Conversation.Message>> {
|
2018-04-08 00:22:32 +00:00
|
|
|
const trans = this.db.transaction(['logs']);
|
|
|
|
const conv = this.index![conversation.key];
|
2018-03-28 13:51:05 +00:00
|
|
|
if(conv === undefined) return [];
|
2018-04-08 00:22:32 +00:00
|
|
|
return (await iterate(trans.objectStore('logs').index('conversation').openCursor(conv.id, 'prev'),
|
2018-03-28 13:51:05 +00:00
|
|
|
(value: StoredMessage) => value.type === Conversation.Message.Type.Event ? new EventMessage(value.text, value.time) :
|
2018-04-08 00:22:32 +00:00
|
|
|
new Message(value.type, core.characters.get(value.sender), value.text, value.time), 20)).reverse();
|
2018-03-28 13:51:05 +00:00
|
|
|
}
|
|
|
|
|
2018-04-08 00:22:32 +00:00
|
|
|
private async loadIndex(character: string): Promise<Index> {
|
|
|
|
if(character === this.loadedCharacter) return this.loadedIndex!;
|
|
|
|
this.loadedCharacter = character;
|
|
|
|
if(character === core.connection.character) {
|
|
|
|
this.loadedDb = this.db;
|
|
|
|
this.loadedIndex = this.index;
|
|
|
|
} else {
|
|
|
|
this.loadedDb = await openDatabase(character);
|
|
|
|
this.loadedIndex = await getIndex(this.loadedDb);
|
|
|
|
}
|
|
|
|
return this.loadedIndex!;
|
2018-03-28 13:51:05 +00:00
|
|
|
}
|
|
|
|
|
2018-04-08 00:22:32 +00:00
|
|
|
async getConversations(character: string): Promise<ReadonlyArray<{key: string, name: string}>> {
|
|
|
|
const index = await this.loadIndex(character);
|
|
|
|
return Object.keys(index).map((k) => index[k]!);
|
|
|
|
}
|
|
|
|
|
|
|
|
async getLogs(character: string, key: string, date: Date): Promise<ReadonlyArray<Conversation.Message>> {
|
|
|
|
const id = (await this.loadIndex(character))[key]!.id;
|
|
|
|
const trans = this.loadedDb!.transaction(['logs']);
|
2018-03-28 13:51:05 +00:00
|
|
|
const day = Math.floor(date.getTime() / dayMs - date.getTimezoneOffset() / 1440);
|
2018-04-08 00:22:32 +00:00
|
|
|
return iterate(trans.objectStore('logs').index('conversation-day').openCursor(getComposite(id, day)),
|
2018-03-28 13:51:05 +00:00
|
|
|
(value: StoredMessage) => value.type === Conversation.Message.Type.Event ? new EventMessage(value.text, value.time) :
|
|
|
|
new Message(value.type, core.characters.get(value.sender), value.text, value.time));
|
|
|
|
}
|
|
|
|
|
2018-04-08 00:22:32 +00:00
|
|
|
async getLogDates(character: string, key: string): Promise<ReadonlyArray<Date>> {
|
|
|
|
const id = (await this.loadIndex(character))[key]!.id;
|
|
|
|
const trans = this.loadedDb!.transaction(['logs']);
|
|
|
|
const offset = new Date().getTimezoneOffset() * 60000;
|
|
|
|
const bound = IDBKeyRange.bound(getComposite(id, 0), getComposite(id, 1000000));
|
|
|
|
return iterate(trans.objectStore('logs').index('conversation-day').openCursor(bound, 'nextunique'), (value: StoredMessage) =>
|
|
|
|
new Date((hasComposite ? <number>value.day : decode((<string>value.day).substr(2))) * dayMs + offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
async getAvailableCharacters(): Promise<ReadonlyArray<string>> {
|
|
|
|
const stored = window.localStorage.getItem(charactersKey);
|
|
|
|
return stored !== null ? JSON.parse(stored) as string[] : [];
|
2018-03-28 13:51:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export class SettingsStore implements Settings.Store {
|
2018-04-08 00:22:32 +00:00
|
|
|
async get<K extends keyof Settings.Keys>(key: K): Promise<Settings.Keys[K] | undefined> {
|
|
|
|
const stored = window.localStorage.getItem(`${core.connection.character}.settings.${key}`);
|
|
|
|
return stored !== null ? JSON.parse(stored) as Settings.Keys[K] : undefined;
|
2018-03-28 13:51:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async set<K extends keyof Settings.Keys>(key: K, value: Settings.Keys[K]): Promise<void> {
|
2018-04-08 00:22:32 +00:00
|
|
|
window.localStorage.setItem(`${core.connection.character}.settings.${key}`, JSON.stringify(value));
|
2018-03-28 13:51:05 +00:00
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
async getAvailableCharacters(): Promise<ReadonlyArray<string>> {
|
2018-04-08 00:22:32 +00:00
|
|
|
const stored = window.localStorage.getItem(charactersKey);
|
|
|
|
return stored !== null ? JSON.parse(stored) as string[] : [];
|
2018-03-28 13:51:05 +00:00
|
|
|
}
|
|
|
|
}
|