ollama/server/modelpath.go

147 lines
3 KiB
Go
Raw Normal View History

2023-07-17 22:44:21 -07:00
package server
import (
2023-08-21 21:56:56 -07:00
"errors"
2023-07-17 22:44:21 -07:00
"fmt"
2023-08-21 18:38:31 -07:00
"net/url"
2023-07-17 22:44:21 -07:00
"os"
"path/filepath"
2023-07-20 11:23:43 -07:00
"runtime"
2023-07-17 22:44:21 -07:00
"strings"
)
type ModelPath struct {
ProtocolScheme string
Registry string
Namespace string
Repository string
Tag string
}
const (
DefaultRegistry = "registry.ollama.ai"
DefaultNamespace = "library"
DefaultTag = "latest"
DefaultProtocolScheme = "https"
)
2023-08-21 21:56:56 -07:00
var (
ErrInvalidImageFormat = errors.New("invalid image format")
ErrInvalidProtocol = errors.New("invalid protocol scheme")
ErrInsecureProtocol = errors.New("insecure protocol http")
)
func ParseModelPath(name string) ModelPath {
2023-08-21 21:56:56 -07:00
mp := ModelPath{
ProtocolScheme: DefaultProtocolScheme,
Registry: DefaultRegistry,
Namespace: DefaultNamespace,
Repository: "",
Tag: DefaultTag,
}
2023-07-17 22:44:21 -07:00
2023-08-21 18:38:31 -07:00
before, after, found := strings.Cut(name, "://")
if found {
mp.ProtocolScheme = before
name = after
2023-08-21 21:56:56 -07:00
}
2023-08-30 14:14:12 -04:00
parts := strings.Split(name, string(os.PathSeparator))
switch len(parts) {
2023-07-17 22:44:21 -07:00
case 3:
mp.Registry = parts[0]
mp.Namespace = parts[1]
mp.Repository = parts[2]
2023-07-17 22:44:21 -07:00
case 2:
mp.Namespace = parts[0]
mp.Repository = parts[1]
2023-07-17 22:44:21 -07:00
case 1:
mp.Repository = parts[0]
2023-07-17 22:44:21 -07:00
}
if repo, tag, found := strings.Cut(mp.Repository, ":"); found {
2023-08-21 21:56:56 -07:00
mp.Repository = repo
mp.Tag = tag
2023-07-17 22:44:21 -07:00
}
return mp
2023-07-17 22:44:21 -07:00
}
func (mp ModelPath) GetNamespaceRepository() string {
return fmt.Sprintf("%s/%s", mp.Namespace, mp.Repository)
}
func (mp ModelPath) GetFullTagname() string {
return fmt.Sprintf("%s/%s/%s:%s", mp.Registry, mp.Namespace, mp.Repository, mp.Tag)
}
func (mp ModelPath) GetShortTagname() string {
if mp.Registry == DefaultRegistry {
if mp.Namespace == DefaultNamespace {
return fmt.Sprintf("%s:%s", mp.Repository, mp.Tag)
}
return fmt.Sprintf("%s/%s:%s", mp.Namespace, mp.Repository, mp.Tag)
2023-07-17 22:44:21 -07:00
}
return fmt.Sprintf("%s/%s/%s:%s", mp.Registry, mp.Namespace, mp.Repository, mp.Tag)
2023-07-17 22:44:21 -07:00
}
func (mp ModelPath) GetManifestPath(createDir bool) (string, error) {
home, err := os.UserHomeDir()
if err != nil {
return "", err
}
path := filepath.Join(home, ".ollama", "models", "manifests", mp.Registry, mp.Namespace, mp.Repository, mp.Tag)
if createDir {
if err := os.MkdirAll(filepath.Dir(path), 0o755); err != nil {
return "", err
}
}
return path, nil
}
2023-08-21 18:38:31 -07:00
func (mp ModelPath) BaseURL() *url.URL {
return &url.URL{
Scheme: mp.ProtocolScheme,
Host: mp.Registry,
}
}
2023-07-18 09:09:45 -07:00
func GetManifestPath() (string, error) {
home, err := os.UserHomeDir()
if err != nil {
return "", err
}
2023-09-05 17:10:40 -07:00
path := filepath.Join(home, ".ollama", "models", "manifests")
2023-09-06 14:30:08 -07:00
if err := os.MkdirAll(path, 0o755); err != nil {
2023-09-05 17:10:40 -07:00
return "", err
}
return path, nil
2023-07-18 09:09:45 -07:00
}
2023-07-17 22:44:21 -07:00
func GetBlobsPath(digest string) (string, error) {
home, err := os.UserHomeDir()
if err != nil {
return "", err
}
2023-07-20 11:23:43 -07:00
if runtime.GOOS == "windows" {
digest = strings.ReplaceAll(digest, ":", "-")
}
2023-07-18 11:24:19 -07:00
path := filepath.Join(home, ".ollama", "models", "blobs", digest)
dirPath := filepath.Dir(path)
if digest == "" {
dirPath = path
}
if err := os.MkdirAll(dirPath, 0o755); err != nil {
2023-07-17 22:44:21 -07:00
return "", err
}
2023-07-18 11:24:19 -07:00
return path, nil
2023-07-17 22:44:21 -07:00
}