package model import ( "cmp" "errors" "hash/maphash" "io" "log/slog" "slices" "strings" "sync" "github.com/ollama/ollama/types/structs" ) // Errors var ( // ErrInvalidName, ErrIncompleteName, and ErrInvalidDigest are not // used by this package, but are exported so that other packages can // use them, instead of defining their own errors for them. ErrInvalidName = errors.New("invalid model name") ErrIncompleteName = errors.New("incomplete model name") ErrInvalidDigest = errors.New("invalid digest") ) // Defaults const ( // DefaultMask is the default mask used by [Name.DisplayShortest]. DefaultMask = "registry.ollama.ai/library/_:latest" // DefaultFill is the default fill used by [ParseName]. DefaultFill = "registry.ollama.ai/library/_:latest" ) const MaxNamePartLen = 128 type PartKind int // Levels of concreteness const ( // Each value aligns with its index in the Name.parts array. PartHost PartKind = iota PartNamespace PartModel PartTag PartBuild PartDigest // Invalid is a special part that is used to indicate that a part is // invalid. It is not a valid part of a Name. // // It should be kept as the last part in the list. PartInvalid ) var kindNames = map[PartKind]string{ PartHost: "Host", PartNamespace: "Namespace", PartModel: "Name", PartTag: "Tag", PartBuild: "Build", PartDigest: "Digest", PartInvalid: "Invalid", } func (k PartKind) String() string { return cmp.Or(kindNames[k], "Unknown") } // Name is an opaque reference to a model. It holds the parts of a model // with the case preserved, but is not directly comparable with other Names // since model names can be represented with different casing depending on // the use case. For instance, "Mistral" and "mistral" are the same model // but each version may have come from different sources (e.g. copied from a // Web page, or from a file path). // // Valid Names can ONLY be constructed by calling [ParseName]. // // A Name is valid if and only if is have a valid Model part. The other parts // are optional. // // A Name is considered "complete" if it has all parts present. To check if a // Name is complete, use [Name.IsComplete]. // // To compare two names in a case-insensitive manner, use [Name.EqualFold]. // // The parts of a Name are: // // - Host: the domain of the model (optional) // - Namespace: the namespace of the model (optional) // - Model: the name of the model (required) // - Tag: the tag of the model (optional) // - Build: the build of the model; usually the quantization or "file type" (optional) // // The parts can be obtained in their original form by calling [Name.Parts]. // // To check if a Name has at minimum a valid model part, use [Name.IsValid]. // // To make a Name by filling in missing parts from another Name, use [Fill]. type Name struct { _ structs.Incomparable parts [6]string // host, namespace, model, tag, build, digest // TODO(bmizerany): track offsets and hold s (raw string) here? We // could pack the offsets all into a single uint64 since the first // parts take less bits since their max offset is less than the max // offset of the next part. This would save a ton of bytes per Name // and mean zero allocations for String. } // ParseNameFill parses s into a Name, and returns the result of filling it with // defaults. The input string must be a valid string // representation of a model name in the form: // // [host/][namespace/][:tag][+build][@-] // // The name part is required, all others are optional. If a part is missing, // it is left empty in the returned Name. If a part is invalid, the zero Ref // value is returned. // // The build part is normalized to uppercase. // // Examples of valid paths: // // "example.com/library/mistral:7b+x" // "example.com/eva/mistral:7b+Q4_0" // "mistral:7b+x" // "example.com/mike/mistral:latest+Q4_0" // "example.com/bruce/mistral:latest" // "example.com/pdevine/thisisfine:7b+Q4_0@sha256-1234567890abcdef" // // Examples of invalid paths: // // "example.com/mistral:7b+" // "example.com/mistral:7b+Q4_0+" // "x/y/z/z:8n+I" // "" // // It returns the zero value if any part is invalid. // // As a rule of thumb, an valid name is one that can be round-tripped with // the [Name.String] method. That means ("x+") is invalid because // [Name.String] will not print a "+" if the build is empty. // // For more about filling in missing parts, see [Fill]. func ParseNameFill(s, defaults string) Name { var r Name parts(s)(func(kind PartKind, part string) bool { if kind == PartInvalid { r = Name{} return false } if kind == PartDigest && !ParseDigest(part).IsValid() { r = Name{} return false } r.parts[kind] = part return true }) if r.IsValid() || r.IsResolved() { if defaults == "" { return r } return Fill(r, ParseNameFill(defaults, "")) } return Name{} } // ParseName is equal to ParseNameFill(s, DefaultFill). func ParseName(s string) Name { return ParseNameFill(s, DefaultFill) } func MustParseNameFill(s, defaults string) Name { r := ParseNameFill(s, "") if !r.IsValid() { panic("model.MustParseName: invalid name: " + s) } return r } // Fill fills in the missing parts of dst with the parts of src. // // The returned Name will only be valid if dst is valid. func Fill(dst, src Name) Name { var r Name for i := range r.parts { r.parts[i] = cmp.Or(dst.parts[i], src.parts[i]) } return r } // WithBuild returns a copy of r with the build set to the given string. func (r Name) WithBuild(build string) Name { r.parts[PartBuild] = build return r } func (r Name) WithDigest(digest Digest) Name { r.parts[PartDigest] = digest.String() return r } var mapHashSeed = maphash.MakeSeed() // MapHash returns a case insensitive hash for use in maps and equality // checks. For a convenient way to compare names, use [Name.EqualFold]. // //nolint:errcheck func (r Name) MapHash() uint64 { // correctly hash the parts with case insensitive comparison var h maphash.Hash h.SetSeed(mapHashSeed) for _, part := range r.Parts() { // downcase the part for hashing for i := range part { c := part[i] if c >= 'A' && c <= 'Z' { c = c - 'A' + 'a' } h.WriteByte(c) } } return h.Sum64() } func (r Name) slice(from, to PartKind) Name { var v Name copy(v.parts[from:to+1], r.parts[from:to+1]) return v } // DisplayShortest returns the shortest possible display string in form: // // [host/][/][:] // // The host is omitted if it is the mask host is the same as r. // The namespace is omitted if the host and the namespace are the same as r. // The tag is omitted if it is the mask tag is the same as r. func (r Name) DisplayShortest(mask string) string { mask = cmp.Or(mask, DefaultMask) d := ParseName(mask) if !d.IsValid() { panic("mask is an invalid Name") } equalSlice := func(form, to PartKind) bool { return r.slice(form, to).EqualFold(d.slice(form, to)) } if equalSlice(PartHost, PartNamespace) { r.parts[PartNamespace] = "" } if equalSlice(PartHost, PartHost) { r.parts[PartHost] = "" } if equalSlice(PartTag, PartTag) { r.parts[PartTag] = "" } return r.slice(PartHost, PartTag).String() } // DisplayLong returns the fullest possible display string in form: // // /: // // If any part is missing, it is omitted from the display string. func (r Name) DisplayLong() string { return r.slice(PartNamespace, PartTag).String() } var seps = [...]string{ PartHost: "/", PartNamespace: "/", PartModel: ":", PartTag: "+", PartBuild: "@", PartDigest: "", } // WriteTo implements io.WriterTo. It writes the fullest possible display // string in form: // // //:+@- // // Missing parts and their separators are not written. // // The full digest is always prefixed with "@". That is if [Name.IsValid] // reports false and [Name.IsResolved] reports true, then the string is // returned as "@-". func (r Name) writeTo(w io.StringWriter) error { var partsWritten int for i := range r.parts { if r.parts[i] == "" { continue } if partsWritten > 0 || i == int(PartDigest) { if _, err := w.WriteString(seps[i-1]); err != nil { return err } } if _, err := w.WriteString(r.parts[i]); err != nil { return err } partsWritten++ } return nil } var builderPool = sync.Pool{ New: func() interface{} { return &strings.Builder{} }, } // String returns the fullest possible display string in form: // // //:+ // // If any part is missing, it is omitted from the display string. // // For the fullest possible display string without the build, use // [Name.DisplayFullest]. func (r Name) String() string { b := builderPool.Get().(*strings.Builder) defer builderPool.Put(b) b.Reset() b.Grow(50) // arbitrarily long enough for most names _ = r.writeTo(b) return b.String() } // GoString implements fmt.GoStringer. It returns a string suitable for // debugging and logging. It is similar to [Name.String] but it always // returns a string that includes all parts of the Name, with missing parts // replaced with a ("?"). func (r Name) GoString() string { for i := range r.parts { r.parts[i] = cmp.Or(r.parts[i], "?") } return r.String() } // LogValue implements slog.Valuer. func (r Name) LogValue() slog.Value { return slog.StringValue(r.GoString()) } // IsComplete reports whether the Name is fully qualified. That is it has a // domain, namespace, name, tag, and build. func (r Name) IsComplete() bool { return !slices.Contains(r.parts[:PartDigest], "") } // IsCompleteNoBuild is like [Name.IsComplete] but it does not require the // build part to be present. func (r Name) IsCompleteNoBuild() bool { return !slices.Contains(r.parts[:PartBuild], "") } // IsResolved reports true if the Name has a valid digest. // // It is possible to have a valid Name, or a complete Name that is not // resolved. func (r Name) IsResolved() bool { return r.Digest().IsValid() } // Digest returns the digest part of the Name, if any. // // If Digest returns a non-empty string, then [Name.IsResolved] will return // true, and digest is considered valid. func (r Name) Digest() Digest { // This was already validated by ParseName, so we can just return it. return Digest{r.parts[PartDigest]} } // EqualFold reports whether r and o are equivalent model names, ignoring // case. func (r Name) EqualFold(o Name) bool { return r.CompareFold(o) == 0 } // CompareFold performs a case-insensitive cmp.Compare on r and o. // // This can be used with [slices.SortFunc]. // // For simple equality checks, use [Name.EqualFold]. func (r Name) CompareFold(o Name) int { return slices.CompareFunc(r.parts[:], o.parts[:], compareFold) } func compareFold(a, b string) int { return slices.CompareFunc([]rune(a), []rune(b), func(a, b rune) int { return cmp.Compare(downcase(a), downcase(b)) }) } func downcase(r rune) rune { if r >= 'A' && r <= 'Z' { return r - 'A' + 'a' } return r } // TODO(bmizerany): driver.Value? (MarshalText etc should be enough) // Parts returns the parts of the Name in order of concreteness. // // The length of the returned slice is always 5. func (r Name) Parts() []string { return slices.Clone(r.parts[:]) } // iter_Seq2 is a iter.Seq2 defined here to avoid the current build // restrictions in the go1.22 iter package requiring the // goexperiment.rangefunc tag to be set via the GOEXPERIMENT=rangefunc flag, // which we are not yet ready to support. // // Once we are ready to support rangefunc, this can be removed and replaced // with the iter.Seq2 type. type iter_Seq2[A, B any] func(func(A, B) bool) // Parts returns a sequence of the parts of a Name string from most specific // to least specific. // // It normalizes the input string by removing "http://" and "https://" only. // No other normalizations are performed. func parts(s string) iter_Seq2[PartKind, string] { return func(yield func(PartKind, string) bool) { //nolint:gosimple if strings.HasPrefix(s, "http://") { s = s[len("http://"):] } //nolint:gosimple if strings.HasPrefix(s, "https://") { s = s[len("https://"):] } if len(s) > MaxNamePartLen || len(s) == 0 { return } yieldValid := func(kind PartKind, part string) bool { if !isValidPart(kind, part) { yield(PartInvalid, "") return false } return yield(kind, part) } numConsecutiveDots := 0 partLen := 0 state, j := PartDigest, len(s) for i := len(s) - 1; i >= 0; i-- { if partLen++; partLen > MaxNamePartLen { // catch a part that is too long early, so // we don't keep spinning on it, waiting for // an isInValidPart check which would scan // over it again. yield(PartInvalid, "") return } switch s[i] { case '@': switch state { case PartDigest: if !yieldValid(PartDigest, s[i+1:j]) { return } if i == 0 { // This is the form // "@" which is valid. // // We're done. return } state, j, partLen = PartBuild, i, 0 default: yield(PartInvalid, "") return } case '+': switch state { case PartBuild, PartDigest: if !yieldValid(PartBuild, s[i+1:j]) { return } state, j, partLen = PartTag, i, 0 default: yield(PartInvalid, "") return } case ':': switch state { case PartTag, PartBuild, PartDigest: if !yieldValid(PartTag, s[i+1:j]) { return } state, j, partLen = PartModel, i, 0 default: yield(PartInvalid, "") return } case '/': switch state { case PartModel, PartTag, PartBuild, PartDigest: if !yieldValid(PartModel, s[i+1:j]) { return } state, j = PartNamespace, i case PartNamespace: if !yieldValid(PartNamespace, s[i+1:j]) { return } state, j, partLen = PartHost, i, 0 default: yield(PartInvalid, "") return } default: if s[i] == '.' { if numConsecutiveDots++; numConsecutiveDots > 1 { yield(PartInvalid, "") return } } else { numConsecutiveDots = 0 } if !isValidByteFor(state, s[i]) { yield(PartInvalid, "") return } } } if state <= PartNamespace { yieldValid(state, s[:j]) } else { yieldValid(PartModel, s[:j]) } } } func (r Name) IsZero() bool { return r.parts == [6]string{} } // IsValid reports if a model has at minimum a valid model part. func (r Name) IsValid() bool { // Parts ensures we only have valid parts, so no need to validate // them here, only check if we have a name or not. return r.parts[PartModel] != "" } // isValidPart reports if s contains all valid characters for the given // part kind. func isValidPart(kind PartKind, s string) bool { if s == "" { return false } for _, c := range []byte(s) { if !isValidByteFor(kind, c) { return false } } return true } func isValidByteFor(kind PartKind, c byte) bool { if kind == PartNamespace && c == '.' { return false } if c == '.' || c == '-' { return true } if c >= 'a' && c <= 'z' || c >= 'A' && c <= 'Z' || c >= '0' && c <= '9' || c == '_' { return true } return false }