feat(bookmark): 添加 Authorization 请求头支持

为 CORS 配置添加 `Authorization` 请求头,以支持携带认证信息的跨域请求。

feat(user-np): 改进错误信息和用户注册流程

- 在注册 VFS 服务失败时返回更详细的错误信息,包括状态码和响应体内容。
- 用户创建失败时返回具体的错误详情,并确保数据库记录被正确回滚。
- 注销用户时先尝试删除 VFS 服务中的相关资源,再执行数据库删除操作。
- 使用 `Unscoped().Delete()` 确保物理删除用户数据。

feat(vfs): 完善用户目录结构及节点创建逻辑

- 创建用户主目录 `/home/username` 及其子目录 `.Recycle_Bin`。
- 调整 `CreateVFSNode` 方法参数类型为值传递。
- 修复创建用户目录时传参不一致的问题。
- 删除用户时递归清理其在 VFS 中的所有节点,并通过代理删除关联的服务节点。

feat(mage): 新增生成 TypeScript 类型定义的任务

新增 `Gen_TS` Mage 任务用于将 OpenAPI 配置文件转换为 TypeScript 类型定义文件,
支持 `vfs`、`bookmark` 和 `user_np` 模块。
This commit is contained in:
zzy
2025-09-28 12:28:22 +08:00
parent a3033bbf67
commit 429a863b76
6 changed files with 83 additions and 24 deletions

View File

@ -18,7 +18,7 @@ func main() {
config := cors.Config{
AllowAllOrigins: true,
AllowMethods: []string{"GET", "POST", "PUT", "PATCH", "DELETE", "HEAD"},
AllowHeaders: []string{"Origin", "Content-Length", "Content-Type"},
AllowHeaders: []string{"Origin", "Content-Length", "Content-Type", "Authorization"},
}
router.Use(cors.New(config))

View File

@ -57,7 +57,7 @@ func (u *UserNPImpl) RegisterVFSService(username, token string) (*string, error)
}
if reqs.StatusCode() != http.StatusCreated {
return nil, fmt.Errorf("failed to register vfs service: %s", reqs.Status())
return nil, fmt.Errorf("failed to register vfs service: %s %s", reqs.Status(), string(reqs.Body))
}
tokenHeader := reqs.HTTPResponse.Header.Get("X-VFS-Token")
@ -148,12 +148,13 @@ func (u *UserNPImpl) UserRegister(c *gin.Context, username string) {
// 保存到数据库
if err := u.db.Create(&user).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "用户创建失败"})
c.JSON(http.StatusInternalServerError, gin.H{"error": "用户创建失败", "detail": err.Error()})
return
}
if token, err := u.RegisterVFSService(username, u.vfsToken); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "无法生成访问令牌"})
u.db.Delete(&user)
c.JSON(http.StatusInternalServerError, gin.H{"error": "无法生成访问令牌", "detail": err.Error()})
u.db.Unscoped().Delete(&user)
} else {
user.Token = token
u.db.Save(&user)
@ -182,7 +183,12 @@ func (u *UserNPImpl) DeleteUser(c *gin.Context, username string) {
return
}
u.db.Delete(&user)
if err := u.UnregisterVFSService(username, u.vfsToken); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "无法注销访问令牌", "detail": err.Error()})
return
}
u.db.Unscoped().Delete(&user)
c.JSON(http.StatusNoContent, nil)
}

View File

@ -152,14 +152,14 @@ func (v *Vfs) CreateUser(username string) (string, error) {
ParentID: 0,
Type: VfsNodeTypeDirectory,
}
if createHomeErr := v.CreateVFSNode(homeDir); createHomeErr != nil {
if createHomeErr := v.CreateVFSNode(*homeDir); createHomeErr != nil {
err = createHomeErr
return "", err
}
}
// 创建用户目录 /home/username
userDir := &VfsNode{
userDir := VfsNode{
Name: username,
ParentID: homeDir.ID,
Type: VfsNodeTypeDirectory,
@ -171,6 +171,22 @@ func (v *Vfs) CreateUser(username string) (string, error) {
return "", err
}
if userDir, getUserErr := v.GetNodeByParentIDAndName(homeDir.ID, username); getUserErr != nil {
panic(getUserErr)
} else {
dotDir := VfsNode{
Name: ".Recycle_Bin",
ParentID: userDir.ID,
Type: VfsNodeTypeDirectory,
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
if createDirErr := v.CreateVFSNode(dotDir); createDirErr != nil {
err = createDirErr
return "", err
}
}
return token, nil
}
@ -540,7 +556,7 @@ func (v *Vfs) CheckNameValid(name string) error {
return nil
}
func (v *Vfs) CreateVFSNode(p *VfsNode) error {
func (v *Vfs) CreateVFSNode(p VfsNode) error {
_, err := v.DB.Exec(`
INSERT INTO vfs_nodes (name, parent_id, type, created_at, updated_at)
VALUES (?, ?, ?, ?, ?)`,

View File

@ -429,21 +429,6 @@ func ModelType2ResponseType(nodeType models.VfsNodeType) api.VFSNodeType {
// CreateUser implements server.StrictServerInterface.
func (v *VfsImpl) CreateUser(ctx context.Context, request api.CreateUserRequestObject) (api.CreateUserResponseObject, error) {
// 从上下文中获取请求头信息
// 注意:在严格模式下,我们需要从上下文获取请求信息
// 但这里我们无法直接访问请求头,因此需要在中间件中处理或使用其他方式传递
// 由于严格模式接口限制,我们无法直接访问请求头
// 这里假设我们通过其他方式验证了权限(比如中间件)
// token := c.GetHeader("X-VFS-Token")
// if token != v.config.RegisterToken || token != v.config.AdminToken {
// c.JSON(http.StatusForbidden, api.Error{
// Errtype: "AccessDenied",
// Message: "Access denied",
// })
// return
// }
// 创建用户
token, err := v.vfs.CreateUser(request.Username)
if err != nil {
@ -506,6 +491,27 @@ func (v *VfsImpl) DeleteUser(ctx context.Context, request api.DeleteUserRequestO
}, nil
}
if userDir, err := v.vfs.GetNodeByPath("/home/" + request.Username + "/"); err != nil {
panic(err)
} else {
if serviceIDs, err := v.vfs.DeleteNodeRecursively(userDir.ID); err != nil {
panic(err)
} else {
for _, serviceID := range serviceIDs {
// 对于服务类型节点,通过代理删除
_, err := v.StrictProxy2Service(http.MethodDelete, nil, v.vfs.GetVFSNode(serviceID))
if err != nil {
return api.DeleteUser500JSONResponse{
ServerInternalErrorJSONResponse: api.ServerInternalErrorJSONResponse{
Errtype: api.ErrorErrtypeServiceProxyError,
Message: err.Error(),
},
}, nil
}
}
}
}
// 从权限系统中移除用户
// 移除用户的所有角色
_, err = v.enfocer.DeleteRolesForUser(request.Username)

1
internal/vfs/vfs_user.go Normal file
View File

@ -0,0 +1 @@
package vfs

View File

@ -71,3 +71,33 @@ func Build_All() error {
return nil
}
func Gen_TS() error {
// 创建输出目录
if err := os.MkdirAll("./bin/ts", 0755); err != nil {
return fmt.Errorf("failed to create output directory: %w", err)
}
// 定义需要生成 TypeScript 类型的配置
configs := []struct {
Input string
Output string
}{
{"config/vfs/vfs.yaml", "bin/ts/vfs.ts"},
{"config/bookmark/bookmark.yaml", "bin/ts/bookmark.ts"},
{"config/user_np/user_np.yaml", "bin/ts/user_np.ts"},
}
// 为每个配置生成 TypeScript 类型文件
for _, cfg := range configs {
fmt.Printf("Generating TypeScript types for %s -> %s\n", cfg.Input, cfg.Output)
cmd := exec.Command("openapi-typescript",
cfg.Input, "--output", cfg.Output)
if err := cmd.Run(); err != nil {
return fmt.Errorf("failed to generate TypeScript types for %s: %w", cfg.Input, err)
}
}
return nil
}