From 747097af03e422c72c2fe0065637e59a1685d219 Mon Sep 17 00:00:00 2001 From: kartofen Date: Sun, 23 Oct 2022 21:34:48 +0300 Subject: nearly completed, forgot to make a repo --- src/database/postgres.ts | 74 +++++++++++++++++ src/files/SwaggerUI.yaml | 206 +++++++++++++++++++++++++++++++++++++++++++++++ src/models/Thread.ts | 17 ++++ src/routes/board.ts | 44 ++++++++++ src/server.ts | 26 ++++++ 5 files changed, 367 insertions(+) create mode 100644 src/database/postgres.ts create mode 100644 src/files/SwaggerUI.yaml create mode 100644 src/models/Thread.ts create mode 100644 src/routes/board.ts create mode 100644 src/server.ts (limited to 'src') diff --git a/src/database/postgres.ts b/src/database/postgres.ts new file mode 100644 index 0000000..2ea19bd --- /dev/null +++ b/src/database/postgres.ts @@ -0,0 +1,74 @@ +import { Client } from 'pg'; +var connectionString = "postgres://scale:secret@db/scale?sslmode=disable"; +let client: Client; + +export function InitPostgres() { + client = new Client(connectionString); + client.connect(); +} + +export function ClosePostgres() { + client.end(); +} + +export function CustomQuery(query: string, callback: (r: { status: number, data: any }) => void) { + client.query(query, (err, res) => { + let r: { status: number, data: any } = { status: 500, data: undefined }; + if (err) { + r.data = err.message; + } else { + r.status = 200; r.data = res.rows; + } + callback(r); + }); +} + +export function GetBoards(callback: (r: { status: number, data: any }) => void) { + let query = ` +select s.nspname as board +from pg_catalog.pg_namespace s +where nspname != 'pg_toast' and nspname != 'pg_catalog' +and nspname != 'public' and nspname != 'information_schema';`; // to remove the default schemas + + client.query(query, (err, res) => { + let r: { status: number, data: any } = { status: 500, data: undefined }; + if (err) { + r.status = 404; r.data = err.message; + } else { + r.status = 200; r.data = []; + for(let board of res.rows) + r.data.push(board.board); + } + callback(r); + }); +} + +export function CreateBoard(name: string, callback: (r: { status: number, data: any }) => void) { + let query = ` +create schema ${name}; +create table ${name}.threads( + id serial primary key, + timestamp bigint, + title text, + content text, + image text, + imagetype text +); +create table ${name}.comments( + id serial primary key, + timestamp bigint, + content text, + image text, + imagetype text +);`; + + client.query(query, (err, res) => { + let r: { status: number, data: any } = { status: 500, data: undefined }; + if (err) { + r.data = err.message; + } else { + r.status = 200; + } + callback(r); + }); +} diff --git a/src/files/SwaggerUI.yaml b/src/files/SwaggerUI.yaml new file mode 100644 index 0000000..019ca5e --- /dev/null +++ b/src/files/SwaggerUI.yaml @@ -0,0 +1,206 @@ +openapi: 3.0.0 +info: + title: Converted from Thread.yaml with typeconv + version: '1' + x-id: Thread.yaml + x-comment: >- + Generated from src/models/Thread.ts by core-types-json-schema + (https://github.com/grantila/core-types-json-schema) on behalf of typeconv + (https://github.com/grantila/typeconv) +paths: + /custom: + post: + summary: Run a custom sql query + responses: + '200': + description: Success + '500': + description: Failed + requestBody: + descripton: The command to run + content: + text/plain: + schema: + title: Command + type: string + /boards: + get: + summary: Get boards + responses: + '200': + description: Success + content: + application/json: + schema: + type: array + items: + type: string + '404': + description: Not Found + /boards/{board}: + post: + summary: Creat a new board + parameters: + - name: board + in: path + required: true + schema: + type: string + responses: + '200': + description: Success + '500': + description: Failed + /board/{board}: + get: + summary: Get threads in board + parameters: + - name: board + in: path + required: true + schema: + type: string + responses: + '200': + description: Success + content: + application/json: + schema: + type: array + items: + $ref: '#/components/schemas/Thread' + '404': + description: Not Found + post: + summary: Post a thread + parameters: + - name: board + in: path + required: true + schema: + type: string + requestBody: + description: Thread to post + content: + application/json: + schema: + $ref: '#/components/schemas/Thread' + responses: + '200': + description: Success + '500': + description: Failed + + /board/{board}/{tid}: + get: + summary: Get a thread + parameters: + - name: board + in: path + required: true + schema: + type: string + - name: tid + in: path + required: true + schema: + type: string + responses: + '200': + description: Success + content: + application/json: + schema: + $ref: '#/components/schemas/Thread' + '404': + description: Not Found + post: + summary: Post a comment + parameters: + - name: board + in: path + required: true + schema: + type: string + - name: tid + in: path + required: true + schema: + type: string + requestBody: + descriptors: The comment to post + content: + application/json: + schema: + $ref: '#components/schemas/Comment' + responses: + '200': + description: Success + '500': + description: Failed + +components: + schemas: + Comment: + properties: + id: + title: Comment.id + type: string + timestamp: + title: Comment.timestamp + type: number + content: + title: Comment.content + type: string + image: + title: Comment.image + type: string + imagetype: + title: Comment.imagetype + type: string + required: + - id + - timestamp + - content + - image + - imagetype + additionalProperties: false + title: Comment + type: object + Thread: + properties: + id: + title: Thread.id + type: string + timestamp: + title: Thread.timestamp + type: number + title: + title: Thread.title + type: string + content: + title: Thread.content + type: string + comments: + items: + $ref: '#/components/schemas/Comment' + title: Thread.comments.[] + title: Thread.comments + type: array + image: + title: Thread.image + type: string + imagetype: + title: Thread.imagetype + type: string + required: + - id + - title + - contnet + - comments + - timestamp + - image + - imageType + additionalProperties: false + title: Thread + type: object diff --git a/src/models/Thread.ts b/src/models/Thread.ts new file mode 100644 index 0000000..5960e11 --- /dev/null +++ b/src/models/Thread.ts @@ -0,0 +1,17 @@ +export type Comment = { + id: string; + timestamp: number; + content: string; + image: string; + imagetype: string //image or video +}; + +export type Thread = { + id: string; + timestamp: number; + title: string; + content: string; + comments: Comment[]; + image: string; + imagetype: string; //image or video +}; diff --git a/src/routes/board.ts b/src/routes/board.ts new file mode 100644 index 0000000..caa2d53 --- /dev/null +++ b/src/routes/board.ts @@ -0,0 +1,44 @@ +import express from 'express'; +import type { Thread, Comment } from '../models/Thread'; +import { CreateBoard, CustomQuery, GetBoards } from '../database/postgres' + +const router = express.Router(); + +router.post('/custom', (req, res) => { + CustomQuery(req.body, (r) => { + res.status(r.status).send(r.data); + }); +}); + +router.get('/boards', (req, res) => { + GetBoards((r) => { + res.status(r.status).send(r.data); + }); +}); + +router.post('/boards/:board', (req, res) => { + CreateBoard(req.params.board, (r) => { + res.status(r.status).send(r.data); + }); +}); + +router.get('/board/:board', (req, res) => { + res.status(404).send(req.params.board); +}); + +router.post('/board/:board', (req, res) => { + let t: Thread = req.body; + console.log(t); + res.status(403).send({board: req.params.board, thread: t}); +}); + +router.get('/board/:board/:tid', (req, res) => { + res.status(404).send([req.params.board, req.params.tid]); +}); + +router.post('/board/:board/:tid', (req, res) => { + let c: Comment = req.body; + res.status(404).send({params: [req.params.board, req.params.tid], body: c}); +}); + +export default router; diff --git a/src/server.ts b/src/server.ts new file mode 100644 index 0000000..0368f99 --- /dev/null +++ b/src/server.ts @@ -0,0 +1,26 @@ +import express from 'express'; +import boardRouter from './routes/board'; + +import swaggerUI from 'swagger-ui-express'; +import YAML from 'yamljs'; +const doc = YAML.load('src/files/SwaggerUI.yaml'); + +import { InitPostgres, ClosePostgres } from './database/postgres'; +InitPostgres(); + +const app = express(); + +app.use(express.json()); +app.use(express.text()); + +app.use('/swagger', swaggerUI.serve, swaggerUI.setup(doc)); + +app.use('/', boardRouter); + +app.get('*', (req, res) => { + res.send("haha i ate your request"); +}); + +app.listen(6969); + +// ClosePostgres(); -- cgit v1.2.3