168 lines
4.0 KiB
Go
168 lines
4.0 KiB
Go
package proxy
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"html/template"
|
|
"io/fs"
|
|
|
|
"github.com/WJQSERVER-STUDIO/go-utils/logger"
|
|
"github.com/cloudwego/hertz/pkg/app"
|
|
)
|
|
|
|
// 日志模块
|
|
var (
|
|
logw = logger.Logw
|
|
logDump = logger.LogDump
|
|
logDebug = logger.LogDebug
|
|
logInfo = logger.LogInfo
|
|
logWarning = logger.LogWarning
|
|
logError = logger.LogError
|
|
)
|
|
|
|
func HandleError(c *app.RequestContext, message string) {
|
|
ErrorPage(c, NewErrorWithStatusLookup(500, message))
|
|
logError(message)
|
|
}
|
|
|
|
type GHProxyErrors struct {
|
|
StatusCode int
|
|
StatusDesc string
|
|
StatusText string
|
|
HelpInfo string
|
|
ErrorMessage string
|
|
}
|
|
|
|
var (
|
|
ErrInvalidURL = &GHProxyErrors{
|
|
StatusCode: 400,
|
|
StatusDesc: "Bad Request",
|
|
StatusText: "无效请求",
|
|
HelpInfo: "请求的URL格式不正确,请检查后重试。",
|
|
}
|
|
ErrAuthHeaderUnavailable = &GHProxyErrors{
|
|
StatusCode: 401,
|
|
StatusDesc: "Unauthorized",
|
|
StatusText: "认证失败",
|
|
HelpInfo: "缺少或无效的鉴权信息。",
|
|
}
|
|
ErrForbidden = &GHProxyErrors{
|
|
StatusCode: 403,
|
|
StatusDesc: "Forbidden",
|
|
StatusText: "权限不足",
|
|
HelpInfo: "您没有权限访问此资源。",
|
|
}
|
|
ErrNotFound = &GHProxyErrors{
|
|
StatusCode: 404,
|
|
StatusDesc: "Not Found",
|
|
StatusText: "页面未找到",
|
|
HelpInfo: "抱歉,您访问的页面不存在。",
|
|
}
|
|
ErrTooManyRequests = &GHProxyErrors{
|
|
StatusCode: 429,
|
|
StatusDesc: "Too Many Requests",
|
|
StatusText: "请求过于频繁",
|
|
HelpInfo: "您的请求过于频繁,请稍后再试。",
|
|
}
|
|
ErrInternalServerError = &GHProxyErrors{
|
|
StatusCode: 500,
|
|
StatusDesc: "Internal Server Error",
|
|
StatusText: "服务器内部错误",
|
|
HelpInfo: "服务器处理您的请求时发生错误,请稍后重试或联系管理员。",
|
|
}
|
|
)
|
|
|
|
var statusErrorMap map[int]*GHProxyErrors
|
|
|
|
func init() {
|
|
statusErrorMap = map[int]*GHProxyErrors{
|
|
ErrInvalidURL.StatusCode: ErrInvalidURL,
|
|
ErrAuthHeaderUnavailable.StatusCode: ErrAuthHeaderUnavailable,
|
|
ErrForbidden.StatusCode: ErrForbidden,
|
|
ErrNotFound.StatusCode: ErrNotFound,
|
|
ErrTooManyRequests.StatusCode: ErrTooManyRequests,
|
|
ErrInternalServerError.StatusCode: ErrInternalServerError,
|
|
}
|
|
}
|
|
|
|
func NewErrorWithStatusLookup(statusCode int, errMsg string) *GHProxyErrors {
|
|
baseErr, found := statusErrorMap[statusCode]
|
|
|
|
if found {
|
|
return &GHProxyErrors{
|
|
StatusCode: baseErr.StatusCode,
|
|
StatusDesc: baseErr.StatusDesc,
|
|
StatusText: baseErr.StatusText,
|
|
HelpInfo: baseErr.HelpInfo,
|
|
ErrorMessage: errMsg,
|
|
}
|
|
} else {
|
|
return &GHProxyErrors{
|
|
StatusCode: statusCode,
|
|
ErrorMessage: errMsg,
|
|
}
|
|
}
|
|
}
|
|
|
|
var errPagesFs fs.FS
|
|
|
|
func InitErrPagesFS(pages fs.FS) error {
|
|
var err error
|
|
errPagesFs, err = fs.Sub(pages, "pages/err")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type ErrorPageData struct {
|
|
StatusCode int
|
|
StatusDesc string
|
|
StatusText string
|
|
HelpInfo string
|
|
ErrorMessage string
|
|
}
|
|
|
|
func ErrPageUnwarper(errInfo *GHProxyErrors) ErrorPageData {
|
|
return ErrorPageData{
|
|
StatusCode: errInfo.StatusCode,
|
|
StatusDesc: errInfo.StatusDesc,
|
|
StatusText: errInfo.StatusText,
|
|
HelpInfo: errInfo.HelpInfo,
|
|
ErrorMessage: errInfo.ErrorMessage,
|
|
}
|
|
}
|
|
|
|
func ErrorPage(c *app.RequestContext, errInfo *GHProxyErrors) {
|
|
pageData, err := htmlTemplateRender(errPagesFs, ErrPageUnwarper(errInfo))
|
|
if err != nil {
|
|
c.JSON(errInfo.StatusCode, map[string]string{"error": errInfo.ErrorMessage})
|
|
logDebug("Error reading page.tmpl: %v", err)
|
|
return
|
|
}
|
|
c.Data(errInfo.StatusCode, "text/html; charset=utf-8", pageData)
|
|
return
|
|
}
|
|
|
|
func htmlTemplateRender(fsys fs.FS, data interface{}) ([]byte, error) {
|
|
tmplPath := "page.tmpl"
|
|
tmpl, err := template.ParseFS(fsys, tmplPath)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error parsing template: %w", err)
|
|
}
|
|
if tmpl == nil {
|
|
return nil, fmt.Errorf("template is nil")
|
|
}
|
|
|
|
// 创建一个 bytes.Buffer 用于存储渲染结果
|
|
var buf bytes.Buffer
|
|
|
|
err = tmpl.Execute(&buf, data)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error executing template: %w", err)
|
|
}
|
|
|
|
// 返回 buffer 的内容作为 []byte
|
|
return buf.Bytes(), nil
|
|
}
|