feat: new user resource integrating postgresql
This commit is contained in:
@@ -1,2 +1,2 @@
|
|||||||
export type Maybe<T> = Just<T> | { just?: never, error: { message: string }};
|
export type Maybe<T> = Just<T> | { just?: never; error: { message: string } };
|
||||||
export type Just<T> = { just: T, error?: never };
|
export type Just<T> = { just: T; error?: never };
|
||||||
|
|||||||
@@ -1,7 +1,22 @@
|
|||||||
export type HttpMethod = "POST" | "GET" | "PUT" | "DELETE";
|
export type HttpMethod = "POST" | "GET" | "PUT" | "DELETE";
|
||||||
|
|
||||||
export default interface StoccaTreRequest {
|
export type RouteDefinition = {
|
||||||
method: HttpMethod,
|
pattern: RegExp;
|
||||||
route: string,
|
method: HttpMethod;
|
||||||
body: string | null,
|
};
|
||||||
|
|
||||||
|
export default class StoccaTreRequest {
|
||||||
|
constructor(
|
||||||
|
public method: HttpMethod,
|
||||||
|
public route: string,
|
||||||
|
public body: string | null,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
match(route: RouteDefinition): RegExpExecArray | null {
|
||||||
|
const patternResult = route.pattern.exec(this.route);
|
||||||
|
if (route.method !== this.method) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return patternResult;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import StoccaTreRequest from "./StoccaTreRequest.ts";
|
import StoccaTreRequest from "./StoccaTreRequest.ts";
|
||||||
import {Maybe} from "./Maybe.ts";
|
import { Maybe } from "./Maybe.ts";
|
||||||
import {JSONObject} from "./JSON.ts";
|
import { JSONObject } from "./JSON.ts";
|
||||||
|
|
||||||
export default interface StoccaTreRequestHandler {
|
export default interface StoccaTreRequestHandler {
|
||||||
handleRequest(request: StoccaTreRequest): Promise<Maybe<JSONObject>>,
|
handleRequest(request: StoccaTreRequest): Promise<Maybe<JSONObject>>;
|
||||||
}
|
}
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
{
|
|
||||||
"username": "root",
|
|
||||||
"port": 8080,
|
|
||||||
"hostname": "127.0.0.1",
|
|
||||||
"password": "sekna123jk"
|
|
||||||
}
|
|
||||||
@@ -1,6 +1,17 @@
|
|||||||
type ServerConfig = {
|
const config = {
|
||||||
username: string,
|
username: Deno.env.get("DB_USER") ?? "postgres",
|
||||||
hostname: string,
|
hostname: Deno.env.get("DB_HOST") ?? "localhost",
|
||||||
password: string,
|
password: Deno.env.get("DB_PW") ?? "",
|
||||||
port: number,
|
dbPort: Number(Deno.env.get("DB_PORT") ?? 5432),
|
||||||
}
|
port: Number(Deno.env.get("PORT") ?? 8080),
|
||||||
|
};
|
||||||
|
|
||||||
|
console.log(`ENV:
|
||||||
|
db username: ${config.username}
|
||||||
|
db pass: ******
|
||||||
|
db port: ${config.dbPort}
|
||||||
|
server port: ${config.port}
|
||||||
|
server hostname: ${config.hostname}
|
||||||
|
`);
|
||||||
|
|
||||||
|
export default config;
|
||||||
|
|||||||
@@ -1,38 +1,41 @@
|
|||||||
import { Client } from "https://deno.land/x/mysql/mod.ts";
|
import { Client } from "postgres";
|
||||||
import config from "./config.json" assert { type: "json" };
|
import config from "./config.ts";
|
||||||
import {Maybe} from "./Maybe.ts";
|
import {Maybe} from "./Maybe.ts";
|
||||||
|
import {JSONObject} from "./JSON.ts";
|
||||||
|
|
||||||
export type WithoutId<T> = Omit<T, "id">;
|
export type WithoutId<T> = Omit<T, "id">;
|
||||||
|
|
||||||
export interface StoccaTreDbConn {
|
export interface StoccaTreDbConn {
|
||||||
query<T>(query: string): Promise<Maybe<T>>;
|
query<T extends JSONObject | JSONObject[]>(query: string): Promise<Maybe<T>>,
|
||||||
}
|
}
|
||||||
export default async function createNewDbConnection(): Promise<StoccaTreDbConn> {
|
|
||||||
const mysqlClient = await new Client().connect({
|
export default function createNewDbConnection(): StoccaTreDbConn {
|
||||||
|
const postgresClient = new Client({
|
||||||
hostname: config.hostname,
|
hostname: config.hostname,
|
||||||
username: config.username,
|
|
||||||
db: "stocca_tre",
|
|
||||||
password: config.password,
|
password: config.password,
|
||||||
|
user: config.username,
|
||||||
|
database: "stocca_tre",
|
||||||
|
port: config.port,
|
||||||
});
|
});
|
||||||
|
await postgresClient.connect();
|
||||||
return {
|
return {
|
||||||
query: async <T>(query: string): Promise<Maybe<T>> => {
|
async queryMany<T extends JSONObject[]>(query: string): Promise<Maybe<{ rows: T, count: number }>> {
|
||||||
let errMessage: string;
|
|
||||||
try {
|
try {
|
||||||
const result = await mysqlClient.query(query);
|
const result = await postgresClient.queryArray<T>(query);
|
||||||
return {
|
return {
|
||||||
just: result as T,
|
just: {
|
||||||
};
|
rows: result.rows,
|
||||||
} catch (e: unknown) {
|
count: result.rowCount ?? NaN,
|
||||||
if (e && typeof (e as { message?: any }).message === "string") {
|
|
||||||
errMessage = (e as { message: string}).message;
|
|
||||||
} else {
|
|
||||||
errMessage = "An unknown error occurred in the database.";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return {
|
|
||||||
error: {
|
|
||||||
message: errMessage
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
} catch (e: unknown) {
|
||||||
|
}
|
||||||
|
},
|
||||||
|
queryOne<T extends JSONObject>(query: string): Promise<Maybe<T>> {
|
||||||
|
try {
|
||||||
|
const result = await postgresClient.queryObject<T>(query);
|
||||||
|
return { just: result };
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,7 @@
|
|||||||
{
|
{
|
||||||
"imports": {
|
"imports": {
|
||||||
|
"zod": "https://deno.land/x/zod@v3.17.3/mod.ts",
|
||||||
|
"postgres": "https://deno.land/x/postgres@v0.16.1/mod.ts",
|
||||||
|
"bcrypt": "https://deno.land/x/bcrypt@v0.4.0/mod.ts"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
import createNewDbConnection, {StoccaTreDbConn} from "./database.ts";
|
import createNewDbConnection, { StoccaTreDbConn } from "./database.ts";
|
||||||
import * as resources from "./resources/main.ts";
|
import * as resources from "./resources/main.ts";
|
||||||
import config from "./config.json" assert { type: "json" };
|
import config from "./config.ts";
|
||||||
import StoccaTreRequest, {HttpMethod} from "./StoccaTreRequest.ts";
|
import StoccaTreRequest, { HttpMethod } from "./StoccaTreRequest.ts";
|
||||||
import StoccaTreRequestHandler from "./StoccaTreRequestHandler.ts";
|
import StoccaTreRequestHandler from "./StoccaTreRequestHandler.ts";
|
||||||
import {JSONObject} from "./JSON.ts";
|
import { JSONObject } from "./JSON.ts";
|
||||||
|
|
||||||
type StoccaTreApiBody = {
|
type StoccaTreApiBody = {
|
||||||
data: JSONObject,
|
data: JSONObject;
|
||||||
error?: string,
|
error?: string;
|
||||||
}
|
};
|
||||||
|
|
||||||
function newStoccaTreApiBody(): StoccaTreApiBody {
|
function newStoccaTreApiBody(): StoccaTreApiBody {
|
||||||
return {
|
return {
|
||||||
@@ -17,22 +17,22 @@ function newStoccaTreApiBody(): StoccaTreApiBody {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type StoccaTreServer = StoccaTreRequestHandler & {
|
type StoccaTreServer = StoccaTreRequestHandler & {
|
||||||
dbConnection: StoccaTreDbConn,
|
dbConnection: StoccaTreDbConn;
|
||||||
}
|
};
|
||||||
|
|
||||||
async function processRequest(server: StoccaTreServer, requestEvent: Deno.RequestEvent) {
|
async function processRequest(server: StoccaTreServer, requestEvent: Deno.RequestEvent) {
|
||||||
const route = /^https?:\/\/[^\/]*(\/.*)$/.exec(requestEvent.request.url)?.[1] ?? "/";
|
const route = /^https?:\/\/[^\/]*(\/.*)$/.exec(requestEvent.request.url)?.[1] ?? "/";
|
||||||
const requestBody = (await requestEvent.request.body?.getReader().read())?.value?.toString() ?? null;
|
const requestBody = (await (await requestEvent.request.blob()).text()) ?? null;
|
||||||
console.log(requestBody);
|
|
||||||
const method: HttpMethod = requestEvent.request.method as HttpMethod;
|
const method: HttpMethod = requestEvent.request.method as HttpMethod;
|
||||||
const body: StoccaTreApiBody = newStoccaTreApiBody();
|
const body: StoccaTreApiBody = newStoccaTreApiBody();
|
||||||
const result = await server.handleRequest({
|
const result = await server.handleRequest(new StoccaTreRequest(method, route, requestBody));
|
||||||
route,
|
|
||||||
body: requestBody,
|
|
||||||
method,
|
|
||||||
});
|
|
||||||
if (result.error) {
|
if (result.error) {
|
||||||
await requestEvent.respondWith(Response.json({ ...body, error: `Internal server error: ${result.error.message}` }, { status: 500 }));
|
await requestEvent.respondWith(
|
||||||
|
Response.json({
|
||||||
|
...body,
|
||||||
|
error: `Internal server error: ${result.error.message}`,
|
||||||
|
}, { status: 500 }),
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
body.data = result.just;
|
body.data = result.just;
|
||||||
await requestEvent.respondWith(Response.json(body, { status: 200 }));
|
await requestEvent.respondWith(Response.json(body, { status: 200 }));
|
||||||
@@ -41,10 +41,11 @@ async function processRequest(server: StoccaTreServer, requestEvent: Deno.Reques
|
|||||||
|
|
||||||
const stoccaTreListener = Deno.listen({ port: config.port ?? 8080 });
|
const stoccaTreListener = Deno.listen({ port: config.port ?? 8080 });
|
||||||
|
|
||||||
console.log(`Stocca Tre Server is running. Access it at: http://localhost:${ config.port }/`);
|
console.log(`Stocca Tre Server is running. Access it at: http://localhost:${config.port}/`);
|
||||||
|
|
||||||
const database = await createNewDbConnection();
|
const database = createNewDbConnection();
|
||||||
const ingredientResource = new resources.IngredientResource(database);
|
const ingredientResource = new resources.IngredientResource(database);
|
||||||
|
const userResource = new resources.UserResource(database);
|
||||||
const stoccaTreServer: StoccaTreServer = {
|
const stoccaTreServer: StoccaTreServer = {
|
||||||
dbConnection: database,
|
dbConnection: database,
|
||||||
handleRequest: (request: StoccaTreRequest) => ingredientResource.handleRequest(request),
|
handleRequest: (request: StoccaTreRequest) => ingredientResource.handleRequest(request),
|
||||||
|
|||||||
@@ -1,10 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "stocca-tre-server",
|
|
||||||
"version": "0.0.1",
|
|
||||||
"scripts": {
|
|
||||||
"start": "deno run --allow-net --import-map=import_map.json main.ts"
|
|
||||||
},
|
|
||||||
"description": "Backend for StoccaTre",
|
|
||||||
"author": "Daniel Ledda",
|
|
||||||
"license": "MIT"
|
|
||||||
}
|
|
||||||
@@ -1,31 +1,42 @@
|
|||||||
import {StoccaTreDbConn, WithoutId} from "../../database.ts";
|
import { StoccaTreDbConn, WithoutId } from "../../database.ts";
|
||||||
import {IngredientModel} from "./IngredientModel.ts";
|
import { IngredientModel } from "./IngredientModel.ts";
|
||||||
import {Maybe} from "../../Maybe.ts";
|
import { Maybe } from "../../Maybe.ts";
|
||||||
|
|
||||||
export default class IngredientCollection {
|
export default class IngredientCollection {
|
||||||
private dbConnection: StoccaTreDbConn;
|
private db: StoccaTreDbConn;
|
||||||
private mapById: Map<number, IngredientModel> = new Map<number, IngredientModel>();
|
private mapById: Map<number, IngredientModel> = new Map<number, IngredientModel>();
|
||||||
private allGotten = false;
|
|
||||||
|
|
||||||
constructor(database: StoccaTreDbConn) {
|
constructor(database: StoccaTreDbConn) {
|
||||||
this.dbConnection = database;
|
this.db = database;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addIngredient(ingredient: WithoutId<IngredientModel>): Promise<any> {
|
async addIngredient(ingredient: WithoutId<IngredientModel>): Promise<Maybe<IngredientModel[]>> {
|
||||||
return await this.dbConnection.query<any>(
|
return await this.db.query(sql =>
|
||||||
`INSERT INTO ingredients (id, name, displayName, displayNameDE) VALUES (NULL, '${ingredient.name}', '${ingredient.displayName}', '${ingredient.displayNameDE}');`
|
sql<IngredientModel[]>`INSERT INTO ingredients ${ sql(ingredient) }`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
async getAllIngredients(): Promise<Maybe<IterableIterator<IngredientModel>>> {
|
async getById(id: number): Promise<Maybe<IngredientModel>> {
|
||||||
if (!this.allGotten) {
|
const found = this.mapById.get(id);
|
||||||
const result = await this.dbConnection.query<IngredientModel[]>("SELECT * FROM ingredients");
|
if (found) {
|
||||||
if (!result.error) {
|
return { just: found };
|
||||||
result.just.forEach((ingredient: IngredientModel) => this.mapById.set(ingredient.id, ingredient));
|
}
|
||||||
} else {
|
const result = await this.db.query(sql => sql<IngredientModel[]>`SELECT * FROM ingredients WHERE id is ${ id }`);
|
||||||
|
if (result.error) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
this.allGotten = true;
|
const ingredient = result.just[0];
|
||||||
|
this.mapById.set(ingredient.id, ingredient);
|
||||||
|
return { just: ingredient };
|
||||||
|
}
|
||||||
|
async getAllIngredients(): Promise<Maybe<IterableIterator<IngredientModel>>> {
|
||||||
|
const result: Maybe<IngredientModel[]> = await this.db.query(sql =>
|
||||||
|
sql<IngredientModel[]>`SELECT * FROM ingredients`
|
||||||
|
);
|
||||||
|
if (!result.error) {
|
||||||
|
result.just.forEach(ingredient => this.mapById.set(ingredient.id, ingredient));
|
||||||
|
} else {
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
return {
|
return {
|
||||||
just: this.mapById.values(),
|
just: this.mapById.values(),
|
||||||
|
|||||||
@@ -1,6 +1,12 @@
|
|||||||
export type IngredientModel = {
|
import { z } from "zod";
|
||||||
id: number,
|
|
||||||
name: string,
|
export const IngredientSchema = z.object({
|
||||||
displayName: string,
|
id: z.number(),
|
||||||
displayNameDE: string,
|
name: z.string().nonempty(),
|
||||||
};
|
displayName: z.string().nonempty(),
|
||||||
|
displayNameDE: z.string().nonempty(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const IngredientSchemaWithoutId = IngredientSchema.omit({ id: true });
|
||||||
|
|
||||||
|
export type IngredientModel = z.infer<typeof IngredientSchema>;
|
||||||
@@ -1,45 +1,48 @@
|
|||||||
import {StoccaTreDbConn} from "../../database.ts";
|
import {StoccaTreDbConn} from "../../database.ts";
|
||||||
import IngredientCollection from "./IngredientCollection.ts";
|
import IngredientCollection from "./IngredientCollection.ts";
|
||||||
import StoccaTreRequest from "../../StoccaTreRequest.ts";
|
import StoccaTreRequest, {RouteDefinition} from "../../StoccaTreRequest.ts";
|
||||||
import {Maybe} from "../../Maybe.ts";
|
import {Maybe} from "../../Maybe.ts";
|
||||||
import {JSONObject} from "../../JSON.ts";
|
import {JSONObject} from "../../JSON.ts";
|
||||||
import {IngredientModel} from "./IngredientModel.ts";
|
import {IngredientSchemaWithoutId} from "./IngredientModel.ts";
|
||||||
|
|
||||||
export default class IngredientResource {
|
export default class IngredientResource {
|
||||||
private dbConnection: StoccaTreDbConn;
|
private dbConnection: StoccaTreDbConn;
|
||||||
private collection: IngredientCollection;
|
private collection: IngredientCollection;
|
||||||
|
private routes: Readonly<Record<string, RouteDefinition>> = {
|
||||||
|
Add: {
|
||||||
|
pattern: /\/add/,
|
||||||
|
method: "POST"
|
||||||
|
},
|
||||||
|
GetAll: {
|
||||||
|
pattern: /\/all/,
|
||||||
|
method: "GET"
|
||||||
|
},
|
||||||
|
} as const;
|
||||||
|
|
||||||
constructor(dbConnection: StoccaTreDbConn) {
|
constructor(dbConnection: StoccaTreDbConn) {
|
||||||
this.dbConnection = dbConnection;
|
this.dbConnection = dbConnection;
|
||||||
this.collection = new IngredientCollection(dbConnection);
|
this.collection = new IngredientCollection(dbConnection);
|
||||||
}
|
}
|
||||||
|
|
||||||
static isIngredient(json: JSONObject): json is IngredientModel {
|
|
||||||
if (json) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
async handleRequest(request: StoccaTreRequest): Promise<Maybe<JSONObject>> {
|
async handleRequest(request: StoccaTreRequest): Promise<Maybe<JSONObject>> {
|
||||||
switch (request.route) {
|
if (request.match(this.routes.Add)) {
|
||||||
case "/add":
|
return await this.addIngredient(request);
|
||||||
if (request.method === "POST") {
|
|
||||||
const ingredient = JSON.parse(request.body ?? "");
|
|
||||||
return await this.collection.addIngredient(JSON.parse(request.body));
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
break;
|
if (request.match(this.routes.GetAll)) {
|
||||||
case "/all":
|
|
||||||
return await this.allIngredients(request);
|
return await this.allIngredients(request);
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
return { error: {message: "Invalid route" }};
|
return { error: { message: "Invalid route" }};
|
||||||
}
|
}
|
||||||
|
|
||||||
async allIngredients(request: StoccaTreRequest): Promise<Maybe<JSONObject>> {
|
private async addIngredient(request: StoccaTreRequest): Promise<Maybe<{ insertedId: number }>> {
|
||||||
|
const ingredient = IngredientSchemaWithoutId.safeParse(JSON.parse(request.body ?? "{}"));
|
||||||
|
if (!ingredient.success) {
|
||||||
|
return { error: new Error("Ingredient was malformed.") };
|
||||||
|
}
|
||||||
|
return await this.collection.addIngredient(ingredient.data);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async allIngredients(request: StoccaTreRequest): Promise<Maybe<JSONObject>> {
|
||||||
const getAllIngredientResult = await this.collection.getAllIngredients();
|
const getAllIngredientResult = await this.collection.getAllIngredients();
|
||||||
if (getAllIngredientResult.error) {
|
if (getAllIngredientResult.error) {
|
||||||
return getAllIngredientResult;
|
return getAllIngredientResult;
|
||||||
|
|||||||
@@ -1 +1,2 @@
|
|||||||
|
export * from "./user/main.ts";
|
||||||
export * from "./ingredient/main.ts";
|
export * from "./ingredient/main.ts";
|
||||||
|
|||||||
50
server/resources/user/UserCollection.ts
Normal file
50
server/resources/user/UserCollection.ts
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
import * as bcrypt from "bcrypt";
|
||||||
|
import { StoccaTreDbConn, WithoutId } from "../../database.ts";
|
||||||
|
import { UserModel } from "./UserModel.ts";
|
||||||
|
import { Maybe } from "../../Maybe.ts";
|
||||||
|
|
||||||
|
export default class UserCollection {
|
||||||
|
private db: StoccaTreDbConn;
|
||||||
|
private mapById: Map<number, UserModel> = new Map<number, UserModel>();
|
||||||
|
|
||||||
|
constructor(database: StoccaTreDbConn) {
|
||||||
|
this.db = database;
|
||||||
|
}
|
||||||
|
|
||||||
|
async addUser(user: WithoutId<UserModel>): Promise<Maybe<{ insertedId: number }>> {
|
||||||
|
let hash: string;
|
||||||
|
try {
|
||||||
|
hash = await bcrypt.hash(user.password);
|
||||||
|
} catch (e: unknown) {
|
||||||
|
const error = e as { message?: string };
|
||||||
|
if (typeof error.message === "string") {
|
||||||
|
return { error: { message: error.message }};
|
||||||
|
}
|
||||||
|
return { error: new Error("Failed to create user") };
|
||||||
|
}
|
||||||
|
user.password = hash;
|
||||||
|
const result: Maybe<UserModel[]> = await this.db.query(sql => sql`
|
||||||
|
INSERT INTO users ${ sql(user, "displayName") }
|
||||||
|
RETURNING *
|
||||||
|
`);
|
||||||
|
if (result.error) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
just: { insertedId: result.just[0]?.id ?? NaN }
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
async getAllUsers(): Promise<Maybe<UserModel[]>> {
|
||||||
|
const result = await this.db.query((sql) => sql<UserModel[]>`SELECT * FROM users`);
|
||||||
|
if (!result.error) {
|
||||||
|
result.just.forEach(user => this.mapById.set(user.id, user));
|
||||||
|
} else {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
just: Array.from(this.mapById.values()),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
12
server/resources/user/UserModel.ts
Normal file
12
server/resources/user/UserModel.ts
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
import { z } from "zod";
|
||||||
|
|
||||||
|
export const UserSchema = z.object({
|
||||||
|
id: z.number(),
|
||||||
|
displayName: z.string(),
|
||||||
|
email: z.string().email(),
|
||||||
|
password: z.string().min(256).max(256),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const UserSchemaWithoutId = UserSchema.omit({ id: true });
|
||||||
|
|
||||||
|
export type UserModel = z.infer<typeof UserSchema>;
|
||||||
56
server/resources/user/UserResource.ts
Normal file
56
server/resources/user/UserResource.ts
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
import {StoccaTreDbConn} from "../../database.ts";
|
||||||
|
import StoccaTreRequest, {RouteDefinition} from "../../StoccaTreRequest.ts";
|
||||||
|
import {Maybe} from "../../Maybe.ts";
|
||||||
|
import {JSONObject} from "../../JSON.ts";
|
||||||
|
import UserCollection from "./UserCollection.ts";
|
||||||
|
import {UserSchemaWithoutId} from "./UserModel.ts";
|
||||||
|
|
||||||
|
export default class UserResource {
|
||||||
|
private dbConnection: StoccaTreDbConn;
|
||||||
|
private collection: UserCollection;
|
||||||
|
private routes: Readonly<Record<string, RouteDefinition>> = {
|
||||||
|
Add: {
|
||||||
|
pattern: /\/add/,
|
||||||
|
method: "POST"
|
||||||
|
},
|
||||||
|
GetAll: {
|
||||||
|
pattern: /\/all/,
|
||||||
|
method: "GET"
|
||||||
|
},
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
constructor(dbConnection: StoccaTreDbConn) {
|
||||||
|
this.dbConnection = dbConnection;
|
||||||
|
this.collection = new UserCollection(dbConnection);
|
||||||
|
}
|
||||||
|
|
||||||
|
async handleRequest(request: StoccaTreRequest): Promise<Maybe<JSONObject>> {
|
||||||
|
if (request.match(this.routes.Add)) {
|
||||||
|
return await this.addUser(request);
|
||||||
|
}
|
||||||
|
if (request.match(this.routes.GetAll)) {
|
||||||
|
return await this.allUsers(request);
|
||||||
|
}
|
||||||
|
return { error: { message: "Invalid route" }};
|
||||||
|
}
|
||||||
|
|
||||||
|
private async addUser(request: StoccaTreRequest): Promise<Maybe<{ insertedId: number }>> {
|
||||||
|
const user = UserSchemaWithoutId.safeParse(JSON.parse(request.body ?? "{}"));
|
||||||
|
if (!user.success) {
|
||||||
|
return { error: new Error("Ingredient was malformed.") };
|
||||||
|
}
|
||||||
|
return await this.collection.addUser(user.data);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async allUsers(request: StoccaTreRequest): Promise<Maybe<JSONObject>> {
|
||||||
|
const allUsers = await this.collection.getAllUsers();
|
||||||
|
if (allUsers.error) {
|
||||||
|
return allUsers;
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
just: {
|
||||||
|
ingredients: Array.from(allUsers.just),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
3
server/resources/user/main.ts
Normal file
3
server/resources/user/main.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export * from "./UserModel.ts";
|
||||||
|
export * from "./UserCollection.ts";
|
||||||
|
export * from "./UserResource.ts";
|
||||||
1
server/start.sh
Normal file
1
server/start.sh
Normal file
@@ -0,0 +1 @@
|
|||||||
|
DB_USER=postgres && DB_PASS=postgres && deno run --allow-env --allow-net --import-map=import_map.json main.ts
|
||||||
Reference in New Issue
Block a user