mirror of
				https://github.com/usememos/memos.git
				synced 2025-10-25 05:46:03 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			167 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			167 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package v1
 | |
| 
 | |
| import (
 | |
| 	"encoding/json"
 | |
| 	"net/http"
 | |
| 
 | |
| 	"github.com/labstack/echo/v4"
 | |
| 
 | |
| 	"github.com/usememos/memos/server/profile"
 | |
| 	"github.com/usememos/memos/store"
 | |
| )
 | |
| 
 | |
| type SystemStatus struct {
 | |
| 	Host    *User           `json:"host"`
 | |
| 	Profile profile.Profile `json:"profile"`
 | |
| 	DBSize  int64           `json:"dbSize"`
 | |
| 
 | |
| 	// System settings
 | |
| 	// Disable password login.
 | |
| 	DisablePasswordLogin bool `json:"disablePasswordLogin"`
 | |
| 	// Disable public memos.
 | |
| 	DisablePublicMemos bool `json:"disablePublicMemos"`
 | |
| 	// Max upload size.
 | |
| 	MaxUploadSizeMiB int `json:"maxUploadSizeMiB"`
 | |
| 	// Customized server profile, including server name and external url.
 | |
| 	CustomizedProfile CustomizedProfile `json:"customizedProfile"`
 | |
| 	// Storage service ID.
 | |
| 	StorageServiceID int32 `json:"storageServiceId"`
 | |
| 	// Local storage path.
 | |
| 	LocalStoragePath string `json:"localStoragePath"`
 | |
| 	// Memo display with updated timestamp.
 | |
| 	MemoDisplayWithUpdatedTs bool `json:"memoDisplayWithUpdatedTs"`
 | |
| }
 | |
| 
 | |
| func (s *APIV1Service) registerSystemRoutes(g *echo.Group) {
 | |
| 	g.GET("/ping", s.PingSystem)
 | |
| 	g.GET("/status", s.GetSystemStatus)
 | |
| 	g.POST("/system/vacuum", s.ExecVacuum)
 | |
| }
 | |
| 
 | |
| // PingSystem godoc
 | |
| //
 | |
| //	@Summary	Ping the system
 | |
| //	@Tags		system
 | |
| //	@Produce	json
 | |
| //	@Success	200	{boolean}	true	"If succeed to ping the system"
 | |
| //	@Router		/api/v1/ping [GET]
 | |
| func (*APIV1Service) PingSystem(c echo.Context) error {
 | |
| 	return c.JSON(http.StatusOK, true)
 | |
| }
 | |
| 
 | |
| // GetSystemStatus godoc
 | |
| //
 | |
| //	@Summary	Get system GetSystemStatus
 | |
| //	@Tags		system
 | |
| //	@Produce	json
 | |
| //	@Success	200	{object}	SystemStatus	"System GetSystemStatus"
 | |
| //	@Failure	401	{object}	nil				"Missing user in session | Unauthorized"
 | |
| //	@Failure	500	{object}	nil				"Failed to find host user | Failed to find system setting list | Failed to unmarshal system setting customized profile value"
 | |
| //	@Router		/api/v1/status [GET]
 | |
| func (s *APIV1Service) GetSystemStatus(c echo.Context) error {
 | |
| 	ctx := c.Request().Context()
 | |
| 
 | |
| 	systemStatus := SystemStatus{
 | |
| 		Profile: profile.Profile{
 | |
| 			Mode:    s.Profile.Mode,
 | |
| 			Version: s.Profile.Version,
 | |
| 		},
 | |
| 		MaxUploadSizeMiB: 32,
 | |
| 		CustomizedProfile: CustomizedProfile{
 | |
| 			Name:       "Memos",
 | |
| 			Locale:     "en",
 | |
| 			Appearance: "system",
 | |
| 		},
 | |
| 		StorageServiceID: DefaultStorage,
 | |
| 		LocalStoragePath: "assets/{timestamp}_{filename}",
 | |
| 	}
 | |
| 
 | |
| 	hostUserType := store.RoleHost
 | |
| 	hostUser, err := s.Store.GetUser(ctx, &store.FindUser{
 | |
| 		Role: &hostUserType,
 | |
| 	})
 | |
| 	if err != nil {
 | |
| 		return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find host user").SetInternal(err)
 | |
| 	}
 | |
| 	if hostUser != nil {
 | |
| 		systemStatus.Host = &User{ID: hostUser.ID}
 | |
| 	}
 | |
| 
 | |
| 	workspaceGeneralSetting, err := s.Store.GetWorkspaceGeneralSetting(ctx)
 | |
| 	if err != nil {
 | |
| 		return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find workspace general setting").SetInternal(err)
 | |
| 	}
 | |
| 	systemStatus.DisablePasswordLogin = workspaceGeneralSetting.DisallowPasswordLogin
 | |
| 
 | |
| 	systemSettingList, err := s.Store.ListWorkspaceSettings(ctx, &store.FindWorkspaceSetting{})
 | |
| 	if err != nil {
 | |
| 		return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find system setting list").SetInternal(err)
 | |
| 	}
 | |
| 	for _, systemSetting := range systemSettingList {
 | |
| 		if systemSetting.Name == SystemSettingServerIDName.String() || systemSetting.Name == SystemSettingSecretSessionName.String() || systemSetting.Name == SystemSettingTelegramBotTokenName.String() {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		var baseValue any
 | |
| 		err := json.Unmarshal([]byte(systemSetting.Value), &baseValue)
 | |
| 		if err != nil {
 | |
| 			// Skip invalid value.
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		switch systemSetting.Name {
 | |
| 		case SystemSettingDisablePublicMemosName.String():
 | |
| 			systemStatus.DisablePublicMemos = baseValue.(bool)
 | |
| 		case SystemSettingMaxUploadSizeMiBName.String():
 | |
| 			systemStatus.MaxUploadSizeMiB = int(baseValue.(float64))
 | |
| 		case SystemSettingCustomizedProfileName.String():
 | |
| 			customizedProfile := CustomizedProfile{}
 | |
| 			if err := json.Unmarshal([]byte(systemSetting.Value), &customizedProfile); err != nil {
 | |
| 				return echo.NewHTTPError(http.StatusInternalServerError, "Failed to unmarshal system setting customized profile value").SetInternal(err)
 | |
| 			}
 | |
| 			systemStatus.CustomizedProfile = customizedProfile
 | |
| 		case SystemSettingStorageServiceIDName.String():
 | |
| 			systemStatus.StorageServiceID = int32(baseValue.(float64))
 | |
| 		case SystemSettingLocalStoragePathName.String():
 | |
| 			systemStatus.LocalStoragePath = baseValue.(string)
 | |
| 		case SystemSettingMemoDisplayWithUpdatedTsName.String():
 | |
| 			systemStatus.MemoDisplayWithUpdatedTs = baseValue.(bool)
 | |
| 		default:
 | |
| 			// Skip unknown system setting.
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return c.JSON(http.StatusOK, systemStatus)
 | |
| }
 | |
| 
 | |
| // ExecVacuum godoc
 | |
| //
 | |
| //	@Summary	Vacuum the database
 | |
| //	@Tags		system
 | |
| //	@Produce	json
 | |
| //	@Success	200	{boolean}	true	"Database vacuumed"
 | |
| //	@Failure	401	{object}	nil		"Missing user in session | Unauthorized"
 | |
| //	@Failure	500	{object}	nil		"Failed to find user | Failed to ExecVacuum database"
 | |
| //	@Router		/api/v1/system/vacuum [POST]
 | |
| func (s *APIV1Service) ExecVacuum(c echo.Context) error {
 | |
| 	ctx := c.Request().Context()
 | |
| 	userID, ok := c.Get(userIDContextKey).(int32)
 | |
| 	if !ok {
 | |
| 		return echo.NewHTTPError(http.StatusUnauthorized, "Missing user in session")
 | |
| 	}
 | |
| 
 | |
| 	user, err := s.Store.GetUser(ctx, &store.FindUser{
 | |
| 		ID: &userID,
 | |
| 	})
 | |
| 	if err != nil {
 | |
| 		return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find user").SetInternal(err)
 | |
| 	}
 | |
| 	if user == nil || user.Role != store.RoleHost {
 | |
| 		return echo.NewHTTPError(http.StatusUnauthorized, "Unauthorized")
 | |
| 	}
 | |
| 
 | |
| 	if err := s.Store.Vacuum(ctx); err != nil {
 | |
| 		return echo.NewHTTPError(http.StatusInternalServerError, "Failed to vacuum database").SetInternal(err)
 | |
| 	}
 | |
| 	return c.JSON(http.StatusOK, true)
 | |
| }
 |