Compare commits

...

25 Commits

Author SHA1 Message Date
a0d52fc093 update data type 2025-05-10 20:39:12 +08:00
93bed37b9c update data type 2025-05-10 20:38:12 +08:00
4eb74a4e3a Merge pull request '升级exception, 简化逻辑' (#1) from feature/upgrade_exception into master
Reviewed-on: #1
2025-05-10 19:37:45 +08:00
1718c9c8ad 优化Error实现 2025-05-10 19:37:27 +08:00
255d79affb merge master && fix both modify 2025-05-10 19:35:05 +08:00
be3556e23e 升级exception, 简化逻辑 2025-05-10 18:19:11 +08:00
ca151fbc1f 优化请求成功默认返回文案 2025-02-07 17:17:24 +08:00
907c206627 默认成功状态码增加默认值 2025-02-07 17:13:53 +08:00
1e7f84ed84 支持打印堆栈 2024-09-30 16:13:43 +08:00
e8561a060c fix init table 2023-11-05 23:38:15 +08:00
84811e426c fix 2023-11-05 22:19:04 +08:00
482ef9296e 优化code reason定义 2023-11-05 22:12:40 +08:00
c293dcfa7f 优化code reason 2023-11-05 01:15:16 +08:00
be993da6db update code define 2023-11-05 00:57:23 +08:00
cd1454d34d 更新code定义 2023-11-05 00:33:32 +08:00
8c8c730a3b 增加是否制定错误码判断 2023-09-28 00:40:57 +08:00
b4732e360b 修复NewFromMessage错误信息异常问题 2023-09-25 18:49:37 +08:00
f674702ad2 解决自定义异常消息无法透出问题
解决自定义异常消息无法透出问题
2023-08-19 10:42:37 +08:00
16920b714d 升级异常接口约束 2023-06-09 18:12:14 +08:00
62d3723566 失败原因默认和message保持一致 2023-06-09 16:24:15 +08:00
e7e59267ca 优化reason 2023-06-09 16:22:55 +08:00
292fcf0839 按照code列表初始化异常码表 2023-06-09 16:20:14 +08:00
a038f90ed8 优化异常定义 2023-06-09 16:10:27 +08:00
290d1a7bd3 fix 2023-05-11 18:21:46 +08:00
2bc6061e1c 增加ToError方法 2023-05-11 18:20:56 +08:00
7 changed files with 179 additions and 162 deletions

View File

@ -3,7 +3,6 @@
<component name="ProjectTasksOptions">
<enabled-global>
<option value="go fmt" />
<option value="goimports" />
</enabled-global>
</component>
</project>

78
.idea/workspace.xml generated
View File

@ -1,78 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="AutoImportSettings">
<option name="autoReloadType" value="ALL" />
</component>
<component name="ChangeListManager">
<list default="true" id="0c7a619f-b520-4d41-ab0d-cfa1799d3cdf" name="Changes" comment="">
<change beforePath="$PROJECT_DIR$/code.go" beforeDir="false" afterPath="$PROJECT_DIR$/code.go" afterDir="false" />
</list>
<option name="SHOW_DIALOG" value="false" />
<option name="HIGHLIGHT_CONFLICTS" value="true" />
<option name="HIGHLIGHT_NON_ACTIVE_CHANGELIST" value="false" />
<option name="LAST_RESOLUTION" value="IGNORE" />
</component>
<component name="FileTemplateManagerImpl">
<option name="RECENT_TEMPLATES">
<list>
<option value="Go File" />
</list>
</option>
</component>
<component name="GOROOT" url="file:///usr/local/Cellar/go@1.17/1.17.13/libexec" />
<component name="Git.Settings">
<option name="RECENT_GIT_ROOT_PATH" value="$PROJECT_DIR$" />
</component>
<component name="GitSEFilterConfiguration">
<file-type-list>
<filtered-out-file-type name="LOCAL_BRANCH" />
<filtered-out-file-type name="REMOTE_BRANCH" />
<filtered-out-file-type name="TAG" />
<filtered-out-file-type name="COMMIT_BY_MESSAGE" />
</file-type-list>
</component>
<component name="GoLibraries">
<option name="indexEntireGoPath" value="false" />
</component>
<component name="MarkdownSettingsMigration">
<option name="stateVersion" value="1" />
</component>
<component name="ProjectId" id="2B4P2ryw4H3y3GlDs3w9wFxFqHs" />
<component name="ProjectViewState">
<option name="hideEmptyMiddlePackages" value="true" />
<option name="showLibraryContents" value="true" />
</component>
<component name="PropertiesComponent"><![CDATA[{
"keyToString": {
"RunOnceActivity.go.formatter.settings.were.checked": "true",
"RunOnceActivity.go.migrated.go.modules.settings": "true",
"RunOnceActivity.go.modules.go.list.on.any.changes.was.set": "true",
"WebServerToolWindowFactoryState": "false",
"dart.analysis.tool.window.visible": "false",
"go.import.settings.migrated": "true",
"last_opened_file_path": "/Users/zhangdeman",
"node.js.detected.package.eslint": "true",
"node.js.selected.package.eslint": "(autodetect)",
"vue.rearranger.settings.migration": "true"
}
}]]></component>
<component name="SpellCheckerSettings" RuntimeDictionaries="0" Folders="0" CustomDictionaries="0" DefaultDictionary="application-level" UseSingleDictionary="true" transferred="true" />
<component name="TypeScriptGeneratedFilesManager">
<option name="version" value="3" />
</component>
<component name="Vcs.Log.Tabs.Properties">
<option name="TAB_STATES">
<map>
<entry key="MAIN">
<value>
<State />
</value>
</entry>
</map>
</option>
</component>
<component name="VgoProject">
<integration-enabled>false</integration-enabled>
<settings-migrated>true</settings-migrated>
</component>
</project>

View File

@ -15,12 +15,16 @@ package exception
type IException interface {
// Error 兼容 go 内置 error
Error() string
// GetCode *获取错误码
GetCode() interface{}
// GetMessage *获取错误信息
GetMessage() string
// GetData 获取异常时的返回数据
GetData() interface{}
// GetHttpCode *获取当前异常要返回的http状态码, 不设置则 默认 200
GetHttpCode() int
// Code 获取错误码
Code() any
// Message 获取错误信息
Message() string
// Data 获取异常时的返回数据
Data() any
// ToError 转换为内置error类型
ToError() error
// IsCode 是否为指定code
IsCode(code any) bool
// GetStack 获取调用堆栈
GetStack() string
}

95
code.go
View File

@ -9,7 +9,6 @@ package exception
import (
"fmt"
"net/http"
)
var (
@ -18,21 +17,57 @@ var (
// Author : go_developer@163.com<白茶清欢>
//
// Date : 20:53 2022/6/25
codeTable map[interface{}]map[string]string
codeTable map[any]Code
codeList []Code
)
func init() {
// 规避没调用 InitCodeTable 导致空指针
codeTable = make(map[interface{}]map[string]string)
codeTable = make(map[any]Code)
codeList = make([]Code, 0)
}
// InitCodeTable 初始化码表, 同时只指定代表业务成功的状态码以及使用的语言
// InitCodeTableWithMessage 初始化码表, 同时只指定代表业务成功的状态码以及使用的语言
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 20:55 2022/6/25
func InitCodeTable(table map[interface{}]map[string]string, convertDefaultSuccessCode interface{}, convertDefaultLanguage string) {
codeTable = table
func InitCodeTableWithMessage(table map[any]map[string]string, convertDefaultSuccessCode any, convertDefaultLanguage string) {
for code, message := range table {
c := Code{
Value: code,
Message: message,
Reason: make(map[string]string),
Resolve: make(map[string]string),
}
codeTable[code] = c
codeList = append(codeList, c)
}
if nil == convertDefaultSuccessCode {
convertDefaultSuccessCode = 0
}
if len(convertDefaultLanguage) == 0 {
convertDefaultLanguage = defaultLanguage
}
defaultSuccessCode = convertDefaultSuccessCode
defaultLanguage = convertDefaultLanguage
}
// InitCodeTableWithCodeList 按照code列表初始化
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 16:19 2023/6/9
func InitCodeTableWithCodeList(list []Code, convertDefaultSuccessCode any, convertDefaultLanguage string) {
codeList = list
for _, itemCode := range list {
if nil == itemCode.Reason {
itemCode.Reason = make(map[string]string)
}
codeTable[itemCode.Value] = itemCode
}
if nil == convertDefaultSuccessCode {
convertDefaultSuccessCode = 0
}
@ -46,10 +81,8 @@ func InitCodeTable(table map[interface{}]map[string]string, convertDefaultSucces
var (
// messageWithCode 自动在message文案后追加状态码
messageWithCode = true
// defaultHttpCode 默认的http状态码
defaultHttpCode = http.StatusOK
// defaultSuccessCode 默认代表成功的状态码
defaultSuccessCode interface{}
defaultSuccessCode any = 0
// defaultLanguage 默认的语言
defaultLanguage = "zh"
)
@ -63,32 +96,48 @@ func MessageWithoutCode() {
messageWithCode = false
}
// getMessage 根据code获取文案
// GetMessage 根据code获取文案
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 21:16 2022/6/25
func getMessage(code interface{}) string {
message, exist := codeTable[code]
if !exist {
// 无论是否开启 messageWithCode , 未知错误强行带 code
return fmt.Sprintf("未知错误【%v】", code)
func GetMessage(code any, defaultMessage ...string) string {
var (
inputCodeInfo Code
exist bool
)
if inputCodeInfo, exist = codeTable[code]; !exist {
// 错误码不存在
if len(defaultMessage) > 0 && len(defaultMessage[0]) > 0 {
return defaultMessage[0]
}
// 无论是否开启 messageWithCode , 未知错误强行带 code
return fmt.Sprintf("unknown error【%v】", code)
}
if messageWithCode {
if code == defaultSuccessCode {
// 请求成功, 一直不带状态码后缀
return message[defaultLanguage]
return inputCodeInfo.Message[defaultLanguage]
}
return fmt.Sprintf(message[defaultLanguage]+"【%v】", code)
if messageWithCode {
return fmt.Sprintf(inputCodeInfo.Message[defaultLanguage]+"【%v】", code)
}
return message[defaultLanguage]
return inputCodeInfo.Message[defaultLanguage]
}
// GetMessage 获取消息信息
// GetCodeList ...
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 12:58 2022/6/26
func GetMessage(code interface{}) string {
return getMessage(code)
// Date : 18:04 2023/6/9
func GetCodeList() []Code {
return codeList
}
// GetCodeTable ...
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 18:05 2023/6/9
func GetCodeTable() map[any]Code {
return codeTable
}

20
define.go Normal file
View File

@ -0,0 +1,20 @@
// Package exception ...
//
// Description : exception ...
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 2023-06-09 15:54
package exception
// Code 状态码的结构
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 15:55 2023/6/9
type Code struct {
Value any `json:"value"` // 状态码的值
Message map[string]string `json:"message"` // 状态码对应的文案(key -> 语言 , value -> 对应语言的描述)
Reason map[string]string `json:"reason"` // 产生此错误码的原因描述(key -> 语言 , value -> 对应语言的原因列表)
Resolve map[string]string `json:"resolve"` // 解决此错误码的方案描述(key -> 语言 , value -> 对应语言的方案列表)
}

View File

@ -8,8 +8,12 @@
package exception
import (
"encoding/json"
"errors"
"fmt"
"runtime"
"strconv"
"strings"
)
// Exception 异常接口的具体实现
@ -18,30 +22,52 @@ import (
//
// Date *: 21:09 2022/6/25
type Exception struct {
code interface{}
code any
message string
httpCode int
data interface{}
data any
stack string
}
func (e *Exception) Error() string {
return e.GetMessage()
mapData := map[string]any{
"code": e.Code(),
"msg": e.Message(),
"data": e.Data(),
}
byteData, _ := json.Marshal(mapData)
return string(byteData)
}
func (e *Exception) GetCode() interface{} {
func (e *Exception) GetStack() string {
return e.stack
}
func (e *Exception) Code() any {
return e.code
}
func (e *Exception) GetMessage() string {
func (e *Exception) Message() string {
return e.message
}
func (e *Exception) GetData() interface{} {
func (e *Exception) Data() any {
return e.data
}
func (e *Exception) GetHttpCode() int {
return e.httpCode
func (e *Exception) ToError() error {
if nil == e {
return nil
}
return errors.New(e.Error())
}
// IsCode 判断是否为指定错误码
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 00:39 2023/9/28
func (e *Exception) IsCode(inputCode any) bool {
return fmt.Sprintf("%v", inputCode) == fmt.Sprintf("%v", e.Code())
}
// NewWithCode 仅使用错误码实例化异常
@ -49,17 +75,8 @@ func (e *Exception) GetHttpCode() int {
// Author : go_developer@163.com<白茶清欢>
//
// Date : 21:22 2022/6/25
func NewWithCode(code interface{}) IException {
return New(code, defaultHttpCode, nil)
}
// NewWithCodeAndHttpCode 使用 code + http_code 获取实例
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 21:25 2022/6/25
func NewWithCodeAndHttpCode(code interface{}, httpCode int) IException {
return New(code, httpCode, nil)
func NewWithCode(code any) IException {
return New(code, map[string]any{}, "")
}
// NewWithCodeAndData 使用 code + data 获取异常实例
@ -67,8 +84,8 @@ func NewWithCodeAndHttpCode(code interface{}, httpCode int) IException {
// Author : go_developer@163.com<白茶清欢>
//
// Date : 21:28 2022/6/25
func NewWithCodeAndData(code interface{}, data interface{}) IException {
return New(code, defaultHttpCode, data)
func NewWithCodeAndData(code any, data map[string]any) IException {
return New(code, data, "")
}
// New 获取异常实例
@ -76,16 +93,34 @@ func NewWithCodeAndData(code interface{}, data interface{}) IException {
// Author : go_developer@163.com<白茶清欢>
//
// Date : 21:28 2022/6/25
func New(code interface{}, httpCode int, data interface{}) IException {
func New(code any, data any, defaultMessage ...string) IException {
if nil == data {
// 保证数据结构的一致性, 同时避免后续使用出现空指针
data = map[string]interface{}{}
data = map[string]any{}
}
pcs := make([]uintptr, 128)
n := runtime.Callers(2, pcs)
frames := runtime.CallersFrames(pcs[:n])
var sb strings.Builder
for {
frame, more := frames.Next()
sb.WriteString(frame.Function)
sb.WriteByte('\n')
sb.WriteByte('\t')
sb.WriteString(frame.File)
sb.WriteByte(':')
sb.WriteString(strconv.Itoa(frame.Line))
sb.WriteByte('\n')
if !more {
break
}
}
return &Exception{
code: code,
message: getMessage(code),
httpCode: httpCode,
message: GetMessage(code, defaultMessage...),
data: data,
stack: sb.String(),
}
}
@ -94,13 +129,13 @@ func New(code interface{}, httpCode int, data interface{}) IException {
// Author : go_developer@163.com<白茶清欢>
//
// Date : 21:36 2022/6/25
func NewFromError(code interface{}, err error) IException {
func NewFromError(code any, err error) IException {
if nil == err {
return nil
}
return New(code, defaultHttpCode, map[string]interface{}{
"error": err.Error(),
})
return New(code, map[string]any{
"err_msg": err.Error(),
}, err.Error())
}
// NewFromMessage 从 code message 生成exception
@ -108,32 +143,20 @@ func NewFromError(code interface{}, err error) IException {
// Author : go_developer@163.com<白茶清欢>
//
// Date : 22:25 2023/2/11
func NewFromMessage(code interface{}, message string) IException {
func NewFromMessage(code any, message string) IException {
if len(message) == 0 {
message = fmt.Sprintf("%v", code)
message = GetMessage(code, "unknown error")
}
return NewFromError(code, errors.New(message))
}
// ToError 转换成内置error
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 21:37 2022/6/25
func ToError(e *Exception) error {
if nil == e {
return nil
}
return errors.New(e.GetMessage())
}
// IsSuccess 判断一个异常是否为成功
//
// Author : go_developer@163.com<白茶清欢>
//
// Date : 21:34 2022/6/25
func IsSuccess(e *Exception) bool {
return nil == e || e.GetCode() == defaultSuccessCode
return nil == e || e.IsCode(defaultSuccessCode)
}
// NewSuccess 代表Success的异常
@ -141,6 +164,6 @@ func IsSuccess(e *Exception) bool {
// Author : go_developer@163.com<白茶清欢>
//
// Date : 22:35 2022/6/25
func NewSuccess(data interface{}) IException {
return New(defaultSuccessCode, defaultHttpCode, data)
func NewSuccess(data any) IException {
return New(defaultSuccessCode, data, "")
}

2
go.mod
View File

@ -1,3 +1,3 @@
module git.zhangdeman.cn/zhangdeman/exception
go 1.17
go 1.20