我有一个 Controller 来处理每个方法的 /users
路由(GET
、POST
、PUT
、 DELETE
) 如下所示(删除了方法主体)。它主要类似于示例 here 中的代码
export class UsersController {
@route.get("")
all(@val.optional() offset: number, @val.optional() limit: number) { }
@route.get(":id")
get(@val.mongoId() id: string) { }
@route.post("")
async save(data: User) { }
@route.put(":id")
async update(@val.mongoId() id: string, data: User) { }
@route.delete(":id")
delete(@val.mongoId() id: string) { }
}
上面的 Controller 将处理:
GET /users?offest&limit
POST /users
GET /users/:id
PUT /users/:id
DELETE /users/:id
我还使用以下端点为当前登录用户创建了另一个速记端点:
GET /users/me
PUT /users/me
DELETE /users/me
来自文档 here ,表示可以使用多个路由装饰器来装饰 Controller / Action 。 我想知道是否有一种方法可以“重用”上面的 Controller 来处理上面的当前登录用户端点。
最佳答案
是的,有可能,Plumier 提供了非常灵活的路由设置,可以解决这个问题。
您需要将 /users/:id
的三个方法移动到单独的 Controller 中,这样就可以制作如下相同的路由模式:
@route.root("/users/me")
@route.root("/users/:id")
@middleware.use({
execute: async i => {
if (i.context.path.search(/users\/me$/i))
i.context.parameters![0] = i.context.state.user.userId
return i.proceed()
}
})
export class UserByIdController {
@route.get("")
get(@val.optional() @val.mongoId() id: string) {}
@route.put("")
update(@val.optional() @val.mongoId() id: string, data: User) {}
@route.delete("")
delete(@val.optional() @val.mongoId() id: string) {}
}
通过使用上面的设置,新 Controller 将像预期的那样生成 6 条路由:
GET /users/me
PUT /users/me
DELETE /users/me
GET /users/:id
PUT /users/:id
DELETE /users/:id
请记住,现在所有包含的方法的 id
参数都必须是可选的才能处理 /users/me
,这就是我们添加 @val 的原因。参数上的 optional()
。
最重要的是我们应用特定于 Controller 的中间件并进行元编程以使用当前用户 ID 填充第一个方法参数(id
参数)。
if (i.context.path.search(/users\/me$/i))
i.context.parameters![0] = i.context.state.user.userId
return i.proceed()
上面的代码片段意味着我们检查请求的当前路径是否以 users/me
结尾,然后为当前操作的第一个参数分配当前登录用户 ID。通过这样做,我们在不影响方法主体内部当前逻辑的情况下进行了一些修改。
重要:路由装饰器的顺序位置 @route.root("/users/me")
很重要,它应该在 的顶部/users/:id
.
关于javascript - 是否可以映射路由/用户/:id and/users/me into single controller in Plumier,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58477849/