2023-06-24 06:18:17 +00:00
|
|
|
package gallery
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2023-07-02 09:15:05 +00:00
|
|
|
"strings"
|
2023-06-24 06:18:17 +00:00
|
|
|
|
|
|
|
"github.com/go-skynet/LocalAI/pkg/utils"
|
|
|
|
"github.com/imdario/mergo"
|
|
|
|
"gopkg.in/yaml.v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Gallery struct {
|
|
|
|
URL string `json:"url" yaml:"url"`
|
|
|
|
Name string `json:"name" yaml:"name"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Installs a model from the gallery (galleryname@modelname)
|
|
|
|
func InstallModelFromGallery(galleries []Gallery, name string, basePath string, req GalleryModel, downloadStatus func(string, string, string, float64)) error {
|
|
|
|
applyModel := func(model *GalleryModel) error {
|
2023-07-31 19:13:16 +00:00
|
|
|
name = strings.ReplaceAll(name, string(os.PathSeparator), "__")
|
|
|
|
|
2023-06-26 10:25:38 +00:00
|
|
|
config, err := GetGalleryConfigFromURL(model.URL)
|
2023-06-24 06:18:17 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-07-20 20:10:12 +00:00
|
|
|
installName := model.Name
|
2023-06-24 06:18:17 +00:00
|
|
|
if req.Name != "" {
|
2023-07-20 20:10:12 +00:00
|
|
|
installName = req.Name
|
2023-06-24 06:18:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
config.Files = append(config.Files, req.AdditionalFiles...)
|
|
|
|
config.Files = append(config.Files, model.AdditionalFiles...)
|
|
|
|
|
|
|
|
// TODO model.Overrides could be merged with user overrides (not defined yet)
|
|
|
|
if err := mergo.Merge(&model.Overrides, req.Overrides, mergo.WithOverride); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-07-20 20:10:12 +00:00
|
|
|
if err := InstallModel(basePath, installName, &config, model.Overrides, downloadStatus); err != nil {
|
2023-06-24 06:18:17 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-07-20 20:10:12 +00:00
|
|
|
models, err := AvailableGalleryModels(galleries, basePath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
model, err := FindGallery(models, name)
|
|
|
|
if err != nil {
|
2023-07-31 19:13:16 +00:00
|
|
|
var err2 error
|
|
|
|
model, err2 = FindGallery(models, strings.ToLower(name))
|
|
|
|
if err2 != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-07-20 20:10:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return applyModel(model)
|
|
|
|
}
|
|
|
|
|
|
|
|
func FindGallery(models []*GalleryModel, name string) (*GalleryModel, error) {
|
|
|
|
// os.PathSeparator is not allowed in model names. Replace them with "__" to avoid conflicts with file paths.
|
|
|
|
name = strings.ReplaceAll(name, string(os.PathSeparator), "__")
|
|
|
|
|
2023-06-24 06:18:17 +00:00
|
|
|
for _, model := range models {
|
|
|
|
if name == fmt.Sprintf("%s@%s", model.Gallery.Name, model.Name) {
|
2023-07-20 20:10:12 +00:00
|
|
|
return model, nil
|
2023-06-24 06:18:17 +00:00
|
|
|
}
|
|
|
|
}
|
2023-07-20 20:10:12 +00:00
|
|
|
return nil, fmt.Errorf("no gallery found with name %q", name)
|
|
|
|
}
|
|
|
|
|
|
|
|
// InstallModelFromGalleryByName loads a model from the gallery by specifying only the name (first match wins)
|
|
|
|
func InstallModelFromGalleryByName(galleries []Gallery, name string, basePath string, req GalleryModel, downloadStatus func(string, string, string, float64)) error {
|
|
|
|
models, err := AvailableGalleryModels(galleries, basePath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
name = strings.ReplaceAll(name, string(os.PathSeparator), "__")
|
|
|
|
var model *GalleryModel
|
|
|
|
for _, m := range models {
|
2023-07-31 19:13:16 +00:00
|
|
|
if name == m.Name || name == strings.ToLower(m.Name) {
|
2023-07-20 20:10:12 +00:00
|
|
|
model = m
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if model == nil {
|
|
|
|
return fmt.Errorf("no model found with name %q", name)
|
|
|
|
}
|
2023-06-24 06:18:17 +00:00
|
|
|
|
2023-07-20 20:10:12 +00:00
|
|
|
return InstallModelFromGallery(galleries, fmt.Sprintf("%s@%s", model.Gallery.Name, model.Name), basePath, req, downloadStatus)
|
2023-06-24 06:18:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// List available models
|
|
|
|
// Models galleries are a list of json files that are hosted on a remote server (for example github).
|
|
|
|
// Each json file contains a list of models that can be downloaded and optionally overrides to define a new model setting.
|
|
|
|
func AvailableGalleryModels(galleries []Gallery, basePath string) ([]*GalleryModel, error) {
|
|
|
|
var models []*GalleryModel
|
|
|
|
|
|
|
|
// Get models from galleries
|
|
|
|
for _, gallery := range galleries {
|
|
|
|
galleryModels, err := getGalleryModels(gallery, basePath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
models = append(models, galleryModels...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return models, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getGalleryModels(gallery Gallery, basePath string) ([]*GalleryModel, error) {
|
|
|
|
var models []*GalleryModel = []*GalleryModel{}
|
|
|
|
|
2023-06-26 10:25:38 +00:00
|
|
|
err := utils.GetURI(gallery.URL, func(url string, d []byte) error {
|
2023-06-24 06:18:17 +00:00
|
|
|
return yaml.Unmarshal(d, &models)
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return models, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add gallery to models
|
|
|
|
for _, model := range models {
|
|
|
|
model.Gallery = gallery
|
|
|
|
// we check if the model was already installed by checking if the config file exists
|
|
|
|
// TODO: (what to do if the model doesn't install a config file?)
|
|
|
|
if _, err := os.Stat(filepath.Join(basePath, fmt.Sprintf("%s.yaml", model.Name))); err == nil {
|
|
|
|
model.Installed = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return models, nil
|
|
|
|
}
|