0b7f43b149
* fix: dashboard custom theme * api: expose HideForGuest
339 lines
7.6 KiB
Go
339 lines
7.6 KiB
Go
package controller
|
|
|
|
import (
|
|
"fmt"
|
|
"html/template"
|
|
"log"
|
|
"net/http"
|
|
"os"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"code.cloudfoundry.org/bytefmt"
|
|
"github.com/gin-contrib/pprof"
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/hashicorp/go-uuid"
|
|
"github.com/nicksnyder/go-i18n/v2/i18n"
|
|
|
|
"github.com/naiba/nezha/model"
|
|
"github.com/naiba/nezha/pkg/mygin"
|
|
"github.com/naiba/nezha/pkg/utils"
|
|
"github.com/naiba/nezha/proto"
|
|
"github.com/naiba/nezha/resource"
|
|
"github.com/naiba/nezha/service/rpc"
|
|
"github.com/naiba/nezha/service/singleton"
|
|
)
|
|
|
|
func ServeWeb(port uint) *http.Server {
|
|
gin.SetMode(gin.ReleaseMode)
|
|
r := gin.Default()
|
|
if singleton.Conf.Debug {
|
|
gin.SetMode(gin.DebugMode)
|
|
pprof.Register(r)
|
|
}
|
|
r.Use(natGateway)
|
|
tmpl := template.New("").Funcs(funcMap)
|
|
var err error
|
|
tmpl, err = tmpl.ParseFS(resource.TemplateFS, "template/**/*.html")
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
tmpl = loadThirdPartyTemplates(tmpl)
|
|
r.SetHTMLTemplate(tmpl)
|
|
r.Use(mygin.RecordPath)
|
|
r.StaticFS("/static", http.FS(resource.StaticFS))
|
|
routers(r)
|
|
page404 := func(c *gin.Context) {
|
|
mygin.ShowErrorPage(c, mygin.ErrInfo{
|
|
Code: http.StatusNotFound,
|
|
Title: "该页面不存在",
|
|
Msg: "该页面内容可能已着陆火星",
|
|
Link: "/",
|
|
Btn: "返回首页",
|
|
}, true)
|
|
}
|
|
r.NoRoute(page404)
|
|
r.NoMethod(page404)
|
|
|
|
srv := &http.Server{
|
|
Addr: fmt.Sprintf(":%d", port),
|
|
ReadHeaderTimeout: time.Second * 5,
|
|
Handler: r,
|
|
}
|
|
return srv
|
|
}
|
|
|
|
func routers(r *gin.Engine) {
|
|
// 通用页面
|
|
cp := commonPage{r: r}
|
|
cp.serve()
|
|
// 游客页面
|
|
gp := guestPage{r}
|
|
gp.serve()
|
|
// 会员页面
|
|
mp := &memberPage{r}
|
|
mp.serve()
|
|
// API
|
|
api := r.Group("api")
|
|
{
|
|
ma := &memberAPI{api}
|
|
ma.serve()
|
|
}
|
|
}
|
|
|
|
func loadThirdPartyTemplates(tmpl *template.Template) *template.Template {
|
|
ret := tmpl
|
|
themes, err := os.ReadDir("resource/template")
|
|
if err != nil {
|
|
log.Printf("NEZHA>> Error reading themes folder: %v", err)
|
|
return ret
|
|
}
|
|
for _, theme := range themes {
|
|
if !theme.IsDir() {
|
|
continue
|
|
}
|
|
|
|
themeDir := theme.Name()
|
|
if strings.HasPrefix(themeDir, "dashboard-") {
|
|
// load dashboard templates, ignore desc file
|
|
ret = loadTemplates(ret, themeDir)
|
|
continue
|
|
}
|
|
|
|
if !strings.HasPrefix(themeDir, "theme-") {
|
|
log.Printf("NEZHA>> Invalid theme name: %s", themeDir)
|
|
continue
|
|
}
|
|
|
|
descPath := filepath.Join("resource", "template", themeDir, "theme.json")
|
|
desc, err := os.ReadFile(filepath.Clean(descPath))
|
|
if err != nil {
|
|
log.Printf("NEZHA>> Error opening %s config: %v", themeDir, err)
|
|
continue
|
|
}
|
|
|
|
themeName, err := utils.GjsonGet(desc, "name")
|
|
if err != nil {
|
|
log.Printf("NEZHA>> Error opening %s config: not a valid description file", theme.Name())
|
|
continue
|
|
}
|
|
|
|
// load templates
|
|
ret = loadTemplates(ret, themeDir)
|
|
|
|
themeKey := strings.TrimPrefix(themeDir, "theme-")
|
|
model.Themes[themeKey] = themeName.String()
|
|
}
|
|
|
|
return ret
|
|
}
|
|
|
|
func loadTemplates(tmpl *template.Template, themeDir string) *template.Template {
|
|
// load templates
|
|
templatePath := filepath.Join("resource", "template", themeDir, "*.html")
|
|
t, err := tmpl.ParseGlob(templatePath)
|
|
if err != nil {
|
|
log.Printf("NEZHA>> Error parsing templates %s: %v", themeDir, err)
|
|
return tmpl
|
|
}
|
|
|
|
return t
|
|
}
|
|
|
|
var funcMap = template.FuncMap{
|
|
"tr": func(id string, dataAndCount ...interface{}) string {
|
|
conf := i18n.LocalizeConfig{
|
|
MessageID: id,
|
|
}
|
|
if len(dataAndCount) > 0 {
|
|
conf.TemplateData = dataAndCount[0]
|
|
}
|
|
if len(dataAndCount) > 1 {
|
|
conf.PluralCount = dataAndCount[1]
|
|
}
|
|
return singleton.Localizer.MustLocalize(&conf)
|
|
},
|
|
"toValMap": func(val interface{}) map[string]interface{} {
|
|
return map[string]interface{}{
|
|
"Value": val,
|
|
}
|
|
},
|
|
"tf": func(t time.Time) string {
|
|
return t.In(singleton.Loc).Format("01/02/2006 15:04:05")
|
|
},
|
|
"len": func(slice []interface{}) string {
|
|
return strconv.Itoa(len(slice))
|
|
},
|
|
"safe": func(s string) template.HTML {
|
|
return template.HTML(s) // #nosec
|
|
},
|
|
"tag": func(s string) template.HTML {
|
|
return template.HTML(`<` + s + `>`) // #nosec
|
|
},
|
|
"stf": func(s uint64) string {
|
|
return time.Unix(int64(s), 0).In(singleton.Loc).Format("01/02/2006 15:04")
|
|
},
|
|
"sf": func(duration uint64) string {
|
|
return time.Duration(time.Duration(duration) * time.Second).String()
|
|
},
|
|
"sft": func(future time.Time) string {
|
|
return time.Until(future).Round(time.Second).String()
|
|
},
|
|
"bf": func(b uint64) string {
|
|
return bytefmt.ByteSize(b)
|
|
},
|
|
"ts": func(s string) string {
|
|
return strings.TrimSpace(s)
|
|
},
|
|
"float32f": func(f float32) string {
|
|
return fmt.Sprintf("%.3f", f)
|
|
},
|
|
"divU64": func(a, b uint64) float32 {
|
|
if b == 0 {
|
|
if a > 0 {
|
|
return 100
|
|
}
|
|
return 0
|
|
}
|
|
if a == 0 {
|
|
// 这是从未在线的情况
|
|
return 0.00001 / float32(b) * 100
|
|
}
|
|
return float32(a) / float32(b) * 100
|
|
},
|
|
"div": func(a, b int) float32 {
|
|
if b == 0 {
|
|
if a > 0 {
|
|
return 100
|
|
}
|
|
return 0
|
|
}
|
|
if a == 0 {
|
|
// 这是从未在线的情况
|
|
return 0.00001 / float32(b) * 100
|
|
}
|
|
return float32(a) / float32(b) * 100
|
|
},
|
|
"addU64": func(a, b uint64) uint64 {
|
|
return a + b
|
|
},
|
|
"add": func(a, b int) int {
|
|
return a + b
|
|
},
|
|
"TransLeftPercent": func(a, b float64) (n float64) {
|
|
n, _ = strconv.ParseFloat(fmt.Sprintf("%.2f", (100-(a/b)*100)), 64)
|
|
if n < 0 {
|
|
n = 0
|
|
}
|
|
return
|
|
},
|
|
"TransLeft": func(a, b uint64) string {
|
|
if a < b {
|
|
return "0B"
|
|
}
|
|
return bytefmt.ByteSize(a - b)
|
|
},
|
|
"TransClassName": func(a float64) string {
|
|
if a == 0 {
|
|
return "offline"
|
|
}
|
|
if a > 50 {
|
|
return "fine"
|
|
}
|
|
if a > 20 {
|
|
return "warning"
|
|
}
|
|
if a > 0 {
|
|
return "error"
|
|
}
|
|
return "offline"
|
|
},
|
|
"UintToFloat": func(a uint64) (n float64) {
|
|
n, _ = strconv.ParseFloat((strconv.FormatUint(a, 10)), 64)
|
|
return
|
|
},
|
|
"dayBefore": func(i int) string {
|
|
year, month, day := time.Now().Date()
|
|
today := time.Date(year, month, day, 0, 0, 0, 0, singleton.Loc)
|
|
return today.AddDate(0, 0, i-29).Format("01/02")
|
|
},
|
|
"className": func(percent float32) string {
|
|
if percent == 0 {
|
|
return ""
|
|
}
|
|
if percent > 95 {
|
|
return "good"
|
|
}
|
|
if percent > 80 {
|
|
return "warning"
|
|
}
|
|
return "danger"
|
|
},
|
|
"statusName": func(val float32) string {
|
|
return singleton.StatusCodeToString(singleton.GetStatusCode(val))
|
|
},
|
|
}
|
|
|
|
func natGateway(c *gin.Context) {
|
|
natConfig := singleton.GetNATConfigByDomain(c.Request.Host)
|
|
if natConfig == nil {
|
|
return
|
|
}
|
|
|
|
singleton.ServerLock.RLock()
|
|
server := singleton.ServerList[natConfig.ServerID]
|
|
singleton.ServerLock.RUnlock()
|
|
if server == nil || server.TaskStream == nil {
|
|
c.Writer.WriteString("server not found or not connected")
|
|
c.Abort()
|
|
return
|
|
}
|
|
|
|
streamId, err := uuid.GenerateUUID()
|
|
if err != nil {
|
|
c.Writer.WriteString(fmt.Sprintf("stream id error: %v", err))
|
|
c.Abort()
|
|
return
|
|
}
|
|
|
|
rpc.NezhaHandlerSingleton.CreateStream(streamId)
|
|
defer rpc.NezhaHandlerSingleton.CloseStream(streamId)
|
|
|
|
taskData, err := utils.Json.Marshal(model.TaskNAT{
|
|
StreamID: streamId,
|
|
Host: natConfig.Host,
|
|
})
|
|
if err != nil {
|
|
c.Writer.WriteString(fmt.Sprintf("task data error: %v", err))
|
|
c.Abort()
|
|
return
|
|
}
|
|
|
|
if err := server.TaskStream.Send(&proto.Task{
|
|
Type: model.TaskTypeNAT,
|
|
Data: string(taskData),
|
|
}); err != nil {
|
|
c.Writer.WriteString(fmt.Sprintf("send task error: %v", err))
|
|
c.Abort()
|
|
return
|
|
}
|
|
|
|
w, err := utils.NewRequestWrapper(c.Request, c.Writer)
|
|
if err != nil {
|
|
c.Writer.WriteString(fmt.Sprintf("request wrapper error: %v", err))
|
|
c.Abort()
|
|
return
|
|
}
|
|
|
|
if err := rpc.NezhaHandlerSingleton.UserConnected(streamId, w); err != nil {
|
|
c.Writer.WriteString(fmt.Sprintf("user connected error: %v", err))
|
|
c.Abort()
|
|
return
|
|
}
|
|
|
|
rpc.NezhaHandlerSingleton.StartStream(streamId, time.Second*10)
|
|
c.Abort()
|
|
}
|