From 7841904e8d39bdcabdba5217adae11d25de376e5 Mon Sep 17 00:00:00 2001
From: Type-32 <87076491+Type-32@users.noreply.github.com>
Date: Sun, 29 Sep 2024 08:07:13 +0800
Subject: [PATCH] feat: finished backend API and database schema
Signed-off-by: Type-32 <87076491+Type-32@users.noreply.github.com>
---
app/pages/galleries/index.vue | 21 +++++++
package-lock.json | 73 ++++++++++++++++++++++++-
package.json | 3 +-
server/api/v1/categories.get.ts | 45 +++++++++++++++
server/api/v1/category/delete.post.ts | 35 ++++++++++++
server/api/v1/category/edit.post.ts | 43 +++++++++++++++
server/api/v1/category/new.post.ts | 31 +++++++++++
server/api/v1/galleries.get.ts | 2 +-
server/api/v1/gallery/delete.post.ts | 35 ++++++++++++
server/api/v1/gallery/edit.post.ts | 13 +++--
server/api/v1/gallery/get-slug.get.ts | 29 ++++++++++
server/api/v1/gallery/new.post.ts | 5 +-
server/api/v1/gallery/publish.post.ts | 39 +++++++++++++
server/api/v1/gallery/unpublish.post.ts | 39 +++++++++++++
14 files changed, 403 insertions(+), 10 deletions(-)
create mode 100644 app/pages/galleries/index.vue
create mode 100644 server/api/v1/categories.get.ts
create mode 100644 server/api/v1/category/delete.post.ts
create mode 100644 server/api/v1/category/edit.post.ts
create mode 100644 server/api/v1/category/new.post.ts
create mode 100644 server/api/v1/gallery/delete.post.ts
create mode 100644 server/api/v1/gallery/get-slug.get.ts
create mode 100644 server/api/v1/gallery/publish.post.ts
create mode 100644 server/api/v1/gallery/unpublish.post.ts
diff --git a/app/pages/galleries/index.vue b/app/pages/galleries/index.vue
new file mode 100644
index 0000000..31a62bb
--- /dev/null
+++ b/app/pages/galleries/index.vue
@@ -0,0 +1,21 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/package-lock.json b/package-lock.json
index b5797b7..44eb727 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -31,7 +31,8 @@
"devDependencies": {
"@types/minio": "^7.1.1",
"nuxt": "^3.13.1",
- "nuxt-file-storage": "^0.2.6"
+ "nuxt-file-storage": "^0.2.6",
+ "prisma": "^5.20.0"
}
},
"node_modules/@alloc/quick-lru": {
@@ -2268,6 +2269,56 @@
}
}
},
+ "node_modules/@prisma/debug": {
+ "version": "5.20.0",
+ "resolved": "http://npm.117112.xyz/@prisma/debug/-/debug-5.20.0.tgz",
+ "integrity": "sha512-oCx79MJ4HSujokA8S1g0xgZUGybD4SyIOydoHMngFYiwEwYDQ5tBQkK5XoEHuwOYDKUOKRn/J0MEymckc4IgsQ==",
+ "devOptional": true,
+ "license": "Apache-2.0"
+ },
+ "node_modules/@prisma/engines": {
+ "version": "5.20.0",
+ "resolved": "http://npm.117112.xyz/@prisma/engines/-/engines-5.20.0.tgz",
+ "integrity": "sha512-DtqkP+hcZvPEbj8t8dK5df2b7d3B8GNauKqaddRRqQBBlgkbdhJkxhoJTrOowlS3vaRt2iMCkU0+CSNn0KhqAQ==",
+ "devOptional": true,
+ "hasInstallScript": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@prisma/debug": "5.20.0",
+ "@prisma/engines-version": "5.20.0-12.06fc58a368dc7be9fbbbe894adf8d445d208c284",
+ "@prisma/fetch-engine": "5.20.0",
+ "@prisma/get-platform": "5.20.0"
+ }
+ },
+ "node_modules/@prisma/engines-version": {
+ "version": "5.20.0-12.06fc58a368dc7be9fbbbe894adf8d445d208c284",
+ "resolved": "http://npm.117112.xyz/@prisma/engines-version/-/engines-version-5.20.0-12.06fc58a368dc7be9fbbbe894adf8d445d208c284.tgz",
+ "integrity": "sha512-Lg8AS5lpi0auZe2Mn4gjuCg081UZf88k3cn0RCwHgR+6cyHHpttPZBElJTHf83ZGsRNAmVCZCfUGA57WB4u4JA==",
+ "devOptional": true,
+ "license": "Apache-2.0"
+ },
+ "node_modules/@prisma/fetch-engine": {
+ "version": "5.20.0",
+ "resolved": "http://npm.117112.xyz/@prisma/fetch-engine/-/fetch-engine-5.20.0.tgz",
+ "integrity": "sha512-JVcaPXC940wOGpCOwuqQRTz6I9SaBK0c1BAyC1pcz9xBi+dzFgUu3G/p9GV1FhFs9OKpfSpIhQfUJE9y00zhqw==",
+ "devOptional": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@prisma/debug": "5.20.0",
+ "@prisma/engines-version": "5.20.0-12.06fc58a368dc7be9fbbbe894adf8d445d208c284",
+ "@prisma/get-platform": "5.20.0"
+ }
+ },
+ "node_modules/@prisma/get-platform": {
+ "version": "5.20.0",
+ "resolved": "http://npm.117112.xyz/@prisma/get-platform/-/get-platform-5.20.0.tgz",
+ "integrity": "sha512-8/+CehTZZNzJlvuryRgc77hZCWrUDYd/PmlZ7p2yNXtmf2Una4BWnTbak3us6WVdqoz5wmptk6IhsXdG2v5fmA==",
+ "devOptional": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@prisma/debug": "5.20.0"
+ }
+ },
"node_modules/@remirror/core-constants": {
"version": "2.0.2",
"resolved": "http://npm.117112.xyz/@remirror/core-constants/-/core-constants-2.0.2.tgz",
@@ -12089,6 +12140,26 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
+ "node_modules/prisma": {
+ "version": "5.20.0",
+ "resolved": "http://npm.117112.xyz/prisma/-/prisma-5.20.0.tgz",
+ "integrity": "sha512-6obb3ucKgAnsGS9x9gLOe8qa51XxvJ3vLQtmyf52CTey1Qcez3A6W6ROH5HIz5Q5bW+0VpmZb8WBohieMFGpig==",
+ "devOptional": true,
+ "hasInstallScript": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@prisma/engines": "5.20.0"
+ },
+ "bin": {
+ "prisma": "build/index.js"
+ },
+ "engines": {
+ "node": ">=16.13"
+ },
+ "optionalDependencies": {
+ "fsevents": "2.3.3"
+ }
+ },
"node_modules/process": {
"version": "0.11.10",
"resolved": "http://npm.117112.xyz/process/-/process-0.11.10.tgz",
diff --git a/package.json b/package.json
index fcb359a..51caad9 100644
--- a/package.json
+++ b/package.json
@@ -34,6 +34,7 @@
"devDependencies": {
"@types/minio": "^7.1.1",
"nuxt": "^3.13.1",
- "nuxt-file-storage": "^0.2.6"
+ "nuxt-file-storage": "^0.2.6",
+ "prisma": "^5.20.0"
}
}
diff --git a/server/api/v1/categories.get.ts b/server/api/v1/categories.get.ts
new file mode 100644
index 0000000..10d7cf2
--- /dev/null
+++ b/server/api/v1/categories.get.ts
@@ -0,0 +1,45 @@
+import { PrismaClient } from '@prisma/client'
+const prisma = new PrismaClient()
+export default defineEventHandler(async (event) => {
+ const query = getQuery(event)
+ try {
+ let pageIndex: number = 0
+ let pageAmount: number = 90
+ if (query.pagination != null && query.paginatePerPage != null) {
+ pageIndex = parseInt(query.pagination as any)
+ pageAmount = parseInt(query.paginatePerPage as any)
+ }
+ let data = null
+ if (query.ids != null) {
+ const queryIds = Array.isArray(query.ids)
+ ? query.ids.map(Number)
+ : [Number(query.ids)];
+ data = await prisma.category.findMany({
+ where: {
+ id: {
+ in: Array.from(queryIds as any[] as number[], Number)
+ }
+ }
+ })
+ } else {
+ data = await prisma.category.findMany({
+ select: {
+ id: true,
+ name: true,
+ galleries: true,
+ }
+ })
+ }
+
+ return data
+ } catch (error: any) {
+ if (error) {
+ console.log(`${event.toString()} -> Error at ${error.message}`)
+
+ return sendError(event, createError({
+ status: error.code as any as number,
+ message: error.message,
+ }))
+ }
+ }
+})
\ No newline at end of file
diff --git a/server/api/v1/category/delete.post.ts b/server/api/v1/category/delete.post.ts
new file mode 100644
index 0000000..3bcb720
--- /dev/null
+++ b/server/api/v1/category/delete.post.ts
@@ -0,0 +1,35 @@
+//@ts-ignore
+import { v4 as uuidv4 } from 'uuid';
+//@ts-ignore
+import jwt from 'jsonwebtoken';
+import { setCookie } from 'h3';
+import { PrismaClient } from '@prisma/client'
+import useAuth from "~/composables/useAuth";
+import useServerAuth from "~/composables/useServerAuth";
+const prisma = new PrismaClient()
+
+export default defineEventHandler(async (event) => {
+ const body = await readBody(event);
+
+ if(!body.id)
+ return sendError(event, createError({statusCode: 400, statusMessage: 'Not enough parameters.'}))
+
+ const header = getHeader(event, 'Authorization')
+ const auth = useServerAuth(event, header as any as string)
+
+ const isServerAuthenticated = await auth.validateServerToken()
+ if (!isServerAuthenticated) {
+ return sendError(event, createError({ statusCode: 403, statusMessage: 'Unauthorized; Please re-login.'}));
+ }
+
+ let data = await prisma.gallery.delete({
+ where: {
+ id: body.id as number,
+ }
+ })
+
+ if (!data)
+ return sendError(event, createError({statusCode: 401, statusMessage: 'Gallery not found' }));
+
+ return data;
+});
\ No newline at end of file
diff --git a/server/api/v1/category/edit.post.ts b/server/api/v1/category/edit.post.ts
new file mode 100644
index 0000000..61a1505
--- /dev/null
+++ b/server/api/v1/category/edit.post.ts
@@ -0,0 +1,43 @@
+//@ts-ignore
+import { v4 as uuidv4 } from 'uuid';
+//@ts-ignore
+import jwt from 'jsonwebtoken';
+import {Media, PrismaClient} from '@prisma/client'
+import useServerAuth from "~/composables/useServerAuth";
+const prisma = new PrismaClient()
+
+export default defineEventHandler(async (event) => {
+ const body = await readBody<{ id: number, name: string, galleries: number[]}>(event);
+ const header = getHeader(event, 'Authorization')
+
+ if (!body.id || !body.name || !body.galleries || !header) {
+ return sendError(event, createError({ statusCode: 400, statusMessage: 'Cannot access without authorization header.' }));
+ }
+
+ const auth = useServerAuth(event, header)
+
+ const isServerAuthenticated = await auth.validateServerToken()
+ if (!isServerAuthenticated) {
+ return sendError(event, createError({ statusCode: 403, statusMessage: 'Unauthorized; Please re-login.'}));
+ }
+
+ let {error} = await prisma.category.update({
+ where: {
+ id: body.id,
+ },
+ data: {
+ updatedAt: new Date().toISOString(),
+ name: body.name,
+ galleries: {
+ connect: galleryIds.map(galleryId => ({
+ id: galleryId,
+ })),
+ },
+ }
+ })
+
+ if (error)
+ return sendError(event, createError({statusCode: 401, statusMessage: 'Gallery not found' }));
+
+ return edit;
+});
\ No newline at end of file
diff --git a/server/api/v1/category/new.post.ts b/server/api/v1/category/new.post.ts
new file mode 100644
index 0000000..3f5a99f
--- /dev/null
+++ b/server/api/v1/category/new.post.ts
@@ -0,0 +1,31 @@
+import {PrismaClient} from '@prisma/client'
+import useServerAuth from "~/composables/useServerAuth";
+const prisma = new PrismaClient()
+
+export default defineEventHandler(async (event) => {
+ const body = await readBody<{name: string}>(event);
+ const header = getHeader(event, 'Authorization')
+ console.log(header)
+
+ if (!body.name || !header) {
+ return sendError(event, createError({ statusCode: 400, statusMessage: 'Requires full parameters or headers' }));
+ }
+
+ const auth = useServerAuth(event, header)
+
+ const isServerAuthenticated = await auth.validateServerToken()
+ if (!isServerAuthenticated) {
+ return sendError(event, createError({ statusCode: 403, statusMessage: 'Unauthorized; Please re-login.'}));
+ }
+
+ let {data, error} = await prisma.category.create({
+ data: {
+ name: body.name as string
+ }
+ })
+
+ if (error)
+ return sendError(event, createError({statusCode: 401, statusMessage: 'Unsuccessful creation. Try again later.' }));
+
+ return data;
+});
\ No newline at end of file
diff --git a/server/api/v1/galleries.get.ts b/server/api/v1/galleries.get.ts
index e8f6907..09889b5 100644
--- a/server/api/v1/galleries.get.ts
+++ b/server/api/v1/galleries.get.ts
@@ -4,7 +4,7 @@ export default defineEventHandler(async (event) => {
const query = getQuery(event)
try {
let pageIndex: number = 0
- let pageAmount: number = 9
+ let pageAmount: number = 90
if (query.pagination != null && query.paginatePerPage != null) {
pageIndex = parseInt(query.pagination as any)
pageAmount = parseInt(query.paginatePerPage as any)
diff --git a/server/api/v1/gallery/delete.post.ts b/server/api/v1/gallery/delete.post.ts
new file mode 100644
index 0000000..ffc07fe
--- /dev/null
+++ b/server/api/v1/gallery/delete.post.ts
@@ -0,0 +1,35 @@
+//@ts-ignore
+import { v4 as uuidv4 } from 'uuid';
+//@ts-ignore
+import jwt from 'jsonwebtoken';
+import { setCookie } from 'h3';
+import { PrismaClient } from '@prisma/client'
+import useAuth from "~/composables/useAuth";
+import useServerAuth from "~/composables/useServerAuth";
+const prisma = new PrismaClient()
+
+export default defineEventHandler(async (event) => {
+ const body = await readBody(event);
+
+ if(!body.id)
+ return sendError(event, createError({statusCode: 400, statusMessage: 'Not enough parameters.'}))
+
+ const header = getHeader(event, 'Authorization')
+ const auth = useServerAuth(event, header as any as string)
+
+ const isServerAuthenticated = await auth.validateServerToken()
+ if (!isServerAuthenticated) {
+ return sendError(event, createError({ statusCode: 403, statusMessage: 'Unauthorized; Please re-login.'}));
+ }
+
+ let {data, error} = await prisma.gallery.delete({
+ where: {
+ id: body.id as number,
+ }
+ })
+
+ if (error)
+ return sendError(event, createError({statusCode: 401, statusMessage: 'Gallery not found' }));
+
+ return data;
+});
\ No newline at end of file
diff --git a/server/api/v1/gallery/edit.post.ts b/server/api/v1/gallery/edit.post.ts
index c5c4a04..b467743 100644
--- a/server/api/v1/gallery/edit.post.ts
+++ b/server/api/v1/gallery/edit.post.ts
@@ -21,19 +21,24 @@ export default defineEventHandler(async (event) => {
return sendError(event, createError({ statusCode: 403, statusMessage: 'Unauthorized; Please re-login.'}));
}
- let edit = await prisma.gallery.update({
+ let {data: edit, error} = await prisma.gallery.update({
where: {
id: body.id,
},
data: {
updatedAt: new Date().toISOString(),
name: body.name,
- published: body.published
+ published: body.published,
+ medias: {
+ connect: body.medias.map(mediaId => ({
+ mediaId: mediaId,
+ })),
+ },
}
})
- if (!edit)
- return sendError(event, createError({statusCode: 401, statusMessage: 'Article not found' }));
+ if (error)
+ return sendError(event, createError({statusCode: 401, statusMessage: 'Gallery not found' }));
return edit;
});
\ No newline at end of file
diff --git a/server/api/v1/gallery/get-slug.get.ts b/server/api/v1/gallery/get-slug.get.ts
new file mode 100644
index 0000000..cd7705d
--- /dev/null
+++ b/server/api/v1/gallery/get-slug.get.ts
@@ -0,0 +1,29 @@
+import { PrismaClient } from '@prisma/client'
+const prisma = new PrismaClient()
+export default defineEventHandler(async (event) => {
+ const query = getQuery(event)
+
+ try {
+ let data = null
+ if (query.slug != null) {
+ data = await prisma.gallery.findFirst({
+ where: {
+ id: query.id as any as number
+ }
+ })
+ } else {
+ return null
+ }
+
+ return data
+ } catch (error: any) {
+ if (error) {
+ console.log(`${event.toString()} -> Error at ${error.message}`)
+
+ return sendError(event, createError({
+ status: error.code as any as number,
+ message: error.message,
+ }))
+ }
+ }
+})
\ No newline at end of file
diff --git a/server/api/v1/gallery/new.post.ts b/server/api/v1/gallery/new.post.ts
index df53163..cb8b988 100644
--- a/server/api/v1/gallery/new.post.ts
+++ b/server/api/v1/gallery/new.post.ts
@@ -15,18 +15,17 @@ export default defineEventHandler(async (event) => {
const isServerAuthenticated = await auth.validateServerToken()
if (!isServerAuthenticated) {
-
return sendError(event, createError({ statusCode: 403, statusMessage: 'Unauthorized; Please re-login.'}));
}
- let data = await prisma.gallery.create({
+ let {data, error} = await prisma.gallery.create({
data: {
name: body.name as string,
published: false
}
})
- if (!data)
+ if (error)
return sendError(event, createError({statusCode: 401, statusMessage: 'Unsuccessful creation. Try again later.' }));
return data;
diff --git a/server/api/v1/gallery/publish.post.ts b/server/api/v1/gallery/publish.post.ts
new file mode 100644
index 0000000..4ba68c8
--- /dev/null
+++ b/server/api/v1/gallery/publish.post.ts
@@ -0,0 +1,39 @@
+//@ts-ignore
+import { v4 as uuidv4 } from 'uuid';
+//@ts-ignore
+import jwt from 'jsonwebtoken';
+import { setCookie } from 'h3';
+import { PrismaClient } from '@prisma/client'
+import useAuth from "~/composables/useAuth";
+import useServerAuth from "~/composables/useServerAuth";
+const prisma = new PrismaClient()
+
+export default defineEventHandler(async (event) => {
+ const body = await readBody(event);
+
+ if (!body.id) {
+ return sendError(event, createError({ statusCode: 400, statusMessage: 'Requires full parameters' }));
+ }
+
+ const header = getHeader(event, 'Authorization')
+ const auth = useServerAuth(event, header as any as string)
+
+ const isServerAuthenticated = await auth.validateServerToken()
+ if (!isServerAuthenticated) {
+ return sendError(event, createError({ statusCode: 403, statusMessage: 'Unauthorized; Please re-login.'}));
+ }
+
+ let {data, error} = await prisma.gallery.update({
+ where: {
+ id: body.id as number,
+ },
+ data: {
+ published: true
+ }
+ })
+
+ if (error)
+ return sendError(event, createError({statusCode: 401, statusMessage: 'An error occurred while publishing this gallery. Try again later.' }));
+
+ return data;
+});
\ No newline at end of file
diff --git a/server/api/v1/gallery/unpublish.post.ts b/server/api/v1/gallery/unpublish.post.ts
new file mode 100644
index 0000000..ba18527
--- /dev/null
+++ b/server/api/v1/gallery/unpublish.post.ts
@@ -0,0 +1,39 @@
+//@ts-ignore
+import { v4 as uuidv4 } from 'uuid';
+//@ts-ignore
+import jwt from 'jsonwebtoken';
+import { setCookie } from 'h3';
+import { PrismaClient } from '@prisma/client'
+import useAuth from "~/composables/useAuth";
+import useServerAuth from "~/composables/useServerAuth";
+const prisma = new PrismaClient()
+
+export default defineEventHandler(async (event) => {
+ const body = await readBody(event);
+
+ if (!body.id) {
+ return sendError(event, createError({ statusCode: 400, statusMessage: 'Requires full parameters' }));
+ }
+
+ const header = getHeader(event, 'Authorization')
+ const auth = useServerAuth(event, header as any as string)
+
+ const isServerAuthenticated = await auth.validateServerToken()
+ if (!isServerAuthenticated) {
+ return sendError(event, createError({ statusCode: 403, statusMessage: 'Unauthorized; Please re-login.'}));
+ }
+
+ let {data, error} = await prisma.gallery.update({
+ where: {
+ id: body.id as number,
+ },
+ data: {
+ published: false
+ }
+ })
+
+ if (error)
+ return sendError(event, createError({statusCode: 401, statusMessage: 'An error occurred while unpublishing this gallery. Try again later.' }));
+
+ return data;
+});
\ No newline at end of file