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

210 lines
5.0 KiB
TypeScript

import crypto from 'crypto'
2 years ago
import { readFile, writeFile, access, constants } from 'fs/promises'
import {
MetadataProps as MetaProps,
Problem,
ProblemId,
Solution,
SolutionId,
User,
UserId,
} from '../../shared/model'
function once<T extends (...args: any) => any>(fn: T, message: string): T {
let flag = false
return ((...args: any) => {
if (flag) {
throw new Error(message ?? `cannot run more than once`)
}
flag = true
return fn(...args)
}) as T
}
type Lock = () => void
type Mutex = {
lock(): Promise<Lock>
}
function createMutex(): Mutex {
let locked = false
const waiters: ((fn: () => void) => void)[] = []
const unlock = () => {
if (waiters.length > 0) {
console.log(`[Mutex] Passing lock to next in queue (of size ${waiters.length})`)
2 years ago
const resolve = waiters.shift()!!
resolve(once(unlock, `lock already released`))
} else {
locked = false
console.log(`[Mutex] Releasing the lock`)
}
}
const lock = (): Promise<Lock> => {
if (locked) {
console.log(`[Mutex] Putting into queue`)
return new Promise(resolve => {
waiters.push(resolve)
})
} else {
console.log(`[Mutex] Acquiring the lock`)
locked = true
return Promise.resolve(once(unlock, `lock already released`))
}
}
return { lock }
}
type DatabaseConnection = {
path: string
initialValue: Database
mu: Mutex
}
export type Database = {
users: Record<string, User>
problems: Record<string, Problem>
solutions: Record<string, Solution>
}
export function createDatabase(path: string, initialValue: Database) {
return {
path,
initialValue,
mu: createMutex(),
}
2 years ago
}
async function withDatabase<R>(
{ path, initialValue, mu }: DatabaseConnection,
fn: (db: Database) => R | Promise<R>
): Promise<R> {
const unlock: Lock = await mu.lock()
2 years ago
try {
await access(path, constants.R_OK)
} catch (e) {
console.log(`[Database] Creating empty database into "${path}"`)
await writeFile(path, JSON.stringify(initialValue, null, 4))
2 years ago
}
console.log(`[Database] Loading database from "${path}"`)
2 years ago
const state = JSON.parse(await readFile(path, 'utf-8'))
const result = await fn(state)
console.log(`[Database] Saving database to "${path}"`)
await writeFile(path, JSON.stringify(state, null, 4))
2 years ago
unlock()
2 years ago
return result
2 years ago
}
//
// Users
//
export const getUsers = (db: DatabaseConnection) =>
withDatabase(db, state => {
return Object.values(state.users)
})
2 years ago
2 years ago
export const getUser: (db: DatabaseConnection, id: string) => Promise<User | null> = (db, id) =>
withDatabase(db, state => {
return state.users[id] ?? null
})
//
// Problems
//
export const createProblem = (
db: DatabaseConnection,
{ content, createdBy }: Omit<Problem, 'id' | 'createdAt'>
): Promise<ProblemId> =>
withDatabase(db, state => {
const nextId = (Object.keys(state.problems).length + 1).toString() as ProblemId
state.problems[nextId] = {
id: nextId,
content,
createdBy,
createdAt: new Date().toJSON(),
}
return nextId
})
2 years ago
export const getProblem = (db: DatabaseConnection, id: string): Promise<Problem | null> =>
withDatabase(db, state => {
2 years ago
return state.problems[id] ?? null
2 years ago
})
export const getProblems = (db: DatabaseConnection): Promise<Problem[]> =>
2 years ago
withDatabase(db, state => {
return Object.values(state.problems)
})
//
// Solutions
//
export const createSolution = (
db: DatabaseConnection,
2 years ago
{ sentBy, forProblem, content }: Omit<Solution, MetaProps | 'status' | 'visible'>
): Promise<SolutionId> =>
withDatabase(db, state => {
2 years ago
const id = crypto.randomBytes(10).toString('hex') as SolutionId
state.solutions[id] = {
id,
2 years ago
createdAt: new Date().toISOString(),
sentBy,
forProblem,
content,
status: 'pending',
2 years ago
visible: false,
2 years ago
}
return id
2 years ago
})
2 years ago
export const getSolution = (db: DatabaseConnection, id: SolutionId): Promise<Solution | null> =>
withDatabase(db, state => {
2 years ago
return state.solutions[id] ?? null
})
export const updateSolution = (
db: DatabaseConnection,
id: SolutionId,
2 years ago
solution: Partial<Omit<Solution, MetaProps>>
): Promise<Solution> =>
withDatabase(db, state => {
2 years ago
state.solutions[id] = {
...state.solutions[id],
...solution,
}
return state.solutions[id]
})
2 years ago
export const getSolutions = (db: DatabaseConnection) =>
2 years ago
withDatabase(db, state => {
let solutions = Object.values(state.solutions)
return solutions
2 years ago
})
2 years ago
export const getVisibleSolutions = (db: DatabaseConnection) =>
withDatabase(db, state => {
return Object.values(state.solutions).filter(s => s.visible)
})