You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

120 lines
2.9 KiB

import crypto from 'node:crypto';
import debug from 'debug';
import { join } from 'path';
import {
DatabaseProvider,
DatabaseError,
DEFAULT_DATABASE,
} from './repositories/database';
import { JSONFilePreset } from 'lowdb/node';
import type { Low } from 'lowdb';
import type { User } from './repositories/user';
import type { Database } from './repositories/database';
import { migrationRunner } from './migrations';
const DEBUG = debug('LowDB');
export default class LowDB extends DatabaseProvider {
#db!: Low<Database>;
// is this really needed?
private async __init() {
// TODO: assume path to db file
const dbFilePath = join(WG_PATH, 'db.json');
this.#db = await JSONFilePreset(dbFilePath, DEFAULT_DATABASE);
}
/**
* @throws
*/
async connect() {
try {
await this.__init();
DEBUG('Running Migrations');
await migrationRunner(this.#db);
DEBUG('Migrations ran successfully');
} catch (e) {
DEBUG(e);
throw new DatabaseError(DatabaseError.ERROR_INIT);
}
DEBUG('Connected successfully');
}
async disconnect() {
DEBUG('Disconnected successfully');
}
async getSystem() {
DEBUG('Get System');
const system = this.#db.data.system;
// system is only null if migration failed
if (system === null) {
throw new DatabaseError(DatabaseError.ERROR_INIT);
}
return system;
}
async getUsers() {
return this.#db.data.users;
}
async getUser(id: string) {
DEBUG('Get User');
return this.#db.data.users.find((user) => user.id === id);
}
async newUserWithPassword(username: string, password: string) {
DEBUG('New User');
// TODO: should be handled by zod. completely remove database error
if (username.length < 8) {
throw new DatabaseError(DatabaseError.ERROR_USERNAME_REQ);
}
if (!isPasswordStrong(password)) {
throw new DatabaseError(DatabaseError.ERROR_PASSWORD_REQ);
}
const isUserExist = this.#db.data.users.find(
(user) => user.username === username
);
if (isUserExist) {
throw new DatabaseError(DatabaseError.ERROR_USER_EXIST);
}
const now = new Date();
const isUserEmpty = this.#db.data.users.length === 0;
const newUser: User = {
id: crypto.randomUUID(),
password: hashPassword(password),
username,
role: isUserEmpty ? 'ADMIN' : 'CLIENT',
enabled: true,
createdAt: now,
updatedAt: now,
};
this.#db.update((data) => data.users.push(newUser));
}
async updateUser(user: User) {
let oldUser = await this.getUser(user.id);
if (oldUser) {
DEBUG('Update User');
oldUser = user;
this.#db.write();
}
}
async deleteUser(id: string) {
DEBUG('Delete User');
const idx = this.#db.data.users.findIndex((user) => user.id === id);
if (idx !== -1) {
this.#db.update((data) => data.users.splice(idx, 1));
}
}
}