file_decoder.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. package service
  2. import (
  3. "encoding/base64"
  4. "fmt"
  5. "io"
  6. "one-api/common"
  7. "one-api/constant"
  8. "one-api/dto"
  9. "strings"
  10. )
  11. func GetFileBase64FromUrl(url string) (*dto.LocalFileData, error) {
  12. var maxFileSize = constant.MaxFileDownloadMB * 1024 * 1024
  13. resp, err := DoDownloadRequest(url)
  14. if err != nil {
  15. return nil, err
  16. }
  17. defer resp.Body.Close()
  18. // Always use LimitReader to prevent oversized downloads
  19. fileBytes, err := io.ReadAll(io.LimitReader(resp.Body, int64(maxFileSize+1)))
  20. if err != nil {
  21. return nil, err
  22. }
  23. // Check actual size after reading
  24. if len(fileBytes) > maxFileSize {
  25. return nil, fmt.Errorf("file size exceeds maximum allowed size: %dMB", constant.MaxFileDownloadMB)
  26. }
  27. // Convert to base64
  28. base64Data := base64.StdEncoding.EncodeToString(fileBytes)
  29. mimeType := resp.Header.Get("Content-Type")
  30. if mimeType == "application/octet-stream" {
  31. if common.DebugEnabled {
  32. println("MIME type is application/octet-stream, trying to guess from URL or filename")
  33. }
  34. // try to guess the MIME type from the url last segment
  35. urlParts := strings.Split(url, "/")
  36. if len(urlParts) > 0 {
  37. lastSegment := urlParts[len(urlParts)-1]
  38. if strings.Contains(lastSegment, ".") {
  39. // Extract the file extension
  40. filename := strings.Split(lastSegment, ".")
  41. if len(filename) > 1 {
  42. ext := strings.ToLower(filename[len(filename)-1])
  43. // Guess MIME type based on file extension
  44. mimeType = GetMimeTypeByExtension(ext)
  45. }
  46. }
  47. } else {
  48. // try to guess the MIME type from the file extension
  49. fileName := resp.Header.Get("Content-Disposition")
  50. if fileName != "" {
  51. // Extract the filename from the Content-Disposition header
  52. parts := strings.Split(fileName, ";")
  53. for _, part := range parts {
  54. if strings.HasPrefix(strings.TrimSpace(part), "filename=") {
  55. fileName = strings.TrimSpace(strings.TrimPrefix(part, "filename="))
  56. // Remove quotes if present
  57. if len(fileName) > 2 && fileName[0] == '"' && fileName[len(fileName)-1] == '"' {
  58. fileName = fileName[1 : len(fileName)-1]
  59. }
  60. // Guess MIME type based on file extension
  61. if ext := strings.ToLower(strings.TrimPrefix(fileName, ".")); ext != "" {
  62. mimeType = GetMimeTypeByExtension(ext)
  63. }
  64. break
  65. }
  66. }
  67. }
  68. }
  69. }
  70. return &dto.LocalFileData{
  71. Base64Data: base64Data,
  72. MimeType: mimeType,
  73. Size: int64(len(fileBytes)),
  74. }, nil
  75. }
  76. func GetMimeTypeByExtension(ext string) string {
  77. // Convert to lowercase for case-insensitive comparison
  78. ext = strings.ToLower(ext)
  79. switch ext {
  80. // Text files
  81. case "txt", "md", "markdown", "csv", "json", "xml", "html", "htm":
  82. return "text/plain"
  83. // Image files
  84. case "jpg", "jpeg":
  85. return "image/jpeg"
  86. case "png":
  87. return "image/png"
  88. case "gif":
  89. return "image/gif"
  90. // Audio files
  91. case "mp3":
  92. return "audio/mp3"
  93. case "wav":
  94. return "audio/wav"
  95. case "mpeg":
  96. return "audio/mpeg"
  97. // Video files
  98. case "mp4":
  99. return "video/mp4"
  100. case "wmv":
  101. return "video/wmv"
  102. case "flv":
  103. return "video/flv"
  104. case "mov":
  105. return "video/mov"
  106. case "mpg":
  107. return "video/mpg"
  108. case "avi":
  109. return "video/avi"
  110. case "mpegps":
  111. return "video/mpegps"
  112. // Document files
  113. case "pdf":
  114. return "application/pdf"
  115. default:
  116. return "application/octet-stream" // Default for unknown types
  117. }
  118. }