feat: 添加短链管理功能,包括创建、更新、删除和列表接口

This commit is contained in:
2026-03-10 19:46:50 +08:00
parent 91eaad04d7
commit 48425c6120
25 changed files with 728 additions and 15 deletions

View File

@@ -0,0 +1,32 @@
import { eq } from 'drizzle-orm';
import { app, db, schema } from '@/app.ts';
app.route({
path: 'n5-shop',
key: 'create',
middleware: ['auth'],
description: `创建商品, 参数:
slug: 唯一业务ID, 必填
title: 商品标题, 必填
platform: 商品平台, 必填
orderLink: 订单链接, 必填
description: 商品描述, 选填
link: 商品链接, 选填
tags: 标签数组, 选填
data: 其他商品信息对象, 选填
userinfo: 卖家信息, 选填
`,
}).define(async (ctx) => {
const tokenUser = ctx.state.tokenUser;
const { userId, createdAt, updatedAt, id: _id, ...rest } = ctx.query.data || {};
if (!rest.slug) ctx.throw(400, 'slug 参数缺失');
if (!rest.title) ctx.throw(400, 'title 参数缺失');
if (!rest.platform) ctx.throw(400, 'platform 参数缺失');
if (!rest.orderLink) ctx.throw(400, 'orderLink 参数缺失');
const result = await db.insert(schema.n5Shop).values({
...rest,
userId: tokenUser.id,
}).returning();
ctx.body = result[0];
return ctx;
}).addTo(app);

View File

@@ -0,0 +1,25 @@
import { eq } from 'drizzle-orm';
import { app, db, schema } from '@/app.ts';
app.route({
path: 'n5-shop',
key: 'delete',
middleware: ['auth'],
description: '删除商品, 参数: id 商品ID',
}).define(async (ctx) => {
const tokenUser = ctx.state.tokenUser;
const { id } = ctx.query.data || {};
if (!id) {
ctx.throw(400, 'id 参数缺失');
}
const existing = await db.select().from(schema.n5Shop).where(eq(schema.n5Shop.id, id)).limit(1);
if (existing.length === 0) {
ctx.throw(404, '商品不存在');
}
if (existing[0].userId !== tokenUser.id) {
ctx.throw(403, '没有权限删除该商品');
}
await db.delete(schema.n5Shop).where(eq(schema.n5Shop.id, id));
ctx.body = { success: true };
return ctx;
}).addTo(app);

View File

@@ -0,0 +1,4 @@
import './list.ts';
import './create.ts';
import './update.ts';
import './delete.ts';

View File

@@ -0,0 +1,58 @@
import { desc, eq, count, or, like, and } from 'drizzle-orm';
import { app, db, schema } from '@/app.ts';
// 列表
app.route({
path: 'n5-shop',
key: 'list',
middleware: ['auth'],
description: '获取商品列表, 参数: page, pageSize, search, sort',
}).define(async (ctx) => {
const tokenUser = ctx.state.tokenUser;
const uid = tokenUser.id;
const { page = 1, pageSize = 20, search, sort = 'DESC' } = ctx.query || {};
const offset = (page - 1) * pageSize;
const orderByField = sort === 'ASC' ? schema.n5Shop.updatedAt : desc(schema.n5Shop.updatedAt);
let whereCondition: any = eq(schema.n5Shop.userId, uid);
if (search) {
whereCondition = and(
eq(schema.n5Shop.userId, uid),
or(
like(schema.n5Shop.title, `%${search}%`),
like(schema.n5Shop.slug, `%${search}%`),
)
);
}
const [list, totalCount] = await Promise.all([
db.select().from(schema.n5Shop).where(whereCondition).limit(pageSize).offset(offset).orderBy(orderByField),
db.select({ count: count() }).from(schema.n5Shop).where(whereCondition),
]);
ctx.body = {
list,
pagination: { page, current: page, pageSize, total: totalCount[0]?.count || 0 },
};
return ctx;
}).addTo(app);
// 获取单个
app.route({
path: 'n5-shop',
key: 'get',
description: '获取单个商品, 参数: id 或 slug',
}).define(async (ctx) => {
const { id, slug } = ctx.query || {};
if (!id && !slug) {
ctx.throw(400, 'id 或 slug 参数缺失');
}
const condition = id ? eq(schema.n5Shop.id, id) : eq(schema.n5Shop.slug, slug);
const existing = await db.select().from(schema.n5Shop).where(condition).limit(1);
if (existing.length === 0) {
ctx.throw(404, '商品不存在');
}
ctx.body = existing[0];
return ctx;
}).addTo(app);

View File

@@ -0,0 +1,35 @@
import { eq } from 'drizzle-orm';
import { app, db, schema } from '@/app.ts';
app.route({
path: 'n5-shop',
key: 'update',
middleware: ['auth'],
description: `更新商品, 参数:
id: 商品ID, 必填
title: 商品标题, 选填
platform: 商品平台, 选填
orderLink: 订单链接, 选填
description: 商品描述, 选填
link: 商品链接, 选填
tags: 标签数组, 选填
data: 其他商品信息对象, 选填
userinfo: 卖家信息, 选填
`,
}).define(async (ctx) => {
const tokenUser = ctx.state.tokenUser;
const { id, userId, createdAt, updatedAt, ...rest } = ctx.query.data || {};
if (!id) {
ctx.throw(400, 'id 参数缺失');
}
const existing = await db.select().from(schema.n5Shop).where(eq(schema.n5Shop.id, id)).limit(1);
if (existing.length === 0) {
ctx.throw(404, '商品不存在');
}
if (existing[0].userId !== tokenUser.id) {
ctx.throw(403, '没有权限更新该商品');
}
const result = await db.update(schema.n5Shop).set({ ...rest }).where(eq(schema.n5Shop.id, id)).returning();
ctx.body = result[0];
return ctx;
}).addTo(app);