2022-03-29 20:53:43 +08:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
2023-01-03 23:05:42 +08:00
|
|
|
"context"
|
2022-03-29 20:53:43 +08:00
|
|
|
"encoding/json"
|
|
|
|
"net/http"
|
2022-11-18 21:17:52 +08:00
|
|
|
"os"
|
2022-03-29 20:53:43 +08:00
|
|
|
|
2023-01-03 23:05:42 +08:00
|
|
|
"github.com/google/uuid"
|
2022-06-27 22:09:06 +08:00
|
|
|
"github.com/usememos/memos/api"
|
2022-09-09 20:00:04 +08:00
|
|
|
"github.com/usememos/memos/common"
|
2023-04-04 08:31:11 +08:00
|
|
|
"github.com/usememos/memos/common/log"
|
|
|
|
"go.uber.org/zap"
|
2022-06-27 22:09:06 +08:00
|
|
|
|
2022-03-29 20:53:43 +08:00
|
|
|
"github.com/labstack/echo/v4"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (s *Server) registerSystemRoutes(g *echo.Group) {
|
|
|
|
g.GET("/ping", func(c echo.Context) error {
|
2023-02-17 23:55:56 +08:00
|
|
|
return c.JSON(http.StatusOK, composeResponse(s.Profile))
|
2022-03-29 20:53:43 +08:00
|
|
|
})
|
2022-05-15 10:57:54 +08:00
|
|
|
|
|
|
|
g.GET("/status", func(c echo.Context) error {
|
2022-08-07 09:23:46 +08:00
|
|
|
ctx := c.Request().Context()
|
2022-07-08 22:16:18 +08:00
|
|
|
hostUserType := api.Host
|
|
|
|
hostUserFind := api.UserFind{
|
|
|
|
Role: &hostUserType,
|
2022-05-15 10:57:54 +08:00
|
|
|
}
|
2022-08-07 09:23:46 +08:00
|
|
|
hostUser, err := s.Store.FindUser(ctx, &hostUserFind)
|
2022-09-09 20:00:04 +08:00
|
|
|
if err != nil && common.ErrorCode(err) != common.NotFound {
|
2022-07-08 22:16:18 +08:00
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find host user").SetInternal(err)
|
2022-05-15 10:57:54 +08:00
|
|
|
}
|
|
|
|
|
2022-07-08 22:16:18 +08:00
|
|
|
if hostUser != nil {
|
2022-06-27 19:11:56 +08:00
|
|
|
// data desensitize
|
2022-07-08 22:16:18 +08:00
|
|
|
hostUser.OpenID = ""
|
2022-12-25 10:28:51 +08:00
|
|
|
hostUser.Email = ""
|
2022-06-27 19:11:56 +08:00
|
|
|
}
|
2022-06-22 19:16:31 +08:00
|
|
|
|
2022-05-15 10:57:54 +08:00
|
|
|
systemStatus := api.SystemStatus{
|
2023-02-14 00:07:31 +08:00
|
|
|
Host: hostUser,
|
|
|
|
Profile: *s.Profile,
|
|
|
|
DBSize: 0,
|
|
|
|
AllowSignUp: false,
|
2023-04-08 18:13:51 +08:00
|
|
|
IgnoreUpgrade: false,
|
2023-02-14 00:07:31 +08:00
|
|
|
DisablePublicMemos: false,
|
2023-05-15 22:59:26 +08:00
|
|
|
MaxUploadSizeMiB: 32,
|
2023-02-14 00:07:31 +08:00
|
|
|
AdditionalStyle: "",
|
|
|
|
AdditionalScript: "",
|
2022-12-18 21:18:30 +08:00
|
|
|
CustomizedProfile: api.CustomizedProfile{
|
2022-12-22 19:48:44 +08:00
|
|
|
Name: "memos",
|
2022-12-23 00:21:53 +08:00
|
|
|
LogoURL: "",
|
2022-12-22 19:48:44 +08:00
|
|
|
Description: "",
|
|
|
|
Locale: "en",
|
|
|
|
Appearance: "system",
|
|
|
|
ExternalURL: "",
|
2022-12-18 21:18:30 +08:00
|
|
|
},
|
2023-04-03 14:13:22 +08:00
|
|
|
StorageServiceID: api.DatabaseStorage,
|
2023-05-15 22:59:26 +08:00
|
|
|
LocalStoragePath: "assets/{timestamp}_{filename}",
|
2022-11-03 21:47:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
systemSettingList, err := s.Store.FindSystemSettingList(ctx, &api.SystemSettingFind{})
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find system setting list").SetInternal(err)
|
|
|
|
}
|
|
|
|
for _, systemSetting := range systemSettingList {
|
2023-04-03 09:36:34 +08:00
|
|
|
if systemSetting.Name == api.SystemSettingServerIDName || systemSetting.Name == api.SystemSettingSecretSessionName || systemSetting.Name == api.SystemSettingOpenAIConfigName {
|
2023-01-03 23:05:42 +08:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-03-18 22:34:22 +08:00
|
|
|
var baseValue any
|
2023-03-11 12:26:40 +08:00
|
|
|
err := json.Unmarshal([]byte(systemSetting.Value), &baseValue)
|
2022-11-03 21:47:36 +08:00
|
|
|
if err != nil {
|
2023-04-04 08:31:11 +08:00
|
|
|
log.Warn("Failed to unmarshal system setting value", zap.String("setting name", systemSetting.Name.String()))
|
|
|
|
continue
|
2022-11-03 21:47:36 +08:00
|
|
|
}
|
|
|
|
|
2023-05-13 22:27:28 +08:00
|
|
|
switch systemSetting.Name {
|
|
|
|
case api.SystemSettingAllowSignUpName:
|
2023-03-11 12:26:40 +08:00
|
|
|
systemStatus.AllowSignUp = baseValue.(bool)
|
2023-05-13 22:27:28 +08:00
|
|
|
|
|
|
|
case api.SystemSettingIgnoreUpgradeName:
|
2023-04-08 18:13:51 +08:00
|
|
|
systemStatus.IgnoreUpgrade = baseValue.(bool)
|
2023-05-13 22:27:28 +08:00
|
|
|
|
|
|
|
case api.SystemSettingDisablePublicMemosName:
|
2023-03-11 12:26:40 +08:00
|
|
|
systemStatus.DisablePublicMemos = baseValue.(bool)
|
2023-05-13 22:27:28 +08:00
|
|
|
|
|
|
|
case api.SystemSettingMaxUploadSizeMiBName:
|
|
|
|
systemStatus.MaxUploadSizeMiB = int(baseValue.(float64))
|
|
|
|
|
|
|
|
case api.SystemSettingAdditionalStyleName:
|
2023-03-11 12:26:40 +08:00
|
|
|
systemStatus.AdditionalStyle = baseValue.(string)
|
2023-05-13 22:27:28 +08:00
|
|
|
|
|
|
|
case api.SystemSettingAdditionalScriptName:
|
2023-03-11 12:26:40 +08:00
|
|
|
systemStatus.AdditionalScript = baseValue.(string)
|
2023-05-13 22:27:28 +08:00
|
|
|
|
|
|
|
case api.SystemSettingCustomizedProfileName:
|
2023-03-11 12:26:40 +08:00
|
|
|
customizedProfile := api.CustomizedProfile{}
|
2023-05-13 22:27:28 +08:00
|
|
|
if err := json.Unmarshal([]byte(systemSetting.Value), &customizedProfile); err != nil {
|
2023-03-11 12:26:40 +08:00
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to unmarshal system setting customized profile value").SetInternal(err)
|
2022-12-18 21:18:30 +08:00
|
|
|
}
|
2023-03-11 12:26:40 +08:00
|
|
|
systemStatus.CustomizedProfile = customizedProfile
|
2023-05-13 22:27:28 +08:00
|
|
|
|
|
|
|
case api.SystemSettingStorageServiceIDName:
|
2023-03-11 12:26:40 +08:00
|
|
|
systemStatus.StorageServiceID = int(baseValue.(float64))
|
2023-05-13 22:27:28 +08:00
|
|
|
|
|
|
|
case api.SystemSettingLocalStoragePathName:
|
2023-03-19 19:37:57 +08:00
|
|
|
systemStatus.LocalStoragePath = baseValue.(string)
|
2023-05-13 22:27:28 +08:00
|
|
|
|
|
|
|
default:
|
|
|
|
log.Warn("Unknown system setting name", zap.String("setting name", systemSetting.Name.String()))
|
2022-11-03 21:47:36 +08:00
|
|
|
}
|
2022-05-15 10:57:54 +08:00
|
|
|
}
|
|
|
|
|
2022-11-19 17:07:40 +08:00
|
|
|
userID, ok := c.Get(getUserIDContextKey()).(int)
|
2022-12-10 13:23:38 +08:00
|
|
|
// Get database size for host user.
|
2022-11-19 17:07:40 +08:00
|
|
|
if ok {
|
|
|
|
user, err := s.Store.FindUser(ctx, &api.UserFind{
|
|
|
|
ID: &userID,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find user").SetInternal(err)
|
|
|
|
}
|
|
|
|
if user != nil && user.Role == api.Host {
|
|
|
|
fi, err := os.Stat(s.Profile.DSN)
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to read database fileinfo").SetInternal(err)
|
|
|
|
}
|
|
|
|
systemStatus.DBSize = fi.Size()
|
|
|
|
}
|
2022-11-18 21:17:52 +08:00
|
|
|
}
|
2023-02-17 23:55:56 +08:00
|
|
|
return c.JSON(http.StatusOK, composeResponse(systemStatus))
|
2022-05-15 10:57:54 +08:00
|
|
|
})
|
2022-11-03 21:47:36 +08:00
|
|
|
|
|
|
|
g.POST("/system/setting", func(c echo.Context) error {
|
|
|
|
ctx := c.Request().Context()
|
|
|
|
userID, ok := c.Get(getUserIDContextKey()).(int)
|
|
|
|
if !ok {
|
|
|
|
return echo.NewHTTPError(http.StatusUnauthorized, "Missing user in session")
|
|
|
|
}
|
|
|
|
|
|
|
|
user, err := s.Store.FindUser(ctx, &api.UserFind{
|
|
|
|
ID: &userID,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find user").SetInternal(err)
|
|
|
|
}
|
2022-12-28 20:22:52 +08:00
|
|
|
if user == nil || user.Role != api.Host {
|
2022-11-03 21:47:36 +08:00
|
|
|
return echo.NewHTTPError(http.StatusUnauthorized, "Unauthorized")
|
|
|
|
}
|
|
|
|
|
|
|
|
systemSettingUpsert := &api.SystemSettingUpsert{}
|
|
|
|
if err := json.NewDecoder(c.Request().Body).Decode(systemSettingUpsert); err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusBadRequest, "Malformatted post system setting request").SetInternal(err)
|
|
|
|
}
|
|
|
|
if err := systemSettingUpsert.Validate(); err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusBadRequest, "system setting invalidate").SetInternal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
systemSetting, err := s.Store.UpsertSystemSetting(ctx, systemSettingUpsert)
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to upsert system setting").SetInternal(err)
|
|
|
|
}
|
2023-02-17 23:55:56 +08:00
|
|
|
return c.JSON(http.StatusOK, composeResponse(systemSetting))
|
2022-11-03 21:47:36 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
g.GET("/system/setting", func(c echo.Context) error {
|
|
|
|
ctx := c.Request().Context()
|
2023-02-24 08:31:54 +08:00
|
|
|
userID, ok := c.Get(getUserIDContextKey()).(int)
|
|
|
|
if !ok {
|
|
|
|
return echo.NewHTTPError(http.StatusUnauthorized, "Missing user in session")
|
|
|
|
}
|
|
|
|
|
|
|
|
user, err := s.Store.FindUser(ctx, &api.UserFind{
|
|
|
|
ID: &userID,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find user").SetInternal(err)
|
|
|
|
}
|
|
|
|
if user == nil || user.Role != api.Host {
|
|
|
|
return echo.NewHTTPError(http.StatusUnauthorized, "Unauthorized")
|
|
|
|
}
|
|
|
|
|
2022-11-03 21:47:36 +08:00
|
|
|
systemSettingList, err := s.Store.FindSystemSettingList(ctx, &api.SystemSettingFind{})
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find system setting list").SetInternal(err)
|
|
|
|
}
|
2023-02-17 23:55:56 +08:00
|
|
|
return c.JSON(http.StatusOK, composeResponse(systemSettingList))
|
2022-11-03 21:47:36 +08:00
|
|
|
})
|
2022-12-07 22:45:47 +08:00
|
|
|
|
|
|
|
g.POST("/system/vacuum", func(c echo.Context) error {
|
|
|
|
ctx := c.Request().Context()
|
|
|
|
userID, ok := c.Get(getUserIDContextKey()).(int)
|
|
|
|
if !ok {
|
|
|
|
return echo.NewHTTPError(http.StatusUnauthorized, "Missing user in session")
|
|
|
|
}
|
2023-02-24 08:31:54 +08:00
|
|
|
|
2022-12-07 22:45:47 +08:00
|
|
|
user, err := s.Store.FindUser(ctx, &api.UserFind{
|
|
|
|
ID: &userID,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find user").SetInternal(err)
|
|
|
|
}
|
|
|
|
if user == nil || user.Role != api.Host {
|
|
|
|
return echo.NewHTTPError(http.StatusUnauthorized, "Unauthorized")
|
|
|
|
}
|
2023-02-24 08:31:54 +08:00
|
|
|
|
2022-12-07 22:45:47 +08:00
|
|
|
if err := s.Store.Vacuum(ctx); err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to vacuum database").SetInternal(err)
|
|
|
|
}
|
2023-02-17 23:55:56 +08:00
|
|
|
return c.JSON(http.StatusOK, true)
|
2022-12-07 22:45:47 +08:00
|
|
|
})
|
2022-03-29 20:53:43 +08:00
|
|
|
}
|
2023-01-03 23:05:42 +08:00
|
|
|
|
|
|
|
func (s *Server) getSystemServerID(ctx context.Context) (string, error) {
|
|
|
|
serverIDValue, err := s.Store.FindSystemSetting(ctx, &api.SystemSettingFind{
|
2023-04-03 09:36:34 +08:00
|
|
|
Name: api.SystemSettingServerIDName,
|
2023-01-03 23:05:42 +08:00
|
|
|
})
|
|
|
|
if err != nil && common.ErrorCode(err) != common.NotFound {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if serverIDValue == nil || serverIDValue.Value == "" {
|
|
|
|
serverIDValue, err = s.Store.UpsertSystemSetting(ctx, &api.SystemSettingUpsert{
|
2023-04-03 09:36:34 +08:00
|
|
|
Name: api.SystemSettingServerIDName,
|
2023-01-03 23:05:42 +08:00
|
|
|
Value: uuid.NewString(),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return serverIDValue.Value, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) getSystemSecretSessionName(ctx context.Context) (string, error) {
|
|
|
|
secretSessionNameValue, err := s.Store.FindSystemSetting(ctx, &api.SystemSettingFind{
|
2023-02-15 22:54:46 +08:00
|
|
|
Name: api.SystemSettingSecretSessionName,
|
2023-01-03 23:05:42 +08:00
|
|
|
})
|
|
|
|
if err != nil && common.ErrorCode(err) != common.NotFound {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if secretSessionNameValue == nil || secretSessionNameValue.Value == "" {
|
|
|
|
secretSessionNameValue, err = s.Store.UpsertSystemSetting(ctx, &api.SystemSettingUpsert{
|
2023-02-15 22:54:46 +08:00
|
|
|
Name: api.SystemSettingSecretSessionName,
|
2023-01-03 23:05:42 +08:00
|
|
|
Value: uuid.NewString(),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return secretSessionNameValue.Value, nil
|
|
|
|
}
|