2019-12-08 16:59:58 +08:00
|
|
|
package controller
|
|
|
|
|
|
|
|
import (
|
2024-10-21 14:30:50 +08:00
|
|
|
"errors"
|
2019-12-08 16:59:58 +08:00
|
|
|
"fmt"
|
2023-11-29 09:42:51 +08:00
|
|
|
"log"
|
2021-07-14 23:53:37 +08:00
|
|
|
"net/http"
|
2024-10-20 11:47:45 +08:00
|
|
|
"strings"
|
2019-12-08 16:59:58 +08:00
|
|
|
"time"
|
|
|
|
|
2024-10-20 00:09:16 +08:00
|
|
|
jwt "github.com/appleboy/gin-jwt/v2"
|
2021-05-10 18:04:38 +08:00
|
|
|
"github.com/gin-contrib/pprof"
|
2019-12-08 16:59:58 +08:00
|
|
|
"github.com/gin-gonic/gin"
|
2024-07-14 19:41:50 +08:00
|
|
|
"github.com/hashicorp/go-uuid"
|
2024-10-19 23:14:53 +08:00
|
|
|
swaggerfiles "github.com/swaggo/files"
|
|
|
|
ginSwagger "github.com/swaggo/gin-swagger"
|
2019-12-08 16:59:58 +08:00
|
|
|
|
2024-10-19 23:14:53 +08:00
|
|
|
docs "github.com/naiba/nezha/cmd/dashboard/docs"
|
2024-07-14 19:41:50 +08:00
|
|
|
"github.com/naiba/nezha/model"
|
|
|
|
"github.com/naiba/nezha/pkg/utils"
|
|
|
|
"github.com/naiba/nezha/proto"
|
|
|
|
"github.com/naiba/nezha/service/rpc"
|
2022-01-09 11:54:14 +08:00
|
|
|
"github.com/naiba/nezha/service/singleton"
|
2019-12-08 16:59:58 +08:00
|
|
|
)
|
|
|
|
|
2024-10-22 23:44:50 +08:00
|
|
|
func ServeWeb() http.Handler {
|
2019-12-08 23:18:29 +08:00
|
|
|
gin.SetMode(gin.ReleaseMode)
|
2021-05-10 18:04:38 +08:00
|
|
|
r := gin.Default()
|
2024-10-19 23:14:53 +08:00
|
|
|
docs.SwaggerInfo.BasePath = "/api/v1"
|
2024-07-14 19:41:50 +08:00
|
|
|
if singleton.Conf.Debug {
|
|
|
|
gin.SetMode(gin.DebugMode)
|
|
|
|
pprof.Register(r)
|
|
|
|
}
|
|
|
|
r.Use(natGateway)
|
2024-10-20 11:47:45 +08:00
|
|
|
if singleton.Conf.Debug {
|
|
|
|
r.GET("/swagger/*any", ginSwagger.WrapHandler(swaggerfiles.Handler))
|
|
|
|
}
|
2024-10-20 14:05:43 +08:00
|
|
|
|
2024-10-20 11:47:45 +08:00
|
|
|
r.Use(recordPath)
|
2019-12-08 16:59:58 +08:00
|
|
|
routers(r)
|
2021-08-10 20:13:17 +08:00
|
|
|
|
2024-10-22 23:44:50 +08:00
|
|
|
return r
|
2019-12-08 16:59:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func routers(r *gin.Engine) {
|
2024-10-20 00:09:16 +08:00
|
|
|
authMiddleware, err := jwt.New(initParams())
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal("JWT Error:" + err.Error())
|
|
|
|
}
|
2024-10-20 23:23:04 +08:00
|
|
|
if err := authMiddleware.MiddlewareInit(); err != nil {
|
|
|
|
log.Fatal("authMiddleware.MiddlewareInit Error:" + err.Error())
|
|
|
|
}
|
2024-10-20 14:05:43 +08:00
|
|
|
api := r.Group("api/v1")
|
|
|
|
api.POST("/login", authMiddleware.LoginHandler)
|
2024-10-20 00:09:16 +08:00
|
|
|
|
2024-10-21 14:30:50 +08:00
|
|
|
optionalAuth := api.Group("", optionalAuthMiddleware(authMiddleware))
|
2024-10-21 23:00:51 +08:00
|
|
|
optionalAuth.GET("/ws/server", commonHandler(serverStream))
|
|
|
|
optionalAuth.GET("/server-group", commonHandler(listServerGroup))
|
2024-10-20 23:23:04 +08:00
|
|
|
|
2024-10-20 14:05:43 +08:00
|
|
|
auth := api.Group("", authMiddleware.MiddlewareFunc())
|
2024-10-22 22:01:01 +08:00
|
|
|
|
2024-10-20 00:09:16 +08:00
|
|
|
auth.GET("/refresh_token", authMiddleware.RefreshHandler)
|
2024-10-21 23:00:51 +08:00
|
|
|
|
2024-10-22 22:01:01 +08:00
|
|
|
auth.POST("/terminal", commonHandler(createTerminal))
|
|
|
|
auth.GET("/ws/terminal/:id", commonHandler(terminalStream))
|
|
|
|
|
2024-10-22 21:19:30 +08:00
|
|
|
auth.GET("/user", commonHandler(listUser))
|
|
|
|
auth.POST("/user", commonHandler(createUser))
|
|
|
|
auth.POST("/batch-delete/user", commonHandler(batchDeleteUser))
|
|
|
|
|
|
|
|
auth.POST("/server-group", commonHandler(createServerGroup))
|
|
|
|
auth.PATCH("/server-group/:id", commonHandler(updateServerGroup))
|
2024-10-21 23:00:51 +08:00
|
|
|
auth.POST("/batch-delete/server-group", commonHandler(batchDeleteServerGroup))
|
|
|
|
|
2024-10-23 17:34:15 +08:00
|
|
|
auth.GET("/server", commonHandler(listServer))
|
2024-10-22 21:19:30 +08:00
|
|
|
auth.PATCH("/server/:id", commonHandler(updateServer))
|
2024-10-21 23:00:51 +08:00
|
|
|
auth.POST("/batch-delete/server", commonHandler(batchDeleteServer))
|
|
|
|
|
|
|
|
auth.GET("/ddns", commonHandler(listDDNS))
|
2024-10-22 00:04:17 +08:00
|
|
|
auth.GET("/ddns/providers", commonHandler(listProviders))
|
2024-10-22 21:19:30 +08:00
|
|
|
auth.POST("/ddns", commonHandler(createDDNS))
|
|
|
|
auth.PATCH("/ddns/:id", commonHandler(updateDDNS))
|
2024-10-21 23:00:51 +08:00
|
|
|
auth.POST("/batch-delete/ddns", commonHandler(batchDeleteDDNS))
|
2019-12-08 16:59:58 +08:00
|
|
|
}
|
2022-04-30 09:32:57 +08:00
|
|
|
|
2024-07-14 19:41:50 +08:00
|
|
|
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)
|
|
|
|
|
2024-08-24 11:11:06 +08:00
|
|
|
taskData, err := utils.Json.Marshal(model.TaskNAT{
|
2024-07-14 19:41:50 +08:00
|
|
|
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()
|
|
|
|
}
|
2024-10-20 11:47:45 +08:00
|
|
|
|
|
|
|
func recordPath(c *gin.Context) {
|
|
|
|
url := c.Request.URL.String()
|
|
|
|
for _, p := range c.Params {
|
|
|
|
url = strings.Replace(url, p.Value, ":"+p.Key, 1)
|
|
|
|
}
|
|
|
|
c.Set("MatchedPath", url)
|
|
|
|
}
|
2024-10-21 14:30:50 +08:00
|
|
|
|
2024-10-21 23:00:51 +08:00
|
|
|
func newErrorResponse(err error) model.CommonResponse[any] {
|
|
|
|
return model.CommonResponse[any]{
|
2024-10-21 14:30:50 +08:00
|
|
|
Success: false,
|
|
|
|
Error: err.Error(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type handlerFunc func(c *gin.Context) error
|
|
|
|
|
|
|
|
// There are many error types in gorm, so create a custom type to represent all
|
|
|
|
// gorm errors here instead
|
|
|
|
type gormError struct {
|
|
|
|
msg string
|
|
|
|
a []interface{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func newGormError(format string, args ...interface{}) error {
|
|
|
|
return &gormError{
|
|
|
|
msg: format,
|
|
|
|
a: args,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ge *gormError) Error() string {
|
|
|
|
return fmt.Sprintf(ge.msg, ge.a...)
|
|
|
|
}
|
|
|
|
|
2024-10-21 23:00:51 +08:00
|
|
|
func commonHandler(handler handlerFunc) func(*gin.Context) {
|
2024-10-21 14:30:50 +08:00
|
|
|
return func(c *gin.Context) {
|
|
|
|
if err := handler(c); err != nil {
|
|
|
|
if _, ok := err.(*gormError); ok {
|
|
|
|
log.Printf("NEZHA>> gorm error: %v", err)
|
2024-10-21 23:00:51 +08:00
|
|
|
c.JSON(http.StatusOK, newErrorResponse(errors.New("database error")))
|
2024-10-21 14:30:50 +08:00
|
|
|
return
|
|
|
|
} else {
|
2024-10-21 23:00:51 +08:00
|
|
|
c.JSON(http.StatusOK, newErrorResponse(err))
|
2024-10-21 14:30:50 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|