2024-02-18 02:12:02 -08:00
|
|
|
package openai
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2024-03-26 10:54:35 -07:00
|
|
|
"sync/atomic"
|
2024-02-18 02:12:02 -08:00
|
|
|
"time"
|
2024-02-20 20:21:19 -05:00
|
|
|
|
2024-10-12 03:45:47 -04:00
|
|
|
"github.com/microcosm-cc/bluemonday"
|
2024-06-23 01:24:36 -07:00
|
|
|
"github.com/mudler/LocalAI/core/config"
|
2024-07-18 06:38:41 +02:00
|
|
|
"github.com/mudler/LocalAI/core/schema"
|
2024-03-01 10:19:53 -05:00
|
|
|
|
2024-02-20 20:21:19 -05:00
|
|
|
"github.com/gofiber/fiber/v2"
|
2024-06-23 01:24:36 -07:00
|
|
|
"github.com/mudler/LocalAI/pkg/utils"
|
2024-02-18 02:12:02 -08:00
|
|
|
)
|
|
|
|
|
2024-07-18 06:38:41 +02:00
|
|
|
var UploadedFiles []schema.File
|
2024-02-18 02:12:02 -08:00
|
|
|
|
2024-03-26 10:54:35 -07:00
|
|
|
const UploadedFilesFile = "uploadedFiles.json"
|
2024-02-21 23:40:25 +01:00
|
|
|
|
2024-02-18 02:12:02 -08:00
|
|
|
// UploadFilesEndpoint https://platform.openai.com/docs/api-reference/files/create
|
2024-03-01 10:19:53 -05:00
|
|
|
func UploadFilesEndpoint(cm *config.BackendConfigLoader, appConfig *config.ApplicationConfig) func(c *fiber.Ctx) error {
|
2024-02-18 02:12:02 -08:00
|
|
|
return func(c *fiber.Ctx) error {
|
|
|
|
file, err := c.FormFile("file")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the file size
|
2024-03-01 10:19:53 -05:00
|
|
|
if file.Size > int64(appConfig.UploadLimitMB*1024*1024) {
|
|
|
|
return c.Status(fiber.StatusBadRequest).SendString(fmt.Sprintf("File size %d exceeds upload limit %d", file.Size, appConfig.UploadLimitMB))
|
2024-02-18 02:12:02 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
purpose := c.FormValue("purpose", "") //TODO put in purpose dirs
|
|
|
|
if purpose == "" {
|
|
|
|
return c.Status(fiber.StatusBadRequest).SendString("Purpose is not defined")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sanitize the filename to prevent directory traversal
|
|
|
|
filename := utils.SanitizeFileName(file.Filename)
|
|
|
|
|
2024-03-01 10:19:53 -05:00
|
|
|
savePath := filepath.Join(appConfig.UploadDir, filename)
|
2024-02-18 02:12:02 -08:00
|
|
|
|
|
|
|
// Check if file already exists
|
|
|
|
if _, err := os.Stat(savePath); !os.IsNotExist(err) {
|
|
|
|
return c.Status(fiber.StatusBadRequest).SendString("File already exists")
|
|
|
|
}
|
|
|
|
|
|
|
|
err = c.SaveFile(file, savePath)
|
|
|
|
if err != nil {
|
2024-10-12 03:45:47 -04:00
|
|
|
return c.Status(fiber.StatusInternalServerError).SendString("Failed to save file: " + bluemonday.StrictPolicy().Sanitize(err.Error()))
|
2024-02-18 02:12:02 -08:00
|
|
|
}
|
|
|
|
|
2024-07-18 06:38:41 +02:00
|
|
|
f := schema.File{
|
2024-03-26 10:54:35 -07:00
|
|
|
ID: fmt.Sprintf("file-%d", getNextFileId()),
|
2024-02-18 02:12:02 -08:00
|
|
|
Object: "file",
|
|
|
|
Bytes: int(file.Size),
|
|
|
|
CreatedAt: time.Now(),
|
|
|
|
Filename: file.Filename,
|
|
|
|
Purpose: purpose,
|
|
|
|
}
|
|
|
|
|
2024-03-26 10:54:35 -07:00
|
|
|
UploadedFiles = append(UploadedFiles, f)
|
|
|
|
utils.SaveConfig(appConfig.UploadDir, UploadedFilesFile, UploadedFiles)
|
2024-02-18 02:12:02 -08:00
|
|
|
return c.Status(fiber.StatusOK).JSON(f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-26 10:54:35 -07:00
|
|
|
var currentFileId int64 = 0
|
|
|
|
|
|
|
|
func getNextFileId() int64 {
|
|
|
|
atomic.AddInt64(¤tId, 1)
|
|
|
|
return currentId
|
|
|
|
}
|
|
|
|
|
2024-02-18 02:12:02 -08:00
|
|
|
// ListFilesEndpoint https://platform.openai.com/docs/api-reference/files/list
|
2024-07-18 06:38:41 +02:00
|
|
|
// @Summary List files.
|
|
|
|
// @Success 200 {object} schema.ListFiles "Response"
|
|
|
|
// @Router /v1/files [get]
|
2024-03-01 10:19:53 -05:00
|
|
|
func ListFilesEndpoint(cm *config.BackendConfigLoader, appConfig *config.ApplicationConfig) func(c *fiber.Ctx) error {
|
2024-02-18 02:12:02 -08:00
|
|
|
|
|
|
|
return func(c *fiber.Ctx) error {
|
2024-07-18 06:38:41 +02:00
|
|
|
var listFiles schema.ListFiles
|
2024-02-18 02:12:02 -08:00
|
|
|
|
|
|
|
purpose := c.Query("purpose")
|
|
|
|
if purpose == "" {
|
2024-03-26 10:54:35 -07:00
|
|
|
listFiles.Data = UploadedFiles
|
2024-02-18 02:12:02 -08:00
|
|
|
} else {
|
2024-03-26 10:54:35 -07:00
|
|
|
for _, f := range UploadedFiles {
|
2024-02-18 02:12:02 -08:00
|
|
|
if purpose == f.Purpose {
|
|
|
|
listFiles.Data = append(listFiles.Data, f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
listFiles.Object = "list"
|
|
|
|
return c.Status(fiber.StatusOK).JSON(listFiles)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-18 06:38:41 +02:00
|
|
|
func getFileFromRequest(c *fiber.Ctx) (*schema.File, error) {
|
2024-02-18 02:12:02 -08:00
|
|
|
id := c.Params("file_id")
|
|
|
|
if id == "" {
|
|
|
|
return nil, fmt.Errorf("file_id parameter is required")
|
|
|
|
}
|
|
|
|
|
2024-03-26 10:54:35 -07:00
|
|
|
for _, f := range UploadedFiles {
|
2024-02-18 02:12:02 -08:00
|
|
|
if id == f.ID {
|
|
|
|
return &f, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, fmt.Errorf("unable to find file id %s", id)
|
|
|
|
}
|
|
|
|
|
2024-07-09 23:09:49 +02:00
|
|
|
// GetFilesEndpoint is the OpenAI API endpoint to get files https://platform.openai.com/docs/api-reference/files/retrieve
|
|
|
|
// @Summary Returns information about a specific file.
|
2024-07-18 06:38:41 +02:00
|
|
|
// @Success 200 {object} schema.File "Response"
|
2024-07-09 23:09:49 +02:00
|
|
|
// @Router /v1/files/{file_id} [get]
|
2024-03-01 10:19:53 -05:00
|
|
|
func GetFilesEndpoint(cm *config.BackendConfigLoader, appConfig *config.ApplicationConfig) func(c *fiber.Ctx) error {
|
2024-02-18 02:12:02 -08:00
|
|
|
return func(c *fiber.Ctx) error {
|
|
|
|
file, err := getFileFromRequest(c)
|
|
|
|
if err != nil {
|
2024-10-12 03:45:47 -04:00
|
|
|
return c.Status(fiber.StatusInternalServerError).SendString(bluemonday.StrictPolicy().Sanitize(err.Error()))
|
2024-02-18 02:12:02 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
return c.JSON(file)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-09 23:09:49 +02:00
|
|
|
type DeleteStatus struct {
|
|
|
|
Id string
|
|
|
|
Object string
|
|
|
|
Deleted bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteFilesEndpoint is the OpenAI API endpoint to delete files https://platform.openai.com/docs/api-reference/files/delete
|
|
|
|
// @Summary Delete a file.
|
|
|
|
// @Success 200 {object} DeleteStatus "Response"
|
|
|
|
// @Router /v1/files/{file_id} [delete]
|
2024-03-01 10:19:53 -05:00
|
|
|
func DeleteFilesEndpoint(cm *config.BackendConfigLoader, appConfig *config.ApplicationConfig) func(c *fiber.Ctx) error {
|
2024-02-18 02:12:02 -08:00
|
|
|
|
|
|
|
return func(c *fiber.Ctx) error {
|
|
|
|
file, err := getFileFromRequest(c)
|
|
|
|
if err != nil {
|
2024-10-12 03:45:47 -04:00
|
|
|
return c.Status(fiber.StatusInternalServerError).SendString(bluemonday.StrictPolicy().Sanitize(err.Error()))
|
2024-02-18 02:12:02 -08:00
|
|
|
}
|
|
|
|
|
2024-03-01 10:19:53 -05:00
|
|
|
err = os.Remove(filepath.Join(appConfig.UploadDir, file.Filename))
|
2024-02-18 02:12:02 -08:00
|
|
|
if err != nil {
|
|
|
|
// If the file doesn't exist then we should just continue to remove it
|
|
|
|
if !errors.Is(err, os.ErrNotExist) {
|
2024-10-12 03:45:47 -04:00
|
|
|
return c.Status(fiber.StatusInternalServerError).SendString(bluemonday.StrictPolicy().Sanitize(fmt.Sprintf("Unable to delete file: %s, %v", file.Filename, err)))
|
2024-02-18 02:12:02 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove upload from list
|
2024-03-26 10:54:35 -07:00
|
|
|
for i, f := range UploadedFiles {
|
2024-02-18 02:12:02 -08:00
|
|
|
if f.ID == file.ID {
|
2024-03-26 10:54:35 -07:00
|
|
|
UploadedFiles = append(UploadedFiles[:i], UploadedFiles[i+1:]...)
|
2024-02-18 02:12:02 -08:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-26 10:54:35 -07:00
|
|
|
utils.SaveConfig(appConfig.UploadDir, UploadedFilesFile, UploadedFiles)
|
2024-02-18 02:12:02 -08:00
|
|
|
return c.JSON(DeleteStatus{
|
|
|
|
Id: file.ID,
|
|
|
|
Object: "file",
|
|
|
|
Deleted: true,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-09 23:09:49 +02:00
|
|
|
// GetFilesContentsEndpoint is the OpenAI API endpoint to get files content https://platform.openai.com/docs/api-reference/files/retrieve-contents
|
|
|
|
// @Summary Returns information about a specific file.
|
|
|
|
// @Success 200 {string} binary "file"
|
|
|
|
// @Router /v1/files/{file_id}/content [get]
|
|
|
|
// GetFilesContentsEndpoint
|
2024-03-01 10:19:53 -05:00
|
|
|
func GetFilesContentsEndpoint(cm *config.BackendConfigLoader, appConfig *config.ApplicationConfig) func(c *fiber.Ctx) error {
|
2024-02-18 02:12:02 -08:00
|
|
|
return func(c *fiber.Ctx) error {
|
|
|
|
file, err := getFileFromRequest(c)
|
|
|
|
if err != nil {
|
2024-10-12 03:45:47 -04:00
|
|
|
return c.Status(fiber.StatusInternalServerError).SendString(bluemonday.StrictPolicy().Sanitize(err.Error()))
|
2024-02-18 02:12:02 -08:00
|
|
|
}
|
|
|
|
|
2024-03-01 10:19:53 -05:00
|
|
|
fileContents, err := os.ReadFile(filepath.Join(appConfig.UploadDir, file.Filename))
|
2024-02-18 02:12:02 -08:00
|
|
|
if err != nil {
|
2024-10-12 03:45:47 -04:00
|
|
|
return c.Status(fiber.StatusInternalServerError).SendString(bluemonday.StrictPolicy().Sanitize(err.Error()))
|
2024-02-18 02:12:02 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
return c.Send(fileContents)
|
|
|
|
}
|
|
|
|
}
|