feat: things are happening...
This commit is contained in:
@@ -1,27 +0,0 @@
|
|||||||
import { h, Rung } from "@djledda/ladder";
|
|
||||||
import StoccaTreLogo from "@/assets/stocca-tre-logo.svg";
|
|
||||||
import "./stocca-tre-root.scss";
|
|
||||||
|
|
||||||
export default class StoccaTreRoot extends Rung {
|
|
||||||
constructor() {
|
|
||||||
super({});
|
|
||||||
}
|
|
||||||
|
|
||||||
build(): Node {
|
|
||||||
return <div className={"stocca-tre-root"}>
|
|
||||||
<header className={"headstock"}>
|
|
||||||
<div>
|
|
||||||
<img className={"logo"} alt={"Stocca Tre Pizzera"} src={StoccaTreLogo} />
|
|
||||||
</div>
|
|
||||||
<ul className={"tabs"}>
|
|
||||||
<li>
|
|
||||||
Bestellen
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
Abstimmung Nächste Runde
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</header>
|
|
||||||
</div>;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
:root {
|
:root {
|
||||||
--bg-color: #e8e2e2;
|
--bg-color: #e8e2e2;
|
||||||
--red-deep: #aa0000;
|
--red-deep: #aa0000;
|
||||||
--red-shallow: #ec563f;
|
--red-shallow: #dc8383;
|
||||||
}
|
}
|
||||||
|
|
||||||
html, body {
|
html, body {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { bootstrap } from '@djledda/ladder';
|
import { bootstrap } from '@djledda/ladder';
|
||||||
import StoccaTreRoot from './StoccaTreRoot';
|
import StoccaTreRoot from './ui/StoccaTreRoot';
|
||||||
import './global.scss';
|
import './global.scss';
|
||||||
|
|
||||||
bootstrap(new StoccaTreRoot(), "root");
|
bootstrap(new StoccaTreRoot(), "root");
|
||||||
81
frontend/src/ui/StoccaTreRoot/index.tsx
Normal file
81
frontend/src/ui/StoccaTreRoot/index.tsx
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
import {Capsule, h, Rung } from "@djledda/ladder";
|
||||||
|
import StoccaTreLogo from "@/assets/stocca-tre-logo.svg";
|
||||||
|
import "./stocca-tre-root.scss";
|
||||||
|
import IngredientsPage from "../pages/IngredientsPage";
|
||||||
|
|
||||||
|
export default class StoccaTreRoot extends Rung {
|
||||||
|
private tabContainer = Capsule.new<HTMLUListElement | null>(null);
|
||||||
|
private tabs: HTMLLIElement[] = [];
|
||||||
|
private mainContent = Capsule.new<HTMLDivElement | null>(null);
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super({});
|
||||||
|
}
|
||||||
|
|
||||||
|
deselectTab(index: number) {
|
||||||
|
this.tabs[index]?.classList.remove("active");
|
||||||
|
}
|
||||||
|
|
||||||
|
selectTab(index: number) {
|
||||||
|
for (let i = 0; i < this.tabs.length; i++) {
|
||||||
|
if (i === index) {
|
||||||
|
this.tabs[i]?.classList.add("active");
|
||||||
|
} else {
|
||||||
|
this.deselectTab(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (index === 2) {
|
||||||
|
this.mainContent.val?.firstChild?.replaceWith(<div>{new IngredientsPage({})}</div>)
|
||||||
|
} else {
|
||||||
|
this.mainContent.val?.firstChild?.replaceWith(<div></div>);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Tab = (props: { label: string }) => {
|
||||||
|
const index = this.tabs.length;
|
||||||
|
const tab = <li
|
||||||
|
onclick={() => {
|
||||||
|
this.selectTab(index);
|
||||||
|
}}>
|
||||||
|
{ props.label }
|
||||||
|
</li> as HTMLLIElement;
|
||||||
|
this.tabs.push(tab);
|
||||||
|
return tab;
|
||||||
|
}
|
||||||
|
|
||||||
|
returnToHome(): void {
|
||||||
|
this.tabs.forEach((tab, i) => this.deselectTab(i));
|
||||||
|
this.mainContent.val?.firstChild?.replaceWith(<this.Home />)
|
||||||
|
}
|
||||||
|
|
||||||
|
Home = () => {
|
||||||
|
return <div>Navigiere auf eine Unterseite!</div>;
|
||||||
|
}
|
||||||
|
|
||||||
|
build(): Node {
|
||||||
|
this.tabs = [];
|
||||||
|
return <div className={"stocca-tre-root"}>
|
||||||
|
<nav className={"headstock"}>
|
||||||
|
<a>
|
||||||
|
<div>
|
||||||
|
<img className={"logo"} onclick={() => this.returnToHome()} alt={"Stocca Tre Pizzera"}
|
||||||
|
src={StoccaTreLogo}/>
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
<ul className={"tabs"} saveTo={this.tabContainer}>
|
||||||
|
<this.Tab label={"Bestellen"} />
|
||||||
|
<this.Tab label={"Abstimmung nächste Runde"} />
|
||||||
|
<this.Tab label={"Zutaten"} />
|
||||||
|
</ul>
|
||||||
|
</nav>
|
||||||
|
<header>
|
||||||
|
</header>
|
||||||
|
<section saveTo={this.mainContent}>
|
||||||
|
{/* Main Content */}
|
||||||
|
<this.Home />
|
||||||
|
</section>
|
||||||
|
<footer>
|
||||||
|
</footer>
|
||||||
|
</div>;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,10 +1,12 @@
|
|||||||
.stocca-tre-root {
|
.stocca-tre-root {
|
||||||
.headstock {
|
|
||||||
width: 1200px;
|
width: 1200px;
|
||||||
|
margin: auto;
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
.headstock {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
background-color: var(--bg-color);
|
background-color: var(--bg-color);
|
||||||
height: 100%;
|
height: 100%;
|
||||||
margin: auto;
|
|
||||||
|
|
||||||
.logo {
|
.logo {
|
||||||
margin: 20px auto auto auto;
|
margin: 20px auto auto auto;
|
||||||
@@ -23,6 +25,10 @@
|
|||||||
flex: 1;
|
flex: 1;
|
||||||
padding: 20px;
|
padding: 20px;
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
|
&:hover, &.active {
|
||||||
|
background-color: var(--red-shallow);
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
47
frontend/src/ui/pages/IngredientsPage/index.tsx
Normal file
47
frontend/src/ui/pages/IngredientsPage/index.tsx
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
import {h, frag, Rung, RungOptions, Capsule} from "@djledda/ladder";
|
||||||
|
|
||||||
|
export default class IngredientsPage extends Rung {
|
||||||
|
private ingredients: any[] = [];
|
||||||
|
private list = Capsule.new<HTMLDivElement | null>(null);
|
||||||
|
private ingredientInput = Capsule.new<HTMLInputElement | null>(null);
|
||||||
|
|
||||||
|
constructor(options: RungOptions) {
|
||||||
|
super({});
|
||||||
|
}
|
||||||
|
|
||||||
|
async addIngredient() {
|
||||||
|
await fetch("http://localhost:8080/ingredients/add", {
|
||||||
|
body: this.ingredientInput.val?.value ?? "{}",
|
||||||
|
method: "POST",
|
||||||
|
});
|
||||||
|
this.ingredients.push(JSON.parse(this.ingredientInput.val?.value ?? "{}"));
|
||||||
|
this.refreshList();
|
||||||
|
}
|
||||||
|
|
||||||
|
async getIngredients() {
|
||||||
|
const result = await (await fetch("http://localhost:8080/ingredients/all")).json();
|
||||||
|
this.ingredients = result.data.ingredients;
|
||||||
|
this.refreshList();
|
||||||
|
}
|
||||||
|
|
||||||
|
refreshList() {
|
||||||
|
this.list.val?.replaceWith(<this.List />);
|
||||||
|
}
|
||||||
|
|
||||||
|
List = (): HTMLUListElement => {
|
||||||
|
const thing = <div saveTo={this.list}>{...this.ingredients.map((ingredient) => {
|
||||||
|
return <div>{ingredient.displayNameDE}, hinzugefügt von {`${ingredient.addedBy}`}</div>;
|
||||||
|
})}</div> as HTMLUListElement;
|
||||||
|
return thing;
|
||||||
|
}
|
||||||
|
|
||||||
|
build(): Node {
|
||||||
|
const node = <>
|
||||||
|
<this.List saveTo={this.list} />
|
||||||
|
<input type={"text"} saveTo={this.ingredientInput} />
|
||||||
|
<button onclick={() => this.addIngredient()}>Submit!</button>
|
||||||
|
</>;
|
||||||
|
this.getIngredients();
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
export type Maybe<T> = Just<T> | { just?: never; error: { message: string } };
|
|
||||||
export type Just<T> = { just: T; error?: never };
|
|
||||||
35
server/Result.ts
Normal file
35
server/Result.ts
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
export class StoccaTreError {
|
||||||
|
public message: string;
|
||||||
|
private statusCode: number = 500;
|
||||||
|
|
||||||
|
constructor(message: string, status: number = 500) {
|
||||||
|
this.message = message;
|
||||||
|
this.statusCode = 500;
|
||||||
|
this.status = status;
|
||||||
|
}
|
||||||
|
|
||||||
|
qualified(qualification: string): this {
|
||||||
|
const insertionPoint = qualification.indexOf("$err");
|
||||||
|
this.message = insertionPoint !== -1
|
||||||
|
? `${qualification.slice(0, insertionPoint)}${this.message}${qualification.slice(insertionPoint + 4)}`
|
||||||
|
: `${qualification}${this.message}`;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
get status(): number {
|
||||||
|
return this.statusCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
set status(code: number) {
|
||||||
|
if (code >= 100 && code < 600) {
|
||||||
|
this.statusCode = code;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
withStatus(code: number): this {
|
||||||
|
this.status = code;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export type Result<T> = [error: undefined, just: T] | [error: StoccaTreError, just?: undefined];
|
||||||
@@ -1,22 +1,25 @@
|
|||||||
|
import Http = Deno.errors.Http;
|
||||||
|
import {JSONObject} from "./JSON.ts";
|
||||||
|
|
||||||
export type HttpMethod = "POST" | "GET" | "PUT" | "DELETE";
|
export type HttpMethod = "POST" | "GET" | "PUT" | "DELETE";
|
||||||
|
|
||||||
export type RouteDefinition = {
|
export type RouteDefinition = {
|
||||||
pattern: RegExp;
|
pattern: RegExp;
|
||||||
method: HttpMethod;
|
method?: HttpMethod;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default class StoccaTreRequest {
|
export default class StoccaTreRequest {
|
||||||
constructor(
|
constructor(
|
||||||
public method: HttpMethod,
|
public method: HttpMethod,
|
||||||
public route: string,
|
public route: string,
|
||||||
public body: string | null,
|
public body: JSONObject | null,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
match(route: RouteDefinition): RegExpExecArray | null {
|
match(route: RouteDefinition): RegExpExecArray | false {
|
||||||
const patternResult = route.pattern.exec(this.route);
|
const patternResult = route.pattern.exec(this.route);
|
||||||
if (route.method !== this.method) {
|
if (route.method && route.method !== this.method) {
|
||||||
return null;
|
return false;
|
||||||
}
|
}
|
||||||
return patternResult;
|
return patternResult ?? false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import StoccaTreRequest from "./StoccaTreRequest.ts";
|
import StoccaTreRequest from "./StoccaTreRequest.ts";
|
||||||
import { Maybe } from "./Maybe.ts";
|
import { Result } from "./Result.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<Result<JSONObject> | null>;
|
||||||
}
|
}
|
||||||
|
|||||||
35
server/StoccaTreServer.ts
Normal file
35
server/StoccaTreServer.ts
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
import StoccaTreRequestHandler from "./StoccaTreRequestHandler.ts";
|
||||||
|
import StoccaTreRequest, { RouteDefinition } from "./StoccaTreRequest.ts";
|
||||||
|
import { Result, StoccaTreError } from "./Result.ts";
|
||||||
|
import { JSONObject } from "./JSON.ts";
|
||||||
|
import { StoccaTreDbConn } from "./database.ts";
|
||||||
|
|
||||||
|
export default class StoccaTreServer implements StoccaTreRequestHandler {
|
||||||
|
private db: StoccaTreDbConn;
|
||||||
|
private routes: {
|
||||||
|
routeDef: RouteDefinition;
|
||||||
|
handler: StoccaTreRequestHandler;
|
||||||
|
}[] = [];
|
||||||
|
|
||||||
|
constructor(dbConnection: StoccaTreDbConn) {
|
||||||
|
this.db = dbConnection;
|
||||||
|
}
|
||||||
|
|
||||||
|
addResource(route: RouteDefinition, handler: StoccaTreRequestHandler) {
|
||||||
|
this.routes.push({ routeDef: route, handler });
|
||||||
|
}
|
||||||
|
|
||||||
|
async handleRequest(request: StoccaTreRequest): Promise<Result<JSONObject>> {
|
||||||
|
let result: Result<JSONObject> | null = null;
|
||||||
|
for (const { routeDef, handler } of this.routes) {
|
||||||
|
if (request.match(routeDef)) {
|
||||||
|
result = await handler.handleRequest(request);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!result) {
|
||||||
|
return [new StoccaTreError(`Invalid route: ${request.route} with method ${request.method}.`, 400)];
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,13 +1,15 @@
|
|||||||
const config = {
|
const config = {
|
||||||
username: Deno.env.get("DB_USER") ?? "postgres",
|
dbUsername: Deno.env.get("DB_USER") ?? "postgres",
|
||||||
hostname: Deno.env.get("DB_HOST") ?? "localhost",
|
dbHostname: Deno.env.get("DB_HOST") ?? "localhost",
|
||||||
password: Deno.env.get("DB_PW") ?? "",
|
dbPassword: Deno.env.get("DB_PASS") ?? "",
|
||||||
dbPort: Number(Deno.env.get("DB_PORT") ?? 5432),
|
dbPort: Number(Deno.env.get("DB_PORT") ?? 5432),
|
||||||
|
hostname: Deno.env.get("HOST") ?? "localhost",
|
||||||
port: Number(Deno.env.get("PORT") ?? 8080),
|
port: Number(Deno.env.get("PORT") ?? 8080),
|
||||||
};
|
};
|
||||||
|
|
||||||
console.log(`ENV:
|
console.log(`ENV:
|
||||||
db username: ${config.username}
|
db username: ${config.dbUsername}
|
||||||
|
db hostname: ${config.dbHostname}
|
||||||
db pass: ******
|
db pass: ******
|
||||||
db port: ${config.dbPort}
|
db port: ${config.dbPort}
|
||||||
server port: ${config.port}
|
server port: ${config.port}
|
||||||
|
|||||||
@@ -1,42 +1,73 @@
|
|||||||
import { Client } from "postgres";
|
import { Client } from "postgres";
|
||||||
import config from "./config.ts";
|
import config from "./config.ts";
|
||||||
import {Maybe} from "./Maybe.ts";
|
import { JSONObject } from "./JSON.ts";
|
||||||
import {JSONObject} from "./JSON.ts";
|
import { Result, StoccaTreError } from "./Result.ts";
|
||||||
|
|
||||||
|
type Interpolable = number | string | bigint | null | boolean;
|
||||||
|
type UnionToIntersection<Union> =
|
||||||
|
(Union extends any ? (x: Union) => any : never) extends (x: infer Intersection) => any
|
||||||
|
? Intersection
|
||||||
|
: never;
|
||||||
|
type SQLWithArg<Arg> = Arg extends string ? `${string}$${Arg}${string}` : never;
|
||||||
|
type SQLWithArgs<Args extends Record<string, Interpolable>> = UnionToIntersection<SQLWithArg<keyof Args>>;
|
||||||
|
type ArgNamesInSQL<SQL extends string> =
|
||||||
|
SQL extends `${string}$${infer Arg1} ${infer SQLAfterArg1}`
|
||||||
|
? Arg1 extends `${infer Arg1NoTrailing}${";" | "," | "'" | "\"" | "(" | ")"}`
|
||||||
|
? Arg1NoTrailing | ArgNamesInSQL<SQLAfterArg1>
|
||||||
|
: Arg1 | ArgNamesInSQL<SQLAfterArg1>
|
||||||
|
: SQL extends `${string}$${infer ArgN}`
|
||||||
|
? ArgN extends `${infer ArgNNoTrailing}${";" | "," | "'" | "\"" | "(" | ")"}`
|
||||||
|
? ArgNNoTrailing
|
||||||
|
: ArgN
|
||||||
|
: never;
|
||||||
|
|
||||||
|
type ParametrisedSQL = `${string}$${string}`;
|
||||||
|
type IsPlainSQL<T> = T extends ParametrisedSQL ? never : T;
|
||||||
|
type ArgsForSQL<SQL extends string> = SQL extends ParametrisedSQL
|
||||||
|
? Record<ArgNamesInSQL<SQL>, Interpolable>
|
||||||
|
: JSONObject | undefined;
|
||||||
|
|
||||||
|
// Helper for casting queries
|
||||||
|
export type Q<T extends JSONObject> = Result<T[]>;
|
||||||
|
|
||||||
export type WithoutId<T> = Omit<T, "id">;
|
export type WithoutId<T> = Omit<T, "id">;
|
||||||
|
|
||||||
export interface StoccaTreDbConn {
|
export interface StoccaTreDbConn {
|
||||||
query<T extends JSONObject | JSONObject[]>(query: string): Promise<Maybe<T>>,
|
query<Q extends string>(query: Q, ...args: Q extends IsPlainSQL<Q> ? [(JSONObject | undefined)?] : [ArgsForSQL<Q>]): Promise<Result<JSONObject[]>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function createNewDbConnection(): StoccaTreDbConn {
|
export default async function createNewDbConnection(): Promise<Result<StoccaTreDbConn>> {
|
||||||
const postgresClient = new Client({
|
let postgresClient: Client;
|
||||||
hostname: config.hostname,
|
try {
|
||||||
password: config.password,
|
postgresClient = new Client({
|
||||||
user: config.username,
|
user: config.dbUsername,
|
||||||
database: "stocca_tre",
|
database: "stocca_tre",
|
||||||
port: config.port,
|
hostname: config.hostname,
|
||||||
|
port: config.dbPort,
|
||||||
|
password: config.dbPassword,
|
||||||
|
tls: {
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
});
|
});
|
||||||
await postgresClient.connect();
|
await postgresClient.connect();
|
||||||
return {
|
|
||||||
async queryMany<T extends JSONObject[]>(query: string): Promise<Maybe<{ rows: T, count: number }>> {
|
|
||||||
try {
|
|
||||||
const result = await postgresClient.queryArray<T>(query);
|
|
||||||
return {
|
|
||||||
just: {
|
|
||||||
rows: result.rows,
|
|
||||||
count: result.rowCount ?? NaN,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
} catch (e: unknown) {
|
} catch (e: unknown) {
|
||||||
|
const error = e as { message?: string };
|
||||||
|
if (error.message) {
|
||||||
|
return [new StoccaTreError(error.message).qualified("Error connecting to database: ")];
|
||||||
|
}
|
||||||
|
return [new StoccaTreError("Error connecting to database.")];
|
||||||
|
}
|
||||||
|
return [,
|
||||||
|
{
|
||||||
|
async query<Q extends string>(query: Q, ...args: Q extends IsPlainSQL<Q> ? [(JSONObject | undefined)?] : [ArgsForSQL<Q>]): Promise<Result<JSONObject[]>> {
|
||||||
|
try {
|
||||||
|
const result = <{rows: JSONObject[]}> await postgresClient.queryObject(query, args[0]);
|
||||||
|
return [, result.rows];
|
||||||
|
} catch (e: unknown) {
|
||||||
|
const error = e as { message?: string };
|
||||||
|
return [new StoccaTreError(error.message ?? "Internal database error.")];
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
queryOne<T extends JSONObject>(query: string): Promise<Maybe<T>> {
|
},
|
||||||
try {
|
];
|
||||||
const result = await postgresClient.queryObject<T>(query);
|
|
||||||
return { just: result };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,8 +2,9 @@ import createNewDbConnection, { StoccaTreDbConn } from "./database.ts";
|
|||||||
import * as resources from "./resources/main.ts";
|
import * as resources from "./resources/main.ts";
|
||||||
import config from "./config.ts";
|
import config from "./config.ts";
|
||||||
import StoccaTreRequest, { HttpMethod } from "./StoccaTreRequest.ts";
|
import StoccaTreRequest, { HttpMethod } from "./StoccaTreRequest.ts";
|
||||||
import StoccaTreRequestHandler from "./StoccaTreRequestHandler.ts";
|
|
||||||
import { JSONObject } from "./JSON.ts";
|
import { JSONObject } from "./JSON.ts";
|
||||||
|
import StoccaTreServer from "./StoccaTreServer.ts";
|
||||||
|
import {Result, StoccaTreError} from "./Result.ts";
|
||||||
|
|
||||||
type StoccaTreApiBody = {
|
type StoccaTreApiBody = {
|
||||||
data: JSONObject;
|
data: JSONObject;
|
||||||
@@ -16,44 +17,68 @@ function newStoccaTreApiBody(): StoccaTreApiBody {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
type StoccaTreServer = StoccaTreRequestHandler & {
|
async function getJSONBody(request: Request): Promise<Result<JSONObject | null>> {
|
||||||
dbConnection: StoccaTreDbConn;
|
let json;
|
||||||
};
|
try {
|
||||||
|
if (request.body) {
|
||||||
|
json = await request.json();
|
||||||
|
} else {
|
||||||
|
json = null;
|
||||||
|
}
|
||||||
|
} catch (e: unknown) {
|
||||||
|
return [new StoccaTreError((e as Error).message ?? "Body was invalid JSON.", 400)];
|
||||||
|
}
|
||||||
|
return [, json];
|
||||||
|
}
|
||||||
|
|
||||||
async function processRequest(server: StoccaTreServer, requestEvent: Deno.RequestEvent) {
|
async function respondWithError(requestEvent: Deno.RequestEvent, error: StoccaTreError) {
|
||||||
const route = /^https?:\/\/[^\/]*(\/.*)$/.exec(requestEvent.request.url)?.[1] ?? "/";
|
const body = newStoccaTreApiBody();
|
||||||
const requestBody = (await (await requestEvent.request.blob()).text()) ?? null;
|
|
||||||
const method: HttpMethod = requestEvent.request.method as HttpMethod;
|
|
||||||
const body: StoccaTreApiBody = newStoccaTreApiBody();
|
|
||||||
const result = await server.handleRequest(new StoccaTreRequest(method, route, requestBody));
|
|
||||||
if (result.error) {
|
|
||||||
await requestEvent.respondWith(
|
await requestEvent.respondWith(
|
||||||
Response.json({
|
Response.json({
|
||||||
...body,
|
...body,
|
||||||
error: `Internal server error: ${result.error.message}`,
|
status: error.status,
|
||||||
}, { status: 500 }),
|
error: error.qualified("Error: ").message,
|
||||||
|
}, { status: error.status }),
|
||||||
);
|
);
|
||||||
} else {
|
|
||||||
body.data = result.just;
|
|
||||||
await requestEvent.respondWith(Response.json(body, { status: 200 }));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const stoccaTreListener = Deno.listen({ port: config.port ?? 8080 });
|
async function processRequest(
|
||||||
|
server: StoccaTreServer,
|
||||||
|
requestEvent: Deno.RequestEvent,
|
||||||
|
) {
|
||||||
|
const [bodyError, requestBody] = await getJSONBody(requestEvent.request);
|
||||||
|
if (bodyError) {
|
||||||
|
await respondWithError(requestEvent, bodyError);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const route = /^https?:\/\/[^\/]*(\/.*)$/.exec(requestEvent.request.url)?.[1] ?? "/";
|
||||||
|
const method: HttpMethod = requestEvent.request.method as HttpMethod;
|
||||||
|
const body: StoccaTreApiBody = newStoccaTreApiBody();
|
||||||
|
const [error, result] = await server.handleRequest(new StoccaTreRequest(method, route, requestBody));
|
||||||
|
if (error) {
|
||||||
|
await respondWithError(requestEvent, error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
body.data = result;
|
||||||
|
await requestEvent.respondWith(Response.json(body, { status: 200 }));
|
||||||
|
}
|
||||||
|
|
||||||
console.log(`Stocca Tre Server is running. Access it at: http://localhost:${config.port}/`);
|
|
||||||
|
|
||||||
const database = createNewDbConnection();
|
const [error, database] = await createNewDbConnection();
|
||||||
const ingredientResource = new resources.IngredientResource(database);
|
|
||||||
const userResource = new resources.UserResource(database);
|
|
||||||
const stoccaTreServer: StoccaTreServer = {
|
|
||||||
dbConnection: database,
|
|
||||||
handleRequest: (request: StoccaTreRequest) => ingredientResource.handleRequest(request),
|
|
||||||
};
|
|
||||||
|
|
||||||
for await (const conn of stoccaTreListener) {
|
if (error) {
|
||||||
|
console.log(error.qualified("Failed to create the database: ").message);
|
||||||
|
} else {
|
||||||
|
const server = new StoccaTreServer(database);
|
||||||
|
server.addResource({ pattern: /ingredient/ }, new resources.IngredientResource(database));
|
||||||
|
server.addResource({ pattern: /user/ }, new resources.UserResource(database));
|
||||||
|
const stoccaTreListener = Deno.listen({ port: config.port ?? 8080 });
|
||||||
|
console.log(`Stocca Tre Server is running. Access it at: http://${ config.hostname }:${ config.port }/`);
|
||||||
|
for await (const conn of stoccaTreListener) {
|
||||||
const httpConn = Deno.serveHttp(conn);
|
const httpConn = Deno.serveHttp(conn);
|
||||||
for await (const requestEvent of httpConn) {
|
for await (const requestEvent of httpConn) {
|
||||||
await processRequest(stoccaTreServer, requestEvent);
|
await processRequest(server, requestEvent);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
import { StoccaTreDbConn, WithoutId } from "../../database.ts";
|
import { Q, StoccaTreDbConn, WithoutId } from "../../database.ts";
|
||||||
import { IngredientModel } from "./IngredientModel.ts";
|
import { IngredientModel } from "./IngredientModel.ts";
|
||||||
import { Maybe } from "../../Maybe.ts";
|
import { Result, StoccaTreError } from "../../Result.ts";
|
||||||
|
|
||||||
|
const TABLE_NAME = "main.ingredients";
|
||||||
|
|
||||||
export default class IngredientCollection {
|
export default class IngredientCollection {
|
||||||
private db: StoccaTreDbConn;
|
private db: StoccaTreDbConn;
|
||||||
@@ -10,36 +12,38 @@ export default class IngredientCollection {
|
|||||||
this.db = database;
|
this.db = database;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addIngredient(ingredient: WithoutId<IngredientModel>): Promise<Maybe<IngredientModel[]>> {
|
async addIngredient(ingredient: WithoutId<IngredientModel>): Promise<Result<IngredientModel>> {
|
||||||
return await this.db.query(sql =>
|
const [error, result] = <Q<IngredientModel>> await this.db.query(`INSERT INTO ${ TABLE_NAME } VALUES (DEFAULT, $name, $displayName, $displayNameDE) RETURNING *;`, ingredient);
|
||||||
sql<IngredientModel[]>`INSERT INTO ingredients ${ sql(ingredient) }`
|
if (error) {
|
||||||
);
|
return [error];
|
||||||
|
}
|
||||||
|
if (result.length > 0) {
|
||||||
|
return [, result[0]];
|
||||||
|
}
|
||||||
|
return [new StoccaTreError("Ingredient wasn't inserted!")];
|
||||||
}
|
}
|
||||||
|
|
||||||
async getById(id: number): Promise<Maybe<IngredientModel>> {
|
async getById(id: number): Promise<Result<IngredientModel>> {
|
||||||
const found = this.mapById.get(id);
|
const found = this.mapById.get(id);
|
||||||
if (found) {
|
if (found) {
|
||||||
return { just: found };
|
return [, found];
|
||||||
}
|
}
|
||||||
const result = await this.db.query(sql => sql<IngredientModel[]>`SELECT * FROM ingredients WHERE id is ${ id }`);
|
const [error, result] = <Q<IngredientModel>> await this.db.query(`SELECT * FROM ${ TABLE_NAME } WHERE id is $id`, { id });
|
||||||
if (result.error) {
|
if (error) {
|
||||||
return result;
|
return [error];
|
||||||
}
|
}
|
||||||
const ingredient = result.just[0];
|
const ingredient = result[0];
|
||||||
this.mapById.set(ingredient.id, ingredient);
|
this.mapById.set(ingredient.id, ingredient);
|
||||||
return { just: ingredient };
|
return [, ingredient];
|
||||||
}
|
}
|
||||||
async getAllIngredients(): Promise<Maybe<IterableIterator<IngredientModel>>> {
|
|
||||||
const result: Maybe<IngredientModel[]> = await this.db.query(sql =>
|
async getAllIngredients(): Promise<Result<IterableIterator<IngredientModel>>> {
|
||||||
sql<IngredientModel[]>`SELECT * FROM ingredients`
|
const [error, result] = <Q<IngredientModel>> await this.db.query(`SELECT * FROM ${ TABLE_NAME }`);
|
||||||
);
|
if (!error) {
|
||||||
if (!result.error) {
|
result.forEach((ingredient: IngredientModel) => this.mapById.set(ingredient.id, ingredient));
|
||||||
result.just.forEach(ingredient => this.mapById.set(ingredient.id, ingredient));
|
|
||||||
} else {
|
} else {
|
||||||
return result;
|
return [error];
|
||||||
}
|
}
|
||||||
return {
|
return [, this.mapById.values()];
|
||||||
just: this.mapById.values(),
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,4 +9,6 @@ export const IngredientSchema = z.object({
|
|||||||
|
|
||||||
export const IngredientSchemaWithoutId = IngredientSchema.omit({ id: true });
|
export const IngredientSchemaWithoutId = IngredientSchema.omit({ id: true });
|
||||||
|
|
||||||
|
|
||||||
export type IngredientModel = z.infer<typeof IngredientSchema>;
|
export type IngredientModel = z.infer<typeof IngredientSchema>;
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import {StoccaTreDbConn} from "../../database.ts";
|
import { StoccaTreDbConn } from "../../database.ts";
|
||||||
import IngredientCollection from "./IngredientCollection.ts";
|
import IngredientCollection from "./IngredientCollection.ts";
|
||||||
import StoccaTreRequest, {RouteDefinition} from "../../StoccaTreRequest.ts";
|
import StoccaTreRequest, { RouteDefinition } from "../../StoccaTreRequest.ts";
|
||||||
import {Maybe} from "../../Maybe.ts";
|
import {Result, StoccaTreError} from "../../Result.ts";
|
||||||
import {JSONObject} from "../../JSON.ts";
|
import { JSONObject } from "../../JSON.ts";
|
||||||
import {IngredientSchemaWithoutId} from "./IngredientModel.ts";
|
import { IngredientModel, IngredientSchemaWithoutId } from "./IngredientModel.ts";
|
||||||
|
|
||||||
export default class IngredientResource {
|
export default class IngredientResource {
|
||||||
private dbConnection: StoccaTreDbConn;
|
private dbConnection: StoccaTreDbConn;
|
||||||
@@ -11,11 +11,11 @@ export default class IngredientResource {
|
|||||||
private routes: Readonly<Record<string, RouteDefinition>> = {
|
private routes: Readonly<Record<string, RouteDefinition>> = {
|
||||||
Add: {
|
Add: {
|
||||||
pattern: /\/add/,
|
pattern: /\/add/,
|
||||||
method: "POST"
|
method: "POST",
|
||||||
},
|
},
|
||||||
GetAll: {
|
GetAll: {
|
||||||
pattern: /\/all/,
|
pattern: /\/all/,
|
||||||
method: "GET"
|
method: "GET",
|
||||||
},
|
},
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
@@ -24,33 +24,40 @@ export default class IngredientResource {
|
|||||||
this.collection = new IngredientCollection(dbConnection);
|
this.collection = new IngredientCollection(dbConnection);
|
||||||
}
|
}
|
||||||
|
|
||||||
async handleRequest(request: StoccaTreRequest): Promise<Maybe<JSONObject>> {
|
async handleRequest(request: StoccaTreRequest): Promise<Result<JSONObject> | null> {
|
||||||
|
let result;
|
||||||
if (request.match(this.routes.Add)) {
|
if (request.match(this.routes.Add)) {
|
||||||
return await this.addIngredient(request);
|
result = await this.addIngredient(request);
|
||||||
}
|
}
|
||||||
if (request.match(this.routes.GetAll)) {
|
if (request.match(this.routes.GetAll)) {
|
||||||
return await this.allIngredients(request);
|
result = await this.allIngredients(request);
|
||||||
}
|
}
|
||||||
return { error: { message: "Invalid route" }};
|
if (result) {
|
||||||
|
const [error] = result;
|
||||||
|
if (error) {
|
||||||
|
return [error.qualified("Could not fulfill ingredient request: ")];
|
||||||
|
} else {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async addIngredient(request: StoccaTreRequest): Promise<Maybe<{ insertedId: number }>> {
|
private async addIngredient(request: StoccaTreRequest): Promise<Result<{ id: number }>> {
|
||||||
const ingredient = IngredientSchemaWithoutId.safeParse(JSON.parse(request.body ?? "{}"));
|
const ingredient = IngredientSchemaWithoutId.safeParse(request.body);
|
||||||
if (!ingredient.success) {
|
if (!ingredient.success) {
|
||||||
return { error: new Error("Ingredient was malformed.") };
|
return [new StoccaTreError("Ingredient definition was malformed.", 400)];
|
||||||
}
|
}
|
||||||
return await this.collection.addIngredient(ingredient.data);
|
return await this.collection.addIngredient(ingredient.data);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async allIngredients(request: StoccaTreRequest): Promise<Maybe<JSONObject>> {
|
private async allIngredients(request: StoccaTreRequest): Promise<Result<JSONObject>> {
|
||||||
const getAllIngredientResult = await this.collection.getAllIngredients();
|
const [error, ingredients] = await this.collection.getAllIngredients();
|
||||||
if (getAllIngredientResult.error) {
|
if (error) {
|
||||||
return getAllIngredientResult;
|
return [error];
|
||||||
}
|
}
|
||||||
return {
|
return [, {
|
||||||
just: {
|
ingredients: Array.from(ingredients),
|
||||||
ingredients: Array.from(getAllIngredientResult.just),
|
}];
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,7 +1,9 @@
|
|||||||
import * as bcrypt from "bcrypt";
|
import * as bcrypt from "bcrypt";
|
||||||
import { StoccaTreDbConn, WithoutId } from "../../database.ts";
|
import {Q, StoccaTreDbConn, WithoutId} from "../../database.ts";
|
||||||
import { UserModel } from "./UserModel.ts";
|
import { UserModel } from "./UserModel.ts";
|
||||||
import { Maybe } from "../../Maybe.ts";
|
import {Result, StoccaTreError} from "../../Result.ts";
|
||||||
|
|
||||||
|
const TABLE_NAME = "main.users";
|
||||||
|
|
||||||
export default class UserCollection {
|
export default class UserCollection {
|
||||||
private db: StoccaTreDbConn;
|
private db: StoccaTreDbConn;
|
||||||
@@ -11,40 +13,34 @@ export default class UserCollection {
|
|||||||
this.db = database;
|
this.db = database;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addUser(user: WithoutId<UserModel>): Promise<Maybe<{ insertedId: number }>> {
|
async addUser(user: WithoutId<UserModel>): Promise<Result<{ id: number }>> {
|
||||||
let hash: string;
|
let hash: string;
|
||||||
try {
|
try {
|
||||||
hash = await bcrypt.hash(user.password);
|
hash = await bcrypt.hash(user.password);
|
||||||
} catch (e: unknown) {
|
} catch (e: unknown) {
|
||||||
const error = e as { message?: string };
|
const error = e as { message?: string };
|
||||||
if (typeof error.message === "string") {
|
if (typeof error.message === "string") {
|
||||||
return { error: { message: error.message }};
|
return [new StoccaTreError(error.message)];
|
||||||
}
|
}
|
||||||
return { error: new Error("Failed to create user") };
|
return [new StoccaTreError("Failed to create user")];
|
||||||
}
|
}
|
||||||
user.password = hash;
|
user.password = hash;
|
||||||
const result: Maybe<UserModel[]> = await this.db.query(sql => sql`
|
const [error, users] = <Q<UserModel>> await this.db.query(`INSERT INTO ${TABLE_NAME} ($displayName, $password) RETURNING *`, user);
|
||||||
INSERT INTO users ${ sql(user, "displayName") }
|
if (error) {
|
||||||
RETURNING *
|
return [error];
|
||||||
`);
|
|
||||||
if (result.error) {
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
return {
|
if (users.length === 0) {
|
||||||
just: { insertedId: result.just[0]?.id ?? NaN }
|
return [new StoccaTreError("Failed to insert user.")];
|
||||||
|
}
|
||||||
|
return [, users[0]];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getAllUsers(): Promise<Result<UserModel[]>> {
|
||||||
|
const [error, users] = <Q<UserModel>> await this.db.query(`SELECT * FROM ${TABLE_NAME}`);
|
||||||
|
if (error) {
|
||||||
|
return [error];
|
||||||
}
|
}
|
||||||
|
users.forEach((user) => this.mapById.set(user.id, user));
|
||||||
async getAllUsers(): Promise<Maybe<UserModel[]>> {
|
return [, Array.from(this.mapById.values())];
|
||||||
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()),
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import {StoccaTreDbConn} from "../../database.ts";
|
import { StoccaTreDbConn } from "../../database.ts";
|
||||||
import StoccaTreRequest, {RouteDefinition} from "../../StoccaTreRequest.ts";
|
import StoccaTreRequest, { RouteDefinition } from "../../StoccaTreRequest.ts";
|
||||||
import {Maybe} from "../../Maybe.ts";
|
import {Result, StoccaTreError} from "../../Result.ts";
|
||||||
import {JSONObject} from "../../JSON.ts";
|
import { JSONObject } from "../../JSON.ts";
|
||||||
import UserCollection from "./UserCollection.ts";
|
import UserCollection from "./UserCollection.ts";
|
||||||
import {UserSchemaWithoutId} from "./UserModel.ts";
|
import { UserSchemaWithoutId } from "./UserModel.ts";
|
||||||
|
|
||||||
export default class UserResource {
|
export default class UserResource {
|
||||||
private dbConnection: StoccaTreDbConn;
|
private dbConnection: StoccaTreDbConn;
|
||||||
@@ -11,11 +11,11 @@ export default class UserResource {
|
|||||||
private routes: Readonly<Record<string, RouteDefinition>> = {
|
private routes: Readonly<Record<string, RouteDefinition>> = {
|
||||||
Add: {
|
Add: {
|
||||||
pattern: /\/add/,
|
pattern: /\/add/,
|
||||||
method: "POST"
|
method: "POST",
|
||||||
},
|
},
|
||||||
GetAll: {
|
GetAll: {
|
||||||
pattern: /\/all/,
|
pattern: /\/all/,
|
||||||
method: "GET"
|
method: "GET",
|
||||||
},
|
},
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
@@ -24,33 +24,29 @@ export default class UserResource {
|
|||||||
this.collection = new UserCollection(dbConnection);
|
this.collection = new UserCollection(dbConnection);
|
||||||
}
|
}
|
||||||
|
|
||||||
async handleRequest(request: StoccaTreRequest): Promise<Maybe<JSONObject>> {
|
async handleRequest(request: StoccaTreRequest): Promise<Result<JSONObject> | null> {
|
||||||
if (request.match(this.routes.Add)) {
|
if (request.match(this.routes.Add)) {
|
||||||
return await this.addUser(request);
|
return await this.addUser(request);
|
||||||
}
|
}
|
||||||
if (request.match(this.routes.GetAll)) {
|
if (request.match(this.routes.GetAll)) {
|
||||||
return await this.allUsers(request);
|
return await this.allUsers(request);
|
||||||
}
|
}
|
||||||
return { error: { message: "Invalid route" }};
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async addUser(request: StoccaTreRequest): Promise<Maybe<{ insertedId: number }>> {
|
private async addUser(request: StoccaTreRequest): Promise<Result<{ id: number }>> {
|
||||||
const user = UserSchemaWithoutId.safeParse(JSON.parse(request.body ?? "{}"));
|
const user = UserSchemaWithoutId.safeParse(request.body);
|
||||||
if (!user.success) {
|
if (!user.success) {
|
||||||
return { error: new Error("Ingredient was malformed.") };
|
return [new StoccaTreError("User definition was malformed.", 400)];
|
||||||
}
|
}
|
||||||
return await this.collection.addUser(user.data);
|
return await this.collection.addUser(user.data);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async allUsers(request: StoccaTreRequest): Promise<Maybe<JSONObject>> {
|
private async allUsers(request: StoccaTreRequest): Promise<Result<JSONObject>> {
|
||||||
const allUsers = await this.collection.getAllUsers();
|
const [error, result] = await this.collection.getAllUsers();
|
||||||
if (allUsers.error) {
|
if (error) {
|
||||||
return allUsers;
|
return [error];
|
||||||
}
|
}
|
||||||
return {
|
return [, { users: result }];
|
||||||
just: {
|
|
||||||
ingredients: Array.from(allUsers.just),
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,3 +1,3 @@
|
|||||||
export * from "./UserModel.ts";
|
export * from "./UserModel.ts";
|
||||||
export * from "./UserCollection.ts";
|
export { default as UserCollection } from "./UserCollection.ts";
|
||||||
export * from "./UserResource.ts";
|
export { default as UserResource } from "./UserResource.ts";
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
DB_USER=postgres && DB_PASS=postgres && deno run --allow-env --allow-net --import-map=import_map.json main.ts
|
DB_USER=postgres DB_PASS=postgres deno run --allow-read --allow-env --allow-net --import-map=import_map.json main.ts
|
||||||
|
|||||||
Reference in New Issue
Block a user