2020-07-22 22:19:39 +08:00
|
|
|
|
package actionutils
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-08 15:17:00 +08:00
|
|
|
|
"encoding/json"
|
|
|
|
|
|
"errors"
|
2020-07-22 22:19:39 +08:00
|
|
|
|
"fmt"
|
2024-07-27 15:42:58 +08:00
|
|
|
|
"net"
|
|
|
|
|
|
"net/http"
|
|
|
|
|
|
"os"
|
|
|
|
|
|
"path/filepath"
|
|
|
|
|
|
"reflect"
|
|
|
|
|
|
"runtime"
|
|
|
|
|
|
"strings"
|
|
|
|
|
|
"time"
|
|
|
|
|
|
|
2023-06-30 18:08:30 +08:00
|
|
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/configloaders"
|
2022-06-08 15:17:00 +08:00
|
|
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/configs"
|
2023-06-30 18:08:30 +08:00
|
|
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
2021-06-05 11:35:36 +08:00
|
|
|
|
rpcerrors "github.com/TeaOSLab/EdgeCommon/pkg/rpc/errors"
|
2022-06-08 15:17:00 +08:00
|
|
|
|
"github.com/iwind/TeaGo/Tea"
|
2020-07-22 22:19:39 +08:00
|
|
|
|
"github.com/iwind/TeaGo/actions"
|
|
|
|
|
|
"github.com/iwind/TeaGo/logs"
|
2022-06-08 15:17:00 +08:00
|
|
|
|
"github.com/iwind/TeaGo/maps"
|
2022-09-07 15:57:06 +08:00
|
|
|
|
"github.com/iwind/gosock/pkg/gosock"
|
2020-07-22 22:19:39 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
2021-05-26 14:43:29 +08:00
|
|
|
|
// Fail 提示服务器错误信息
|
2023-06-30 18:08:30 +08:00
|
|
|
|
func Fail(actionPtr actions.ActionWrapper, err error) {
|
2022-06-08 15:17:00 +08:00
|
|
|
|
if err == nil {
|
|
|
|
|
|
err = errors.New("unknown error")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-06-30 18:08:30 +08:00
|
|
|
|
var langCode = configloaders.FindAdminLangForAction(actionPtr)
|
|
|
|
|
|
var serverErrString = codes.AdminCommon_ServerError.For(langCode)
|
2022-06-08 15:17:00 +08:00
|
|
|
|
|
2023-06-30 18:08:30 +08:00
|
|
|
|
logs.Println("[" + reflect.TypeOf(actionPtr).String() + "]" + findStack(err.Error()))
|
|
|
|
|
|
|
|
|
|
|
|
_, _, isLocalAPI, issuesHTML := parseAPIErr(actionPtr, err)
|
2023-06-10 16:31:20 +08:00
|
|
|
|
if isLocalAPI && len(issuesHTML) > 0 {
|
2023-06-30 18:08:30 +08:00
|
|
|
|
actionPtr.Object().Fail(serverErrString + "(" + err.Error() + ";最近一次错误提示:" + issuesHTML + ")")
|
2023-06-10 16:31:20 +08:00
|
|
|
|
} else {
|
2023-06-30 18:08:30 +08:00
|
|
|
|
actionPtr.Object().Fail(serverErrString + "(" + err.Error() + ")")
|
2020-07-22 22:19:39 +08:00
|
|
|
|
}
|
2022-06-08 15:17:00 +08:00
|
|
|
|
|
2023-06-10 16:31:20 +08:00
|
|
|
|
}
|
2022-09-07 15:57:06 +08:00
|
|
|
|
|
2023-06-10 16:31:20 +08:00
|
|
|
|
// FailPage 提示页面错误信息
|
2023-06-30 18:08:30 +08:00
|
|
|
|
func FailPage(actionPtr actions.ActionWrapper, err error) {
|
2023-06-10 16:31:20 +08:00
|
|
|
|
if err == nil {
|
|
|
|
|
|
err = errors.New("unknown error")
|
2022-09-07 15:57:06 +08:00
|
|
|
|
}
|
2022-06-08 15:17:00 +08:00
|
|
|
|
|
2023-06-30 18:08:30 +08:00
|
|
|
|
var langCode = configloaders.FindAdminLangForAction(actionPtr)
|
|
|
|
|
|
var serverErrString = codes.AdminCommon_ServerError.For(langCode)
|
|
|
|
|
|
|
|
|
|
|
|
logs.Println("[" + reflect.TypeOf(actionPtr).String() + "]" + findStack(err.Error()))
|
|
|
|
|
|
|
|
|
|
|
|
actionPtr.Object().ResponseWriter.WriteHeader(http.StatusInternalServerError)
|
2023-06-10 16:31:20 +08:00
|
|
|
|
|
2023-06-30 18:08:30 +08:00
|
|
|
|
if len(actionPtr.Object().Request.Header.Get("X-Requested-With")) > 0 {
|
|
|
|
|
|
actionPtr.Object().WriteString(serverErrString)
|
2021-04-05 08:18:17 +08:00
|
|
|
|
} else {
|
2023-06-30 18:08:30 +08:00
|
|
|
|
apiNodeIsStarting, apiNodeProgress, _, issuesHTML := parseAPIErr(actionPtr, err)
|
2022-06-08 15:17:00 +08:00
|
|
|
|
var html = `<!DOCTYPE html>
|
2021-04-05 08:18:17 +08:00
|
|
|
|
<html>
|
2022-06-08 15:17:00 +08:00
|
|
|
|
<head>
|
2023-04-09 17:29:29 +08:00
|
|
|
|
<title>正在处理...</title>
|
2022-06-08 15:17:00 +08:00
|
|
|
|
<meta charset="UTF-8"/>
|
|
|
|
|
|
<style type="text/css">
|
|
|
|
|
|
hr { border-top: 1px #ccc solid; }
|
|
|
|
|
|
.red { color: red; }
|
|
|
|
|
|
</style>
|
|
|
|
|
|
</head>
|
2021-04-05 08:18:17 +08:00
|
|
|
|
<body>
|
|
|
|
|
|
<div style="background: #eee; border: 1px #ccc solid; padding: 10px; font-size: 12px; line-height: 1.8">
|
2022-09-07 15:57:06 +08:00
|
|
|
|
`
|
|
|
|
|
|
if apiNodeIsStarting { // API节点正在启动
|
|
|
|
|
|
html += "<div class=\"red\">API节点正在启动,请耐心等待完成"
|
|
|
|
|
|
|
|
|
|
|
|
if len(apiNodeProgress) > 0 {
|
2023-04-09 17:29:29 +08:00
|
|
|
|
html += ":" + apiNodeProgress + "(刷新当前页面查看最新状态)"
|
2022-09-07 15:57:06 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
html += "</div>"
|
|
|
|
|
|
} else {
|
2023-06-30 18:08:30 +08:00
|
|
|
|
html += serverErrString + `
|
2022-06-08 15:17:00 +08:00
|
|
|
|
<div>可以通过查看 <strong><em>$安装目录/logs/run.log</em></strong> 日志文件查看具体的错误提示。</div>
|
|
|
|
|
|
<hr/>
|
|
|
|
|
|
<div class="red">Error: ` + err.Error() + `</div>`
|
|
|
|
|
|
|
2022-09-07 15:57:06 +08:00
|
|
|
|
if len(issuesHTML) > 0 {
|
|
|
|
|
|
html += ` <hr/>
|
2022-06-08 15:17:00 +08:00
|
|
|
|
<div class="red">` + issuesHTML + `</div>`
|
2022-09-07 15:57:06 +08:00
|
|
|
|
}
|
2022-06-08 15:17:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-06-30 18:08:30 +08:00
|
|
|
|
actionPtr.Object().WriteString(html + `
|
2022-06-08 15:17:00 +08:00
|
|
|
|
</div>
|
2021-04-05 08:18:17 +08:00
|
|
|
|
</body>
|
|
|
|
|
|
</html>`)
|
|
|
|
|
|
}
|
2020-07-22 22:19:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2021-05-26 14:43:29 +08:00
|
|
|
|
// MatchPath 判断动作的文件路径是否相当
|
2020-07-22 22:19:39 +08:00
|
|
|
|
func MatchPath(action *actions.ActionObject, path string) bool {
|
|
|
|
|
|
return action.Request.URL.Path == path
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2021-05-26 14:43:29 +08:00
|
|
|
|
// FindParentAction 查找父级Action
|
2020-10-06 21:02:37 +08:00
|
|
|
|
func FindParentAction(actionPtr actions.ActionWrapper) *ParentAction {
|
2023-08-08 18:32:58 +08:00
|
|
|
|
action, ok := actionPtr.(interface {
|
2023-08-08 14:17:16 +08:00
|
|
|
|
Parent() *ParentAction
|
|
|
|
|
|
})
|
|
|
|
|
|
if ok {
|
|
|
|
|
|
return action.Parent()
|
2020-10-06 21:02:37 +08:00
|
|
|
|
}
|
2023-08-08 14:17:16 +08:00
|
|
|
|
|
2020-10-06 21:02:37 +08:00
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-07-22 22:19:39 +08:00
|
|
|
|
func findStack(err string) string {
|
|
|
|
|
|
_, currentFilename, _, currentOk := runtime.Caller(1)
|
|
|
|
|
|
if currentOk {
|
|
|
|
|
|
for i := 1; i < 32; i++ {
|
|
|
|
|
|
_, filename, lineNo, ok := runtime.Caller(i)
|
|
|
|
|
|
if !ok {
|
|
|
|
|
|
break
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if filename == currentFilename || filepath.Base(filename) == "parent_action.go" {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
goPath := os.Getenv("GOPATH")
|
|
|
|
|
|
if len(goPath) > 0 {
|
|
|
|
|
|
absGoPath, err := filepath.Abs(goPath)
|
|
|
|
|
|
if err == nil {
|
|
|
|
|
|
filename = strings.TrimPrefix(filename, absGoPath)[1:]
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if strings.Contains(filename, "src") {
|
|
|
|
|
|
filename = filename[strings.Index(filename, "src"):]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-08-08 18:32:58 +08:00
|
|
|
|
err += "\n\t\t" + filename + ":" + fmt.Sprintf("%d", lineNo)
|
2020-07-22 22:19:39 +08:00
|
|
|
|
|
|
|
|
|
|
break
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
2023-06-10 16:31:20 +08:00
|
|
|
|
|
|
|
|
|
|
// 分析API节点的错误信息
|
|
|
|
|
|
func parseAPIErr(action actions.ActionWrapper, err error) (apiNodeIsStarting bool, apiNodeProgress string, isLocalAPI bool, issuesHTML string) {
|
|
|
|
|
|
// 当前API终端地址
|
|
|
|
|
|
var apiEndpoints = []string{}
|
|
|
|
|
|
apiConfig, apiConfigErr := configs.LoadAPIConfig()
|
|
|
|
|
|
if apiConfigErr == nil && apiConfig != nil {
|
2023-08-12 17:58:00 +08:00
|
|
|
|
apiEndpoints = append(apiEndpoints, apiConfig.RPCEndpoints...)
|
2023-06-10 16:31:20 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var isRPCConnError bool
|
2023-08-12 17:58:00 +08:00
|
|
|
|
_, isRPCConnError = rpcerrors.HumanError(err, apiEndpoints, Tea.ConfigFile(configs.ConfigFileName))
|
2023-06-10 16:31:20 +08:00
|
|
|
|
if isRPCConnError {
|
|
|
|
|
|
// API节点是否正在启动
|
|
|
|
|
|
var sock = gosock.NewTmpSock("edge-api")
|
|
|
|
|
|
reply, err := sock.SendTimeout(&gosock.Command{
|
|
|
|
|
|
Code: "starting",
|
|
|
|
|
|
Params: nil,
|
|
|
|
|
|
}, 1*time.Second)
|
|
|
|
|
|
if err == nil && reply != nil {
|
|
|
|
|
|
var params = maps.NewMap(reply.Params)
|
|
|
|
|
|
if params.GetBool("isStarting") {
|
|
|
|
|
|
apiNodeIsStarting = true
|
|
|
|
|
|
|
|
|
|
|
|
var progressMap = params.GetMap("progress")
|
|
|
|
|
|
apiNodeProgress = progressMap.GetString("description")
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 本地的一些错误提示
|
|
|
|
|
|
if isRPCConnError {
|
|
|
|
|
|
host, _, hostErr := net.SplitHostPort(action.Object().Request.Host)
|
|
|
|
|
|
if hostErr == nil {
|
|
|
|
|
|
for _, endpoint := range apiEndpoints {
|
|
|
|
|
|
if strings.HasPrefix(endpoint, "http://"+host) || strings.HasPrefix(endpoint, "https://"+host) || strings.HasPrefix(endpoint, host) {
|
|
|
|
|
|
isLocalAPI = true
|
|
|
|
|
|
break
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if isLocalAPI {
|
|
|
|
|
|
// 读取本地API节点的issues
|
|
|
|
|
|
issuesData, issuesErr := os.ReadFile(Tea.Root + "/edge-api/logs/issues.log")
|
|
|
|
|
|
if issuesErr == nil {
|
|
|
|
|
|
var issueMaps = []maps.Map{}
|
|
|
|
|
|
issuesErr = json.Unmarshal(issuesData, &issueMaps)
|
|
|
|
|
|
if issuesErr == nil && len(issueMaps) > 0 {
|
|
|
|
|
|
var issueMap = issueMaps[0]
|
|
|
|
|
|
issuesHTML = "本地API节点启动错误:" + issueMap.GetString("message") + ",处理建议:" + issueMap.GetString("suggestion")
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|