205 lines
5.5 KiB
Go
205 lines
5.5 KiB
Go
|
package controller
|
|||
|
|
|||
|
import (
|
|||
|
"errors"
|
|||
|
"fmt"
|
|||
|
"html/template"
|
|||
|
"io"
|
|||
|
"net"
|
|||
|
"net/http"
|
|||
|
"strconv"
|
|||
|
"time"
|
|||
|
|
|||
|
"github.com/labstack/echo/v4"
|
|||
|
"github.com/libretro/netplay-lobby-server-go/domain"
|
|||
|
"github.com/libretro/netplay-lobby-server-go/model/entity"
|
|||
|
)
|
|||
|
|
|||
|
// Template 抽象模板渲染。
|
|||
|
type Template struct {
|
|||
|
templates *template.Template
|
|||
|
}
|
|||
|
|
|||
|
// Render 实现 echo 模板渲染接口
|
|||
|
func (t *Template) Render(w io.Writer, name string, data interface{}, c echo.Context) error {
|
|||
|
return t.templates.ExecuteTemplate(w, name, data)
|
|||
|
}
|
|||
|
|
|||
|
// SessionDomain 接口,用于将控制器逻辑与领域代码解耦。
|
|||
|
type SessionDomain interface {
|
|||
|
Add(request *domain.AddSessionRequest, ip net.IP) (*entity.Session, error)
|
|||
|
Get(roomID int32) (*entity.Session, error)
|
|||
|
List() ([]entity.Session, error)
|
|||
|
GetTunnel(tunnelName string) *domain.MitmInfo
|
|||
|
PurgeOld() error
|
|||
|
}
|
|||
|
|
|||
|
// ListSessionsResponse 是一个自定义的 DTO,用于向后兼容。
|
|||
|
type SessionsResponse struct {
|
|||
|
Fields entity.Session `json:"fields"`
|
|||
|
}
|
|||
|
|
|||
|
// SessionController 处理所有与会话相关的请求
|
|||
|
type SessionController struct {
|
|||
|
sessionDomain SessionDomain
|
|||
|
}
|
|||
|
|
|||
|
// NewSessionController 返回一个新的会话控制器
|
|||
|
func NewSessionController(sessionDomain SessionDomain) *SessionController {
|
|||
|
return &SessionController{sessionDomain}
|
|||
|
}
|
|||
|
|
|||
|
// RegisterRoutes 在 echo 框架实例中注册所有控制器路由。
|
|||
|
func (c *SessionController) RegisterRoutes(server *echo.Echo) {
|
|||
|
server.POST("/add", c.Add)
|
|||
|
server.POST("/add/", c.Add) // 旧路径
|
|||
|
server.GET("/list", c.List)
|
|||
|
server.GET("/list/", c.List) // 旧路径
|
|||
|
server.GET("/tunnel", c.Tunnel)
|
|||
|
server.GET("/tunnel/", c.Tunnel) // 旧路径
|
|||
|
server.GET("/", c.Index)
|
|||
|
server.GET("/:roomID", c.Get)
|
|||
|
server.GET("/:roomID/", c.Get) // 旧路径
|
|||
|
}
|
|||
|
|
|||
|
// PrerenderTemplates 预渲染所有模板
|
|||
|
func (c *SessionController) PrerenderTemplates(server *echo.Echo, filePattern string) error {
|
|||
|
templates, err := template.New("").Funcs(
|
|||
|
template.FuncMap{
|
|||
|
"prettyBool": func(b bool) string {
|
|||
|
if b {
|
|||
|
return "Yes"
|
|||
|
}
|
|||
|
return "No"
|
|||
|
},
|
|||
|
"prettyDate": func(d time.Time) string {
|
|||
|
utc, _ := time.LoadLocation("UTC")
|
|||
|
return d.In(utc).Format(time.RFC822)
|
|||
|
},
|
|||
|
},
|
|||
|
).ParseGlob(filePattern)
|
|||
|
|
|||
|
if err != nil {
|
|||
|
return fmt.Errorf("无法解析模板: %w", err)
|
|||
|
}
|
|||
|
|
|||
|
t := &Template{
|
|||
|
templates: templates,
|
|||
|
}
|
|||
|
server.Renderer = t
|
|||
|
return nil
|
|||
|
}
|
|||
|
|
|||
|
// Index 处理器
|
|||
|
// GET /
|
|||
|
func (c *SessionController) Index(ctx echo.Context) error {
|
|||
|
logger := ctx.Logger()
|
|||
|
|
|||
|
sessions, err := c.sessionDomain.List()
|
|||
|
if err != nil {
|
|||
|
logger.Errorf("无法渲染会话列表: %v", err)
|
|||
|
return ctx.NoContent(http.StatusInternalServerError)
|
|||
|
}
|
|||
|
|
|||
|
return ctx.Render(http.StatusOK, "index.html", sessions)
|
|||
|
}
|
|||
|
|
|||
|
// Get 处理器
|
|||
|
// GET /:roomID
|
|||
|
func (c *SessionController) Get(ctx echo.Context) error {
|
|||
|
logger := ctx.Logger()
|
|||
|
|
|||
|
roomIDString := ctx.Param("roomID")
|
|||
|
roomID, err := strconv.ParseInt(roomIDString, 10, 32)
|
|||
|
if err != nil {
|
|||
|
logger.Errorf("无法通过 roomID 获取会话。RoomID 不是有效的 int32: %v", err)
|
|||
|
return ctx.NoContent(http.StatusBadRequest)
|
|||
|
}
|
|||
|
|
|||
|
session, err := c.sessionDomain.Get(int32(roomID))
|
|||
|
if err != nil || session == nil {
|
|||
|
logger.Errorf("无法获取会话: %v", err)
|
|||
|
return ctx.NoContent(http.StatusNotFound)
|
|||
|
}
|
|||
|
|
|||
|
// 出于兼容旧实现的原因,我们需要将会话放在一个包装对象中,
|
|||
|
// 该对象的会话可以通过键 "fields" 访问。旧实现还返回一个条目的列表...
|
|||
|
response := make([]SessionsResponse, 1)
|
|||
|
response[0] = SessionsResponse{*session}
|
|||
|
return ctx.JSONPretty(http.StatusOK, response, " ")
|
|||
|
}
|
|||
|
|
|||
|
// List 处理器
|
|||
|
// GET /list
|
|||
|
func (c *SessionController) List(ctx echo.Context) error {
|
|||
|
logger := ctx.Logger()
|
|||
|
|
|||
|
sessions, err := c.sessionDomain.List()
|
|||
|
if err != nil {
|
|||
|
logger.Errorf("无法渲染会话列表: %v", err)
|
|||
|
return ctx.NoContent(http.StatusInternalServerError)
|
|||
|
}
|
|||
|
|
|||
|
// 出于兼容旧实现的原因,我们需要将会话放在一个包装对象中,
|
|||
|
// 该对象的会话可以通过键 "fields" 访问
|
|||
|
response := make([]SessionsResponse, len(sessions))
|
|||
|
for i, session := range sessions {
|
|||
|
response[i].Fields = session
|
|||
|
}
|
|||
|
|
|||
|
return ctx.JSONPretty(http.StatusOK, response, " ")
|
|||
|
}
|
|||
|
|
|||
|
// Add 处理器
|
|||
|
// POST /add
|
|||
|
func (c *SessionController) Add(ctx echo.Context) error {
|
|||
|
logger := ctx.Logger()
|
|||
|
var err error
|
|||
|
var session *entity.Session
|
|||
|
|
|||
|
var req domain.AddSessionRequest
|
|||
|
if err := ctx.Bind(&req); err != nil {
|
|||
|
logger.Errorf("无法解析传入的会话: %v", err)
|
|||
|
return ctx.NoContent(http.StatusBadRequest)
|
|||
|
}
|
|||
|
|
|||
|
ip := net.ParseIP(ctx.RealIP())
|
|||
|
|
|||
|
if session, err = c.sessionDomain.Add(&req, ip); err != nil {
|
|||
|
logger.Errorf("不会添加会话: %v", err)
|
|||
|
|
|||
|
if errors.Is(err, domain.ErrSessionRejected) {
|
|||
|
logger.Errorf("会话被拒绝: %v", session)
|
|||
|
return ctx.NoContent(http.StatusBadRequest)
|
|||
|
} else if errors.Is(err, domain.ErrRateLimited) {
|
|||
|
return ctx.NoContent(http.StatusTooManyRequests)
|
|||
|
}
|
|||
|
return ctx.NoContent(http.StatusBadRequest)
|
|||
|
}
|
|||
|
|
|||
|
result := "status=OK\n"
|
|||
|
result += session.PrintForRetroarch()
|
|||
|
return ctx.String(http.StatusOK, result)
|
|||
|
}
|
|||
|
|
|||
|
// Tunnel 处理器
|
|||
|
// GET /tunnel
|
|||
|
func (c *SessionController) Tunnel(ctx echo.Context) error {
|
|||
|
logger := ctx.Logger()
|
|||
|
|
|||
|
tunnelName := ctx.QueryParam("name")
|
|||
|
if tunnelName == "" {
|
|||
|
return ctx.NoContent(http.StatusBadRequest)
|
|||
|
}
|
|||
|
|
|||
|
tunnel := c.sessionDomain.GetTunnel(tunnelName)
|
|||
|
if tunnel == nil {
|
|||
|
logger.Errorf("找不到隧道服务器: '%s'", tunnelName)
|
|||
|
return ctx.NoContent(http.StatusNotFound)
|
|||
|
}
|
|||
|
|
|||
|
result := "status=OK\n"
|
|||
|
result += tunnel.PrintForRetroarch()
|
|||
|
return ctx.String(http.StatusOK, result)
|
|||
|
}
|