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" )
)
2023-08-22 09:39:42 -07:00
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 ) )
2023-08-22 09:39:42 -07:00
switch len ( parts ) {
2023-07-17 22:44:21 -07:00
case 3 :
2023-08-22 09:39:42 -07:00
mp . Registry = parts [ 0 ]
mp . Namespace = parts [ 1 ]
mp . Repository = parts [ 2 ]
2023-07-17 22:44:21 -07:00
case 2 :
2023-08-22 09:39:42 -07:00
mp . Namespace = parts [ 0 ]
mp . Repository = parts [ 1 ]
2023-07-17 22:44:21 -07:00
case 1 :
2023-08-22 09:39:42 -07:00
mp . Repository = parts [ 0 ]
2023-07-17 22:44:21 -07:00
}
2023-08-22 09:39:42 -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
}
2023-08-22 09:39:42 -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 {
2023-07-21 15:42:19 -07:00
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
}
2023-07-21 15:42:19 -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
}
2023-10-27 10:19:59 -04:00
// modelsDir returns the value of the OLLAMA_MODELS environment variable or the user's home directory if OLLAMA_MODELS is not set.
// The models directory is where Ollama stores its model files and manifests.
func modelsDir ( ) ( string , error ) {
if models , exists := os . LookupEnv ( "OLLAMA_MODELS" ) ; exists {
return models , nil
}
2023-07-17 22:44:21 -07:00
home , err := os . UserHomeDir ( )
if err != nil {
return "" , err
}
2023-10-27 10:19:59 -04:00
return filepath . Join ( home , ".ollama" , "models" ) , nil
}
2023-07-17 22:44:21 -07:00
2023-10-27 10:19:59 -04:00
// GetManifestPath returns the path to the manifest file for the given model path, it is up to the caller to create the directory if it does not exist.
func ( mp ModelPath ) GetManifestPath ( ) ( string , error ) {
dir , err := modelsDir ( )
if err != nil {
return "" , err
2023-07-17 22:44:21 -07:00
}
2023-10-27 10:19:59 -04:00
return filepath . Join ( dir , "manifests" , mp . Registry , mp . Namespace , mp . Repository , mp . Tag ) , nil
2023-07-17 22:44:21 -07:00
}
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 ) {
2023-10-27 10:19:59 -04:00
dir , err := modelsDir ( )
2023-07-18 09:09:45 -07:00
if err != nil {
return "" , err
}
2023-10-27 10:19:59 -04:00
path := filepath . Join ( dir , "manifests" )
2023-09-06 14:30:08 -07:00
if err := os . MkdirAll ( path , 0 o755 ) ; 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 ) {
2023-10-27 10:19:59 -04:00
dir , err := modelsDir ( )
2023-07-17 22:44:21 -07:00
if err != nil {
return "" , err
}
2023-07-20 11:23:43 -07:00
if runtime . GOOS == "windows" {
digest = strings . ReplaceAll ( digest , ":" , "-" )
}
2023-10-27 10:19:59 -04:00
path := filepath . Join ( dir , "blobs" , digest )
2023-09-11 14:54:52 -07:00
dirPath := filepath . Dir ( path )
if digest == "" {
dirPath = path
}
if err := os . MkdirAll ( dirPath , 0 o755 ) ; 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
}