package fileserver import ( "bytes" "context" "encoding/base64" "fmt" "io" "net/http" "os" "path/filepath" "regexp" "strings" "time" "github.com/disintegration/imaging" "github.com/golang-jwt/jwt/v5" "github.com/labstack/echo/v4" "github.com/pkg/errors" "golang.org/x/sync/semaphore" "github.com/usememos/memos/internal/profile" "github.com/usememos/memos/internal/util" "github.com/usememos/memos/plugin/storage/s3" storepb "github.com/usememos/memos/proto/gen/store" apiv1 "github.com/usememos/memos/server/router/api/v1" "github.com/usememos/memos/store" ) const ( // ThumbnailCacheFolder is the folder name where the thumbnail images are stored. ThumbnailCacheFolder = ".thumbnail_cache" // thumbnailMaxSize is the maximum size in pixels for the largest dimension of the thumbnail image. thumbnailMaxSize = 600 ) var SupportedThumbnailMimeTypes = []string{ "image/png", "image/jpeg", } // FileServerService handles HTTP file serving with proper range request support. // This service bypasses gRPC-Gateway to use native HTTP serving via http.ServeContent(), // which is required for Safari video/audio playback. type FileServerService struct { Profile *profile.Profile Store *store.Store Secret string // thumbnailSemaphore limits concurrent thumbnail generation to prevent memory exhaustion thumbnailSemaphore *semaphore.Weighted } // NewFileServerService creates a new file server service. func NewFileServerService(profile *profile.Profile, store *store.Store, secret string) *FileServerService { return &FileServerService{ Profile: profile, Store: store, Secret: secret, thumbnailSemaphore: semaphore.NewWeighted(3), // Limit to 3 concurrent thumbnail generations } } // RegisterRoutes registers HTTP file serving routes. func (s *FileServerService) RegisterRoutes(echoServer *echo.Echo) { fileGroup := echoServer.Group("/file") // Serve attachment binary files fileGroup.GET("/attachments/:uid/:filename", s.serveAttachmentFile) // Serve user avatar images fileGroup.GET("/users/:identifier/avatar", s.serveUserAvatar) } // serveAttachmentFile serves attachment binary content using native HTTP. // This properly handles range requests required by Safari for video/audio playback. func (s *FileServerService) serveAttachmentFile(c echo.Context) error { ctx := c.Request().Context() uid := c.Param("uid") thumbnail := c.QueryParam("thumbnail") == "true" // Get attachment from database attachment, err := s.Store.GetAttachment(ctx, &store.FindAttachment{ UID: &uid, GetBlob: true, }) if err != nil { return echo.NewHTTPError(http.StatusInternalServerError, "failed to get attachment").SetInternal(err) } if attachment == nil { return echo.NewHTTPError(http.StatusNotFound, "attachment not found") } // Check permissions - verify memo visibility if attachment belongs to a memo if err := s.checkAttachmentPermission(ctx, c, attachment); err != nil { return err } // Get the binary content blob, err := s.getAttachmentBlob(attachment) if err != nil { return echo.NewHTTPError(http.StatusInternalServerError, "failed to get attachment blob").SetInternal(err) } // Handle thumbnail requests for images if thumbnail && s.isImageType(attachment.Type) { thumbnailBlob, err := s.getOrGenerateThumbnail(ctx, attachment) if err != nil { // Log warning but fall back to original image c.Logger().Warnf("failed to get thumbnail: %v", err) } else { blob = thumbnailBlob } } // Determine content type contentType := attachment.Type if strings.HasPrefix(contentType, "text/") { contentType += "; charset=utf-8" } // Prevent XSS attacks by serving potentially unsafe files as octet-stream if strings.EqualFold(contentType, "image/svg+xml") || strings.EqualFold(contentType, "text/html") || strings.EqualFold(contentType, "application/xhtml+xml") { contentType = "application/octet-stream" } // Set common headers c.Response().Header().Set("Content-Type", contentType) c.Response().Header().Set("Cache-Control", "public, max-age=3600") // For video/audio: Use http.ServeContent for automatic range request support // This is critical for Safari which REQUIRES range request support if strings.HasPrefix(contentType, "video/") || strings.HasPrefix(contentType, "audio/") { // ServeContent automatically handles: // - Range request parsing // - HTTP 206 Partial Content responses // - Content-Range headers // - Accept-Ranges: bytes header modTime := time.Unix(attachment.UpdatedTs, 0) http.ServeContent(c.Response(), c.Request(), attachment.Filename, modTime, bytes.NewReader(blob)) return nil } // For other files: Simple blob response return c.Blob(http.StatusOK, contentType, blob) } // serveUserAvatar serves user avatar images. // Supports both user ID and username as identifier. func (s *FileServerService) serveUserAvatar(c echo.Context) error { ctx := c.Request().Context() identifier := c.Param("identifier") // Try to find user by ID or username user, err := s.getUserByIdentifier(ctx, identifier) if err != nil { return echo.NewHTTPError(http.StatusInternalServerError, "failed to get user").SetInternal(err) } if user == nil { return echo.NewHTTPError(http.StatusNotFound, "user not found") } if user.AvatarURL == "" { return echo.NewHTTPError(http.StatusNotFound, "avatar not found") } // Extract image info from data URI imageType, base64Data, err := s.extractImageInfo(user.AvatarURL) if err != nil { return echo.NewHTTPError(http.StatusInternalServerError, "failed to extract image info").SetInternal(err) } // Decode base64 data imageData, err := base64.StdEncoding.DecodeString(base64Data) if err != nil { return echo.NewHTTPError(http.StatusInternalServerError, "failed to decode image data").SetInternal(err) } // Set cache headers for avatars c.Response().Header().Set("Content-Type", imageType) c.Response().Header().Set("Cache-Control", "public, max-age=3600") return c.Blob(http.StatusOK, imageType, imageData) } // getUserByIdentifier finds a user by either ID or username. func (s *FileServerService) getUserByIdentifier(ctx context.Context, identifier string) (*store.User, error) { // Try to parse as ID first if userID, err := util.ConvertStringToInt32(identifier); err == nil { return s.Store.GetUser(ctx, &store.FindUser{ID: &userID}) } // Otherwise, treat as username return s.Store.GetUser(ctx, &store.FindUser{Username: &identifier}) } // extractImageInfo extracts image type and base64 data from a data URI. // Data URI format: ... func (*FileServerService) extractImageInfo(dataURI string) (string, string, error) { dataURIRegex := regexp.MustCompile(`^data:(?P.+);base64,(?P.+)`) matches := dataURIRegex.FindStringSubmatch(dataURI) if len(matches) != 3 { return "", "", errors.New("invalid data URI format") } imageType := matches[1] base64Data := matches[2] return imageType, base64Data, nil } // checkAttachmentPermission verifies the user has permission to access the attachment. func (s *FileServerService) checkAttachmentPermission(ctx context.Context, c echo.Context, attachment *store.Attachment) error { // If attachment is not linked to a memo, allow access if attachment.MemoID == nil { return nil } // Check memo visibility memo, err := s.Store.GetMemo(ctx, &store.FindMemo{ ID: attachment.MemoID, }) if err != nil { return echo.NewHTTPError(http.StatusInternalServerError, "failed to find memo").SetInternal(err) } if memo == nil { return echo.NewHTTPError(http.StatusNotFound, "memo not found") } // Public memos are accessible to everyone if memo.Visibility == store.Public { return nil } // For non-public memos, check authentication user, err := s.getCurrentUser(ctx, c) if err != nil { return echo.NewHTTPError(http.StatusInternalServerError, "failed to get current user").SetInternal(err) } if user == nil { return echo.NewHTTPError(http.StatusUnauthorized, "unauthorized access") } // Private memos can only be accessed by the creator if memo.Visibility == store.Private && user.ID != attachment.CreatorID { return echo.NewHTTPError(http.StatusForbidden, "forbidden access") } return nil } // getCurrentUser retrieves the current authenticated user from the Echo context. // It checks both session cookies and Bearer tokens for authentication. func (s *FileServerService) getCurrentUser(ctx context.Context, c echo.Context) (*store.User, error) { // Try session cookie authentication first if cookie, err := c.Cookie(apiv1.SessionCookieName); err == nil && cookie.Value != "" { user, err := s.authenticateBySession(ctx, cookie.Value) if err == nil && user != nil { return user, nil } } // Try JWT Bearer token authentication authHeader := c.Request().Header.Get("Authorization") if authHeader != "" { parts := strings.Fields(authHeader) if len(parts) == 2 && strings.ToLower(parts[0]) == "bearer" { user, err := s.authenticateByJWT(ctx, parts[1]) if err == nil && user != nil { return user, nil } } } // No valid authentication found return nil, nil } // authenticateBySession authenticates a user using session ID from cookie. func (s *FileServerService) authenticateBySession(ctx context.Context, sessionCookieValue string) (*store.User, error) { if sessionCookieValue == "" { return nil, errors.New("session cookie value not found") } // Parse the cookie value to extract userID and sessionID userID, sessionID, err := s.parseSessionCookieValue(sessionCookieValue) if err != nil { return nil, errors.Wrap(err, "invalid session cookie format") } // Get the user user, err := s.Store.GetUser(ctx, &store.FindUser{ ID: &userID, }) if err != nil { return nil, errors.Wrap(err, "failed to get user") } if user == nil { return nil, errors.New("user not found") } if user.RowStatus == store.Archived { return nil, errors.New("user is archived") } // Get user sessions and validate the sessionID sessions, err := s.Store.GetUserSessions(ctx, user.ID) if err != nil { return nil, errors.Wrap(err, "failed to get user sessions") } if !s.validateUserSession(sessionID, sessions) { return nil, errors.New("invalid or expired session") } return user, nil } // authenticateByJWT authenticates a user using JWT access token from Authorization header. func (s *FileServerService) authenticateByJWT(ctx context.Context, accessToken string) (*store.User, error) { if accessToken == "" { return nil, errors.New("access token not found") } claims := &apiv1.ClaimsMessage{} _, err := jwt.ParseWithClaims(accessToken, claims, func(t *jwt.Token) (any, error) { if t.Method.Alg() != jwt.SigningMethodHS256.Name { return nil, errors.Errorf("unexpected access token signing method=%v, expect %v", t.Header["alg"], jwt.SigningMethodHS256) } if kid, ok := t.Header["kid"].(string); ok { if kid == apiv1.KeyID { return []byte(s.Secret), nil } } return nil, errors.Errorf("unexpected access token kid=%v", t.Header["kid"]) }) if err != nil { return nil, errors.Wrap(err, "Invalid or expired access token") } // Get user from JWT claims userID, err := util.ConvertStringToInt32(claims.Subject) if err != nil { return nil, errors.Wrap(err, "malformed ID in the token") } user, err := s.Store.GetUser(ctx, &store.FindUser{ ID: &userID, }) if err != nil { return nil, errors.Wrap(err, "failed to get user") } if user == nil { return nil, errors.Errorf("user %q not exists", userID) } if user.RowStatus == store.Archived { return nil, errors.Errorf("user %q is archived", userID) } // Validate that this access token exists in the user's access tokens accessTokens, err := s.Store.GetUserAccessTokens(ctx, user.ID) if err != nil { return nil, errors.Wrapf(err, "failed to get user access tokens") } if !s.validateAccessToken(accessToken, accessTokens) { return nil, errors.New("invalid access token") } return user, nil } // parseSessionCookieValue parses the session cookie value to extract userID and sessionID. func (*FileServerService) parseSessionCookieValue(cookieValue string) (int32, string, error) { parts := strings.SplitN(cookieValue, "-", 2) if len(parts) != 2 { return 0, "", errors.New("invalid session cookie format") } userID, err := util.ConvertStringToInt32(parts[0]) if err != nil { return 0, "", errors.Errorf("invalid user ID in session cookie: %v", err) } return userID, parts[1], nil } // validateUserSession checks if a session exists and is still valid using sliding expiration. func (*FileServerService) validateUserSession(sessionID string, userSessions []*storepb.SessionsUserSetting_Session) bool { for _, session := range userSessions { if sessionID == session.SessionId { // Use sliding expiration: check if last_accessed_time + 14 days > current_time if session.LastAccessedTime != nil { expirationTime := session.LastAccessedTime.AsTime().Add(apiv1.SessionSlidingDuration) if expirationTime.Before(time.Now()) { return false } } return true } } return false } // validateAccessToken checks if the provided JWT token exists in the user's access tokens list. func (*FileServerService) validateAccessToken(accessTokenString string, userAccessTokens []*storepb.AccessTokensUserSetting_AccessToken) bool { for _, userAccessToken := range userAccessTokens { if accessTokenString == userAccessToken.AccessToken { return true } } return false } // isImageType checks if the mime type is an image that supports thumbnails. func (*FileServerService) isImageType(mimeType string) bool { return mimeType == "image/png" || mimeType == "image/jpeg" } // getAttachmentBlob retrieves the binary content of an attachment from storage. func (s *FileServerService) getAttachmentBlob(attachment *store.Attachment) ([]byte, error) { // For local storage, read the file from the local disk. if attachment.StorageType == storepb.AttachmentStorageType_LOCAL { attachmentPath := filepath.FromSlash(attachment.Reference) if !filepath.IsAbs(attachmentPath) { attachmentPath = filepath.Join(s.Profile.Data, attachmentPath) } file, err := os.Open(attachmentPath) if err != nil { if os.IsNotExist(err) { return nil, errors.Wrap(err, "file not found") } return nil, errors.Wrap(err, "failed to open the file") } defer file.Close() blob, err := io.ReadAll(file) if err != nil { return nil, errors.Wrap(err, "failed to read the file") } return blob, nil } // For S3 storage, download the file from S3. if attachment.StorageType == storepb.AttachmentStorageType_S3 { if attachment.Payload == nil { return nil, errors.New("attachment payload is missing") } s3Object := attachment.Payload.GetS3Object() if s3Object == nil { return nil, errors.New("S3 object payload is missing") } if s3Object.S3Config == nil { return nil, errors.New("S3 config is missing") } if s3Object.Key == "" { return nil, errors.New("S3 object key is missing") } s3Client, err := s3.NewClient(context.Background(), s3Object.S3Config) if err != nil { return nil, errors.Wrap(err, "failed to create S3 client") } blob, err := s3Client.GetObject(context.Background(), s3Object.Key) if err != nil { return nil, errors.Wrap(err, "failed to get object from S3") } return blob, nil } // For database storage, return the blob from the database. return attachment.Blob, nil } // getOrGenerateThumbnail returns the thumbnail image of the attachment. // Uses semaphore to limit concurrent thumbnail generation and prevent memory exhaustion. func (s *FileServerService) getOrGenerateThumbnail(ctx context.Context, attachment *store.Attachment) ([]byte, error) { thumbnailCacheFolder := filepath.Join(s.Profile.Data, ThumbnailCacheFolder) if err := os.MkdirAll(thumbnailCacheFolder, os.ModePerm); err != nil { return nil, errors.Wrap(err, "failed to create thumbnail cache folder") } filePath := filepath.Join(thumbnailCacheFolder, fmt.Sprintf("%d%s", attachment.ID, filepath.Ext(attachment.Filename))) // Check if thumbnail already exists if _, err := os.Stat(filePath); err == nil { // Thumbnail exists, read and return it thumbnailFile, err := os.Open(filePath) if err != nil { return nil, errors.Wrap(err, "failed to open thumbnail file") } defer thumbnailFile.Close() blob, err := io.ReadAll(thumbnailFile) if err != nil { return nil, errors.Wrap(err, "failed to read thumbnail file") } return blob, nil } else if !os.IsNotExist(err) { return nil, errors.Wrap(err, "failed to check thumbnail image stat") } // Thumbnail doesn't exist, acquire semaphore to limit concurrent generation if err := s.thumbnailSemaphore.Acquire(ctx, 1); err != nil { return nil, errors.Wrap(err, "failed to acquire thumbnail generation semaphore") } defer s.thumbnailSemaphore.Release(1) // Double-check if thumbnail was created while waiting for semaphore if _, err := os.Stat(filePath); err == nil { thumbnailFile, err := os.Open(filePath) if err != nil { return nil, errors.Wrap(err, "failed to open thumbnail file") } defer thumbnailFile.Close() blob, err := io.ReadAll(thumbnailFile) if err != nil { return nil, errors.Wrap(err, "failed to read thumbnail file") } return blob, nil } // Generate the thumbnail blob, err := s.getAttachmentBlob(attachment) if err != nil { return nil, errors.Wrap(err, "failed to get attachment blob") } // Decode image - this is memory intensive img, err := imaging.Decode(bytes.NewReader(blob), imaging.AutoOrientation(true)) if err != nil { return nil, errors.Wrap(err, "failed to decode thumbnail image") } // The largest dimension is set to thumbnailMaxSize and the smaller dimension is scaled proportionally. // Small images are not enlarged. width := img.Bounds().Dx() height := img.Bounds().Dy() var thumbnailWidth, thumbnailHeight int // Only resize if the image is larger than thumbnailMaxSize if max(width, height) > thumbnailMaxSize { if width >= height { // Landscape or square - constrain width, maintain aspect ratio for height thumbnailWidth = thumbnailMaxSize thumbnailHeight = 0 } else { // Portrait - constrain height, maintain aspect ratio for width thumbnailWidth = 0 thumbnailHeight = thumbnailMaxSize } } else { // Keep original dimensions for small images thumbnailWidth = width thumbnailHeight = height } // Resize the image to the calculated dimensions. thumbnailImage := imaging.Resize(img, thumbnailWidth, thumbnailHeight, imaging.Lanczos) // Save thumbnail to disk if err := imaging.Save(thumbnailImage, filePath); err != nil { return nil, errors.Wrap(err, "failed to save thumbnail file") } // Read the saved thumbnail and return it thumbnailFile, err := os.Open(filePath) if err != nil { return nil, errors.Wrap(err, "failed to open thumbnail file") } defer thumbnailFile.Close() thumbnailBlob, err := io.ReadAll(thumbnailFile) if err != nil { return nil, errors.Wrap(err, "failed to read thumbnail file") } return thumbnailBlob, nil }