mirror of https://github.com/cloudreve/Cloudreve
463 lines
13 KiB
Go
463 lines
13 KiB
Go
package serializer
|
||
|
||
import (
|
||
"context"
|
||
"errors"
|
||
"fmt"
|
||
"strings"
|
||
|
||
"github.com/cloudreve/Cloudreve/v4/pkg/filemanager/lock"
|
||
"github.com/cloudreve/Cloudreve/v4/pkg/logging"
|
||
"github.com/gin-gonic/gin"
|
||
"github.com/samber/lo"
|
||
)
|
||
|
||
// AppError 应用错误,实现了error接口
|
||
type AppError struct {
|
||
Code int
|
||
Msg string
|
||
RawError error
|
||
}
|
||
|
||
// NewError 返回新的错误对象
|
||
func NewError(code int, msg string, err error) AppError {
|
||
return AppError{
|
||
Code: code,
|
||
Msg: msg,
|
||
RawError: err,
|
||
}
|
||
}
|
||
|
||
// NewErrorFromResponse 从 serializer.Response 构建错误
|
||
func NewErrorFromResponse(resp *Response) AppError {
|
||
return AppError{
|
||
Code: resp.Code,
|
||
Msg: resp.Msg,
|
||
RawError: errors.New(resp.Error),
|
||
}
|
||
}
|
||
|
||
// WithError 将应用error携带标准库中的error
|
||
func (err *AppError) WithError(raw error) AppError {
|
||
return AppError{
|
||
Code: err.Code,
|
||
Msg: err.Msg,
|
||
RawError: raw,
|
||
}
|
||
}
|
||
|
||
// Error 返回业务代码确定的可读错误信息
|
||
func (err AppError) Error() string {
|
||
if err.RawError != nil {
|
||
return fmt.Sprintf("%s: %s", err.Msg, err.RawError.Error())
|
||
}
|
||
return err.Msg
|
||
}
|
||
|
||
func (err AppError) ErrCode() int {
|
||
var inheritedErr AppError
|
||
if errors.As(err.RawError, &inheritedErr) {
|
||
return inheritedErr.ErrCode()
|
||
}
|
||
return err.Code
|
||
}
|
||
|
||
func (err AppError) Unwrap() error {
|
||
return err.RawError
|
||
}
|
||
|
||
// 三位数错误编码为复用http原本含义
|
||
// 五位数错误编码为应用自定义错误
|
||
// 五开头的五位数错误编码为服务器端错误,比如数据库操作失败
|
||
// 四开头的五位数错误编码为客户端错误,有时候是客户端代码写错了,有时候是用户操作错误
|
||
const (
|
||
// CodeNotFullySuccess 未完全成功
|
||
CodeNotFullySuccess = 203
|
||
// CodeCheckLogin 未登录
|
||
CodeCheckLogin = 401
|
||
// CodeNoPermissionErr 未授权访问
|
||
CodeNoPermissionErr = 403
|
||
// CodeNotFound 资源未找到
|
||
CodeNotFound = 404
|
||
// CodeConflict 资源冲突
|
||
CodeConflict = 409
|
||
// CodeUploadFailed 上传出错
|
||
CodeUploadFailed = 40002
|
||
// CodeCreateFolderFailed 目录创建失败
|
||
CodeCreateFolderFailed = 40003
|
||
// CodeObjectExist 对象已存在
|
||
CodeObjectExist = 40004
|
||
// CodeSignExpired 签名过期
|
||
CodeSignExpired = 40005
|
||
// CodePolicyNotAllowed 当前存储策略不允许
|
||
CodePolicyNotAllowed = 40006
|
||
// CodeGroupNotAllowed 用户组无法进行此操作
|
||
CodeGroupNotAllowed = 40007
|
||
// CodeAdminRequired 非管理用户组
|
||
CodeAdminRequired = 40008
|
||
// CodeMasterNotFound 主机节点未注册
|
||
CodeMasterNotFound = 40009
|
||
// CodeUploadSessionExpired 上传会话已过期
|
||
CodeUploadSessionExpired = 40011
|
||
// CodeInvalidChunkIndex 无效的分片序号
|
||
CodeInvalidChunkIndex = 40012
|
||
// CodeInvalidContentLength 无效的正文长度
|
||
CodeInvalidContentLength = 40013
|
||
// CodePhoneRequired 未绑定手机
|
||
CodePhoneRequired = 40010
|
||
// CodeBatchSourceSize 超出批量获取外链限制
|
||
CodeBatchSourceSize = 40014
|
||
// CodeBatchAria2Size 超出最大 Aria2 任务数量限制
|
||
CodeBatchAria2Size = 40015
|
||
// CodeParentNotExist 父目录不存在
|
||
CodeParentNotExist = 40016
|
||
// CodeUserBaned 用户不活跃
|
||
CodeUserBaned = 40017
|
||
// CodeUserNotActivated 用户不活跃
|
||
CodeUserNotActivated = 40018
|
||
// CodeFeatureNotEnabled 此功能未开启
|
||
CodeFeatureNotEnabled = 40019
|
||
// CodeCredentialInvalid 凭证无效
|
||
CodeCredentialInvalid = 40020
|
||
// CodeUserNotFound 用户不存在
|
||
CodeUserNotFound = 40021
|
||
// Code2FACodeErr 二步验证代码错误
|
||
Code2FACodeErr = 40022
|
||
// CodeLoginSessionNotExist 登录会话不存在
|
||
CodeLoginSessionNotExist = 40023
|
||
// CodeInitializeAuthn 无法初始化 WebAuthn
|
||
CodeInitializeAuthn = 40024
|
||
// CodeWebAuthnCredentialError WebAuthn 凭证无效
|
||
CodeWebAuthnCredentialError = 40025
|
||
// CodeCaptchaError 验证码错误
|
||
CodeCaptchaError = 40026
|
||
// CodeCaptchaRefreshNeeded 验证码需要刷新
|
||
CodeCaptchaRefreshNeeded = 40027
|
||
// CodeFailedSendEmail 邮件发送失败
|
||
CodeFailedSendEmail = 40028
|
||
// CodeInvalidTempLink 临时链接无效
|
||
CodeInvalidTempLink = 40029
|
||
// CodeTempLinkExpired 临时链接过期
|
||
CodeTempLinkExpired = 40030
|
||
// CodeEmailProviderBaned 邮箱后缀被禁用
|
||
CodeEmailProviderBaned = 40031
|
||
// CodeEmailExisted 邮箱已被使用
|
||
CodeEmailExisted = 40032
|
||
// CodeEmailSent 邮箱已重新发送
|
||
CodeEmailSent = 40033
|
||
// CodeUserCannotActivate 用户无法激活
|
||
CodeUserCannotActivate = 40034
|
||
// 存储策略不存在
|
||
CodePolicyNotExist = 40035
|
||
// 无法删除默认存储策略
|
||
CodeDeleteDefaultPolicy = 40036
|
||
// 存储策略下还有文件
|
||
CodePolicyUsedByFiles = 40037
|
||
// 存储策略绑定了用户组
|
||
CodePolicyUsedByGroups = 40038
|
||
// 用户组不存在
|
||
CodeGroupNotFound = 40039
|
||
// 对系统用户组执行非法操作
|
||
CodeInvalidActionOnSystemGroup = 40040
|
||
// 用户组正在被使用
|
||
CodeGroupUsedByUser = 40041
|
||
// 为初始用户更改用户组
|
||
CodeChangeGroupForDefaultUser = 40042
|
||
// 对系统用户执行非法操作
|
||
CodeInvalidActionOnDefaultUser = 40043
|
||
// 文件不存在
|
||
CodeFileNotFound = 40044
|
||
// 列取文件失败
|
||
CodeListFilesError = 40045
|
||
// 对系统节点进行非法操作
|
||
CodeInvalidActionOnSystemNode = 40046
|
||
// 创建文件系统出错
|
||
CodeCreateFSError = 40047
|
||
// 创建任务出错
|
||
CodeCreateTaskError = 40048
|
||
// 文件尺寸太大
|
||
CodeFileTooLarge = 40049
|
||
// 文件类型不允许
|
||
CodeFileTypeNotAllowed = 40050
|
||
// 用户容量不足
|
||
CodeInsufficientCapacity = 40051
|
||
// 对象名非法
|
||
CodeIllegalObjectName = 40052
|
||
// 不支持对根目录执行此操作
|
||
CodeRootProtected = 40053
|
||
// 当前目录下已经有同名文件正在上传中
|
||
CodeConflictUploadOngoing = 40054
|
||
// 文件信息不一致
|
||
CodeMetaMismatch = 40055
|
||
// 不支持该格式的压缩文件
|
||
CodeUnsupportedArchiveType = 40056
|
||
// 可用存储策略发生变化
|
||
CodePolicyChanged = 40057
|
||
// 分享链接无效
|
||
CodeShareLinkNotFound = 40058
|
||
// 不能转存自己的分享
|
||
CodeSaveOwnShare = 40059
|
||
// 从机无法向主机发送回调请求
|
||
CodeSlavePingMaster = 40060
|
||
// Cloudreve 版本不一致
|
||
CodeVersionMismatch = 40061
|
||
// 积分不足
|
||
CodeInsufficientCredit = 40062
|
||
// 用户组冲突
|
||
CodeGroupConflict = 40063
|
||
// 当前已处于此用户组中
|
||
CodeGroupInvalid = 40064
|
||
// 兑换码无效
|
||
CodeInvalidGiftCode = 40065
|
||
// 已绑定了对应账号
|
||
CodeOpenIDBindConflict = 40066
|
||
// 对应账号已被绑定其他账号
|
||
CodeOpenIDBindOtherAccount = 40067
|
||
// 未绑定对应账号
|
||
CodeOpenIDNotLinked = 40068
|
||
// 密码不正确
|
||
CodeIncorrectPassword = 40069
|
||
// 分享无法预览
|
||
CodeDisabledSharePreview = 40070
|
||
// 签名无效
|
||
CodeInvalidSign = 40071
|
||
// 管理员无法购买用户组
|
||
CodeFulfillAdminGroup = 40072
|
||
// Lock confliced
|
||
CodeLockConflict = 40073
|
||
// Too many uris
|
||
CodeTooManyUris = 40074
|
||
// Lock token expired
|
||
CodeLockExpired = 40075
|
||
// Current updated version is stale
|
||
CodeStaleVersion = 40076
|
||
// CodeEntityNotExist Entity not exist
|
||
CodeEntityNotExist = 40077
|
||
// CodeFileDeleted File is deleted in recycle bin
|
||
CodeFileDeleted = 40078
|
||
// CodeFileCountLimitedReached file count limited reached
|
||
CodeFileCountLimitedReached = 40079
|
||
// CodeInvalidPassword invalid password
|
||
CodeInvalidPassword = 40080
|
||
// CodeBatchOperationNotFullyCompleted batch operation not fully completed
|
||
CodeBatchOperationNotFullyCompleted = 40081
|
||
// CodeOwnerOnly owner operation only
|
||
CodeOwnerOnly = 40082
|
||
// CodePurchaseRequired purchase required
|
||
CodePurchaseRequired = 40083
|
||
// CodeManagedAccountMinimumOpenID managed account minimum openid
|
||
CodeManagedAccountMinimumOpenID = 40084
|
||
// CodeAmountTooSmall amount too small
|
||
CodeAmountTooSmall = 40085
|
||
// CodeNodeUsedByStoragePolicy node used by storage policy
|
||
CodeNodeUsedByStoragePolicy = 40086
|
||
// CodeDomainNotLicensed domain not licensed
|
||
CodeDomainNotLicensed = 40087
|
||
// CodeDBError 数据库操作失败
|
||
CodeDBError = 50001
|
||
// CodeEncryptError 加密失败
|
||
CodeEncryptError = 50002
|
||
// CodeIOFailed IO操作失败
|
||
CodeIOFailed = 50004
|
||
// CodeInternalSetting 内部设置参数错误
|
||
CodeInternalSetting = 50005
|
||
// CodeCacheOperation 缓存操作失败
|
||
CodeCacheOperation = 50006
|
||
// CodeCallbackError 回调失败
|
||
CodeCallbackError = 50007
|
||
// 后台设置更新失败
|
||
CodeUpdateSetting = 50008
|
||
// 跨域策略添加失败
|
||
CodeAddCORS = 50009
|
||
// 节点不可用
|
||
CodeNodeOffline = 50010
|
||
// 文件元信息查询失败
|
||
CodeQueryMetaFailed = 50011
|
||
//CodeParamErr 各种奇奇怪怪的参数错误
|
||
CodeParamErr = 40001
|
||
// CodeNotSet 未定错误,后续尝试从error中获取
|
||
CodeNotSet = -1
|
||
)
|
||
|
||
// DBErrDeprecated 数据库操作失败
|
||
func DBErr(c context.Context, msg string, err error) Response {
|
||
if msg == "" {
|
||
msg = "Database operation failed."
|
||
}
|
||
return ErrWithDetails(c, CodeDBError, msg, err)
|
||
}
|
||
|
||
// DBErrDeprecated 数据库操作失败
|
||
func DBErrDeprecated(msg string, err error) Response {
|
||
if msg == "" {
|
||
msg = "Database operation failed."
|
||
}
|
||
return ErrDeprecated(CodeDBError, msg, err)
|
||
}
|
||
|
||
// ParamErr 各种参数错误
|
||
func ParamErr(c context.Context, msg string, err error) Response {
|
||
if msg == "" {
|
||
msg = "Invalid parameters."
|
||
}
|
||
return ErrWithDetails(c, CodeParamErr, msg, err)
|
||
}
|
||
|
||
// ParamErrDeprecated 各种参数错误
|
||
// Deprecated
|
||
func ParamErrDeprecated(msg string, err error) Response {
|
||
if msg == "" {
|
||
msg = "Invalid parameters."
|
||
}
|
||
return ErrDeprecated(CodeParamErr, msg, err)
|
||
}
|
||
|
||
// ErrDeprecated 通用错误处理
|
||
func ErrDeprecated(errCode int, msg string, err error) Response {
|
||
// 底层错误是AppError,则尝试从AppError中获取详细信息
|
||
var appError AppError
|
||
if errors.As(err, &appError) {
|
||
errCode = appError.Code
|
||
err = appError.RawError
|
||
msg = appError.Msg
|
||
}
|
||
|
||
res := Response{
|
||
Code: errCode,
|
||
Msg: msg,
|
||
}
|
||
// 生产环境隐藏底层报错
|
||
if err != nil && gin.Mode() != gin.ReleaseMode {
|
||
res.Error = err.Error()
|
||
}
|
||
return res
|
||
}
|
||
|
||
// ErrWithDetails 通用错误处理
|
||
func ErrWithDetails(c context.Context, errCode int, msg string, err error) Response {
|
||
res := Response{
|
||
Code: errCode,
|
||
Msg: msg,
|
||
CorrelationID: logging.CorrelationID(c).String(),
|
||
}
|
||
|
||
// 底层错误是AppError,则尝试从AppError中获取详细信息
|
||
var appError AppError
|
||
if errors.As(err, &appError) {
|
||
res.Code = appError.ErrCode()
|
||
err = appError.RawError
|
||
res.Msg = appError.Msg
|
||
|
||
// Special case for error with detail data
|
||
switch res.Code {
|
||
case CodeLockConflict:
|
||
var lockConflict lock.ConflictError
|
||
if errors.As(err, &lockConflict) {
|
||
res.Data = lockConflict
|
||
}
|
||
case CodeBatchOperationNotFullyCompleted:
|
||
var errs *AggregateError
|
||
if errors.As(err, &errs) {
|
||
res.AggregatedError = errs.Expand(c)
|
||
}
|
||
}
|
||
}
|
||
|
||
// 生产环境隐藏底层报错
|
||
if err != nil && gin.Mode() != gin.ReleaseMode {
|
||
res.Error = err.Error()
|
||
}
|
||
|
||
return res
|
||
}
|
||
|
||
// Err Builds error response without addition details, code and message will
|
||
// be retrieved from error if possible
|
||
func Err(c context.Context, err error) Response {
|
||
return ErrWithDetails(c, CodeNotSet, "", err)
|
||
}
|
||
|
||
// AggregateError is a special error type that contains multiple errors
|
||
type AggregateError struct {
|
||
errs map[string]error
|
||
}
|
||
|
||
// NewAggregateError creates a new AggregateError
|
||
func NewAggregateError() *AggregateError {
|
||
return &AggregateError{
|
||
errs: make(map[string]error, 0),
|
||
}
|
||
}
|
||
|
||
func (e *AggregateError) Error() string {
|
||
return fmt.Sprintf("aggregate error: one or more operation failed")
|
||
}
|
||
|
||
// Add adds an error to the aggregate
|
||
func (e *AggregateError) Add(id string, err error) {
|
||
e.errs[id] = err
|
||
}
|
||
|
||
// Merge merges another aggregate error into this one
|
||
func (e *AggregateError) Merge(err error) bool {
|
||
var errs *AggregateError
|
||
if errors.As(err, &errs) {
|
||
for id, err := range errs.errs {
|
||
e.errs[id] = err
|
||
}
|
||
|
||
return true
|
||
}
|
||
return false
|
||
}
|
||
|
||
// Raw returns the raw error map
|
||
func (e *AggregateError) Raw() map[string]error {
|
||
return e.errs
|
||
}
|
||
|
||
func (e *AggregateError) Remove(id string) {
|
||
delete(e.errs, id)
|
||
}
|
||
|
||
// Expand expands the aggregate error into a list of responses
|
||
func (e *AggregateError) Expand(ctx context.Context) map[string]Response {
|
||
return lo.MapEntries(e.errs, func(id string, err error) (string, Response) {
|
||
return id, Err(ctx, err)
|
||
})
|
||
}
|
||
|
||
// Aggregate aggregates the error and returns nil if there is no error;
|
||
// otherwise returns the error itself
|
||
func (e *AggregateError) Aggregate() error {
|
||
if len(e.errs) == 0 {
|
||
return nil
|
||
}
|
||
|
||
msg := "One or more operation failed"
|
||
if len(e.errs) == 1 {
|
||
for _, err := range e.errs {
|
||
msg = err.Error()
|
||
}
|
||
}
|
||
|
||
return NewError(CodeBatchOperationNotFullyCompleted, msg, e)
|
||
}
|
||
|
||
func (e *AggregateError) FormatFirstN(n int) string {
|
||
if len(e.errs) == 0 {
|
||
return ""
|
||
}
|
||
|
||
res := make([]string, 0, n)
|
||
for id, err := range e.errs {
|
||
res = append(res, fmt.Sprintf("%s: %s", id, err.Error()))
|
||
if len(res) >= n {
|
||
break
|
||
}
|
||
}
|
||
|
||
return strings.Join(res, ", ")
|
||
|
||
}
|