2023-07-16 17:02:22 -07:00
package server
import (
"bytes"
2023-07-25 17:08:51 -04:00
"context"
2023-07-16 17:02:22 -07:00
"crypto/sha256"
2023-08-28 20:50:24 -07:00
"encoding/hex"
2023-07-16 17:02:22 -07:00
"encoding/json"
"errors"
"fmt"
"io"
"log"
2024-01-18 10:52:01 -08:00
"log/slog"
2023-07-16 17:02:22 -07:00
"net/http"
2023-08-21 18:38:31 -07:00
"net/url"
2023-07-16 17:02:22 -07:00
"os"
"path/filepath"
2023-08-21 18:24:42 -07:00
"runtime"
2024-02-14 11:29:49 -08:00
"strconv"
2023-07-16 17:02:22 -07:00
"strings"
2023-09-01 17:12:20 -05:00
"text/template"
2023-07-16 17:02:22 -07:00
2023-11-29 11:11:42 -08:00
"golang.org/x/exp/slices"
2023-07-16 17:02:22 -07:00
"github.com/jmorganca/ollama/api"
2023-07-21 13:33:56 -07:00
"github.com/jmorganca/ollama/llm"
2023-07-16 17:02:22 -07:00
"github.com/jmorganca/ollama/parser"
2024-02-14 11:29:49 -08:00
"github.com/jmorganca/ollama/version"
2023-07-16 17:02:22 -07:00
)
2024-02-14 11:29:49 -08:00
type registryOptions struct {
Insecure bool
Username string
Password string
Token string
}
2023-07-16 17:02:22 -07:00
type Model struct {
2023-11-30 10:30:23 -08:00
Name string ` json:"name" `
2023-12-01 11:37:17 -08:00
Config ConfigV2
2023-11-30 10:30:23 -08:00
ShortName string
ModelPath string
2024-01-25 12:12:36 -08:00
ParentModel string
2023-11-30 10:30:23 -08:00
AdapterPaths [ ] string
ProjectorPaths [ ] string
Template string
System string
License [ ] string
Digest string
2023-12-11 13:56:22 -08:00
Size int64
2023-11-30 10:30:23 -08:00
Options map [ string ] interface { }
2024-01-25 12:12:36 -08:00
Messages [ ] Message
}
type Message struct {
Role string ` json:"role" `
Content string ` json:"content" `
2023-07-16 17:02:22 -07:00
}
type ManifestV2 struct {
SchemaVersion int ` json:"schemaVersion" `
MediaType string ` json:"mediaType" `
2023-11-22 13:28:49 -08:00
Config * Layer ` json:"config" `
2023-07-16 17:02:22 -07:00
Layers [ ] * Layer ` json:"layers" `
}
type ConfigV2 struct {
2023-12-09 02:05:43 -08:00
ModelFormat string ` json:"model_format" `
ModelFamily string ` json:"model_family" `
ModelFamilies [ ] string ` json:"model_families" `
ModelType string ` json:"model_type" `
FileType string ` json:"file_type" `
2023-07-21 13:33:56 -07:00
// required by spec
2023-07-16 17:02:22 -07:00
Architecture string ` json:"architecture" `
OS string ` json:"os" `
2023-12-01 11:37:17 -08:00
RootFS RootFS ` json:"rootfs" `
2023-07-16 17:02:22 -07:00
}
2023-11-29 11:11:42 -08:00
func ( c * ConfigV2 ) SetModelFormat ( format string ) {
if c . ModelFormat == "" {
c . ModelFormat = format
}
}
func ( c * ConfigV2 ) SetModelFamily ( families ... string ) {
for _ , family := range families {
if c . ModelFamily == "" {
c . ModelFamily = family
}
if ! slices . Contains ( c . ModelFamilies , family ) {
c . ModelFamilies = append ( c . ModelFamilies , family )
}
}
}
func ( c * ConfigV2 ) SetModelType ( modelType string ) {
if c . ModelType == "" {
c . ModelType = modelType
}
}
func ( c * ConfigV2 ) SetFileType ( fileType string ) {
if c . FileType == "" {
c . FileType = fileType
}
}
2023-07-16 17:02:22 -07:00
type RootFS struct {
Type string ` json:"type" `
DiffIDs [ ] string ` json:"diff_ids" `
}
2023-09-28 10:00:34 -07:00
func ( m * ManifestV2 ) GetTotalSize ( ) ( total int64 ) {
2023-07-18 09:09:45 -07:00
for _ , layer := range m . Layers {
total += layer . Size
}
2023-09-28 10:00:34 -07:00
2023-07-18 09:09:45 -07:00
total += m . Config . Size
return total
}
2023-08-28 20:50:24 -07:00
func GetManifest ( mp ModelPath ) ( * ManifestV2 , string , error ) {
2023-10-27 10:19:59 -04:00
fp , err := mp . GetManifestPath ( )
2023-07-17 11:03:55 -07:00
if err != nil {
2023-08-28 20:50:24 -07:00
return nil , "" , err
2023-07-17 11:03:55 -07:00
}
2023-07-17 14:21:27 -07:00
2023-07-21 23:02:12 -07:00
if _ , err = os . Stat ( fp ) ; err != nil {
2023-08-28 20:50:24 -07:00
return nil , "" , err
2023-07-16 17:02:22 -07:00
}
var manifest * ManifestV2
2023-07-17 14:21:27 -07:00
bts , err := os . ReadFile ( fp )
2023-07-16 17:02:22 -07:00
if err != nil {
2023-08-28 20:50:24 -07:00
return nil , "" , fmt . Errorf ( "couldn't open file '%s'" , fp )
2023-07-16 17:02:22 -07:00
}
2023-08-28 20:50:24 -07:00
shaSum := sha256 . Sum256 ( bts )
shaStr := hex . EncodeToString ( shaSum [ : ] )
2023-07-17 14:21:27 -07:00
if err := json . Unmarshal ( bts , & manifest ) ; err != nil {
2023-08-28 20:50:24 -07:00
return nil , "" , err
2023-07-16 17:02:22 -07:00
}
2023-08-28 20:50:24 -07:00
return manifest , shaStr , nil
2023-07-16 17:02:22 -07:00
}
func GetModel ( name string ) ( * Model , error ) {
2023-08-22 09:39:42 -07:00
mp := ParseModelPath ( name )
2023-08-28 20:50:24 -07:00
manifest , digest , err := GetManifest ( mp )
2023-07-16 17:02:22 -07:00
if err != nil {
return nil , err
}
model := & Model {
2023-10-19 10:39:58 -04:00
Name : mp . GetFullTagname ( ) ,
ShortName : mp . GetShortTagname ( ) ,
Digest : digest ,
Template : "{{ .Prompt }}" ,
License : [ ] string { } ,
2023-12-11 13:56:22 -08:00
Size : manifest . GetTotalSize ( ) ,
2023-07-16 17:02:22 -07:00
}
2023-12-01 11:37:17 -08:00
filename , err := GetBlobsPath ( manifest . Config . Digest )
if err != nil {
return nil , err
}
configFile , err := os . Open ( filename )
if err != nil {
return nil , err
}
defer configFile . Close ( )
if err := json . NewDecoder ( configFile ) . Decode ( & model . Config ) ; err != nil {
return nil , err
}
2023-07-16 17:02:22 -07:00
for _ , layer := range manifest . Layers {
2023-07-17 22:44:21 -07:00
filename , err := GetBlobsPath ( layer . Digest )
2023-07-17 11:03:55 -07:00
if err != nil {
return nil , err
}
2023-07-16 17:02:22 -07:00
switch layer . MediaType {
case "application/vnd.ollama.image.model" :
model . ModelPath = filename
2024-01-25 12:12:36 -08:00
model . ParentModel = layer . From
2023-08-04 18:56:40 -04:00
case "application/vnd.ollama.image.embed" :
2023-10-16 11:07:37 -04:00
// Deprecated in versions > 0.1.2
// TODO: remove this warning in a future version
2024-01-18 10:52:01 -08:00
slog . Info ( "WARNING: model contains embeddings, but embeddings in modelfiles have been deprecated and will be ignored." )
2023-08-03 17:16:05 -07:00
case "application/vnd.ollama.image.adapter" :
model . AdapterPaths = append ( model . AdapterPaths , filename )
2023-11-30 10:30:23 -08:00
case "application/vnd.ollama.image.projector" :
model . ProjectorPaths = append ( model . ProjectorPaths , filename )
2023-07-17 14:21:27 -07:00
case "application/vnd.ollama.image.template" :
bts , err := os . ReadFile ( filename )
if err != nil {
return nil , err
}
model . Template = string ( bts )
case "application/vnd.ollama.image.system" :
bts , err := os . ReadFile ( filename )
2023-07-16 17:02:22 -07:00
if err != nil {
return nil , err
}
2023-07-17 14:21:27 -07:00
model . System = string ( bts )
2023-07-19 19:43:00 -07:00
case "application/vnd.ollama.image.prompt" :
bts , err := os . ReadFile ( filename )
if err != nil {
return nil , err
}
model . Template = string ( bts )
2023-07-16 17:02:22 -07:00
case "application/vnd.ollama.image.params" :
2023-07-17 12:08:10 -07:00
params , err := os . Open ( filename )
if err != nil {
return nil , err
}
defer params . Close ( )
2023-07-16 17:02:22 -07:00
2023-07-31 15:07:04 -04:00
// parse model options parameters into a map so that we can see which fields have been specified explicitly
2023-08-01 13:36:31 -04:00
if err = json . NewDecoder ( params ) . Decode ( & model . Options ) ; err != nil {
2023-07-31 15:07:04 -04:00
return nil , err
}
2024-01-25 12:12:36 -08:00
case "application/vnd.ollama.image.messages" :
msgs , err := os . Open ( filename )
if err != nil {
return nil , err
}
defer msgs . Close ( )
if err = json . NewDecoder ( msgs ) . Decode ( & model . Messages ) ; err != nil {
return nil , err
}
2023-09-06 11:04:17 -07:00
case "application/vnd.ollama.image.license" :
bts , err := os . ReadFile ( filename )
if err != nil {
return nil , err
}
model . License = append ( model . License , string ( bts ) )
2023-07-16 17:02:22 -07:00
}
}
return model , nil
}
2023-11-21 15:43:17 -05:00
func realpath ( mfDir , from string ) string {
abspath , err := filepath . Abs ( from )
2023-11-14 12:30:34 -08:00
if err != nil {
2023-11-21 15:43:17 -05:00
return from
2023-09-11 11:46:35 -07:00
}
2023-11-14 12:30:34 -08:00
home , err := os . UserHomeDir ( )
2023-07-19 21:55:15 -07:00
if err != nil {
2023-11-14 12:30:34 -08:00
return abspath
2023-07-16 17:02:22 -07:00
}
2023-11-21 15:43:17 -05:00
if from == "~" {
2023-11-14 12:30:34 -08:00
return home
2023-11-21 15:43:17 -05:00
} else if strings . HasPrefix ( from , "~/" ) {
return filepath . Join ( home , from [ 2 : ] )
}
if _ , err := os . Stat ( filepath . Join ( mfDir , from ) ) ; err == nil {
// this is a file relative to the Modelfile
return filepath . Join ( mfDir , from )
2023-07-16 17:02:22 -07:00
}
2023-11-14 12:30:34 -08:00
return abspath
}
2023-11-21 15:43:17 -05:00
func CreateModel ( ctx context . Context , name , modelFileDir string , commands [ ] parser . Command , fn func ( resp api . ProgressResponse ) ) error {
2024-01-19 14:58:36 -08:00
deleteMap := make ( map [ string ] struct { } )
if manifest , _ , err := GetManifest ( ParseModelPath ( name ) ) ; err == nil {
for _ , layer := range append ( manifest . Layers , manifest . Config ) {
deleteMap [ layer . Digest ] = struct { } { }
}
}
2023-07-21 13:33:56 -07:00
config := ConfigV2 {
OS : "linux" ,
2023-11-14 12:30:34 -08:00
Architecture : "amd64" ,
2023-11-22 13:28:49 -08:00
RootFS : RootFS {
Type : "layers" ,
} ,
2023-07-21 13:33:56 -07:00
}
2023-11-22 13:28:49 -08:00
var layers Layers
2024-01-25 12:12:36 -08:00
messages := [ ] string { }
2023-11-14 12:30:34 -08:00
2023-07-28 11:29:00 -04:00
params := make ( map [ string ] [ ] string )
2023-11-14 12:30:34 -08:00
fromParams := make ( map [ string ] any )
2023-07-16 17:02:22 -07:00
for _ , c := range commands {
2023-11-14 12:30:34 -08:00
mediatype := fmt . Sprintf ( "application/vnd.ollama.image.%s" , c . Name )
2023-07-16 17:02:22 -07:00
switch c . Name {
case "model" :
2023-11-15 10:59:38 -08:00
if strings . HasPrefix ( c . Args , "@" ) {
blobPath , err := GetBlobsPath ( strings . TrimPrefix ( c . Args , "@" ) )
if err != nil {
return err
}
c . Args = blobPath
}
2023-11-21 15:43:17 -05:00
bin , err := os . Open ( realpath ( modelFileDir , c . Args ) )
2023-07-16 17:02:22 -07:00
if err != nil {
2023-11-14 12:30:34 -08:00
// not a file on disk so must be a model reference
modelpath := ParseModelPath ( c . Args )
manifest , _ , err := GetManifest ( modelpath )
switch {
case errors . Is ( err , os . ErrNotExist ) :
fn ( api . ProgressResponse { Status : "pulling model" } )
2024-02-14 11:29:49 -08:00
if err := PullModel ( ctx , c . Args , & registryOptions { } , fn ) ; err != nil {
2023-07-25 14:25:13 -04:00
return err
}
2023-11-14 12:30:34 -08:00
manifest , _ , err = GetManifest ( modelpath )
2023-07-21 13:33:56 -07:00
if err != nil {
return err
}
2023-11-14 12:30:34 -08:00
case err != nil :
return err
2023-07-16 17:02:22 -07:00
}
2023-07-21 13:33:56 -07:00
2023-10-06 16:05:32 -04:00
fn ( api . ProgressResponse { Status : "reading model metadata" } )
2023-11-14 12:30:34 -08:00
fromConfigPath , err := GetBlobsPath ( manifest . Config . Digest )
2023-08-17 21:52:11 -07:00
if err != nil {
return err
}
2023-11-14 12:30:34 -08:00
fromConfigFile , err := os . Open ( fromConfigPath )
2023-08-17 21:52:11 -07:00
if err != nil {
return err
}
2023-11-14 12:30:34 -08:00
defer fromConfigFile . Close ( )
2023-08-17 21:52:11 -07:00
2023-11-14 12:30:34 -08:00
var fromConfig ConfigV2
if err := json . NewDecoder ( fromConfigFile ) . Decode ( & fromConfig ) ; err != nil {
2023-08-17 21:52:11 -07:00
return err
}
2023-11-24 13:58:09 -05:00
// if the model is still not in gguf format, error out
if fromConfig . ModelFormat != "gguf" {
return fmt . Errorf ( "%s is not in gguf format, this base model is not compatible with this version of ollama" , c . Args )
}
2023-11-29 11:11:42 -08:00
config . SetModelFormat ( fromConfig . ModelFormat )
config . SetModelFamily ( append ( fromConfig . ModelFamilies , fromConfig . ModelFamily ) ... )
config . SetModelType ( fromConfig . ModelType )
config . SetFileType ( fromConfig . FileType )
2023-08-17 21:52:11 -07:00
2023-11-14 12:30:34 -08:00
for _ , layer := range manifest . Layers {
deleteMap [ layer . Digest ] = struct { } { }
if layer . MediaType == "application/vnd.ollama.image.params" {
fromParamsPath , err := GetBlobsPath ( layer . Digest )
2023-09-05 11:05:03 -07:00
if err != nil {
return err
}
2023-11-14 12:30:34 -08:00
fromParamsFile , err := os . Open ( fromParamsPath )
2023-09-05 11:05:03 -07:00
if err != nil {
return err
}
2023-11-14 12:30:34 -08:00
defer fromParamsFile . Close ( )
2023-09-05 11:05:03 -07:00
2023-11-14 12:30:34 -08:00
if err := json . NewDecoder ( fromParamsFile ) . Decode ( & fromParams ) ; err != nil {
2023-09-05 11:05:03 -07:00
return err
}
}
2023-11-22 13:28:49 -08:00
layer , err := NewLayerFromLayer ( layer . Digest , layer . MediaType , modelpath . GetShortTagname ( ) )
2023-07-16 17:02:22 -07:00
if err != nil {
return err
}
2023-11-14 12:30:34 -08:00
2023-11-22 13:28:49 -08:00
layers . Add ( layer )
2023-07-16 17:02:22 -07:00
}
2023-11-14 12:30:34 -08:00
deleteMap [ manifest . Config . Digest ] = struct { } { }
continue
2023-07-16 17:02:22 -07:00
}
2023-11-14 12:30:34 -08:00
defer bin . Close ( )
2023-08-03 17:16:05 -07:00
2023-11-24 11:57:20 -08:00
var offset int64
2023-11-24 13:58:09 -05:00
CREATE :
2023-11-24 11:57:20 -08:00
for {
fn ( api . ProgressResponse { Status : "creating model layer" } )
2023-08-03 17:16:05 -07:00
2023-11-24 11:57:20 -08:00
bin . Seek ( offset , io . SeekStart )
ggml , err := llm . DecodeGGML ( bin )
2023-11-24 13:58:09 -05:00
if err != nil {
switch {
case errors . Is ( err , io . EOF ) :
break CREATE
case errors . Is ( err , llm . ErrUnsupportedFormat ) :
return fmt . Errorf ( "model binary specified in FROM field is not a valid gguf format model, %w" , err )
default :
return err
}
2023-11-24 11:57:20 -08:00
}
2023-11-14 12:30:34 -08:00
2023-11-29 11:11:42 -08:00
config . SetModelFormat ( ggml . Name ( ) )
config . SetModelFamily ( ggml . ModelFamily ( ) )
config . SetModelType ( ggml . ModelType ( ) )
config . SetFileType ( ggml . FileType ( ) )
2023-08-03 17:16:05 -07:00
2023-11-24 11:57:20 -08:00
mediatype := mediatype
if ggml . ModelFamily ( ) == "clip" {
mediatype = "application/vnd.ollama.image.projector"
}
2023-08-03 17:16:05 -07:00
2023-11-24 11:57:20 -08:00
sr := io . NewSectionReader ( bin , offset , ggml . Size )
layer , err := NewLayer ( sr , mediatype )
if err != nil {
return err
}
layers . Add ( layer )
offset += ggml . Size
}
2023-11-14 12:30:34 -08:00
case "adapter" :
2023-12-01 13:50:55 -05:00
if strings . HasPrefix ( c . Args , "@" ) {
blobPath , err := GetBlobsPath ( strings . TrimPrefix ( c . Args , "@" ) )
if err != nil {
return err
}
c . Args = blobPath
}
2023-12-05 14:57:33 -05:00
2023-11-14 12:30:34 -08:00
fn ( api . ProgressResponse { Status : "creating adapter layer" } )
2023-11-21 15:43:17 -05:00
bin , err := os . Open ( realpath ( modelFileDir , c . Args ) )
2023-08-03 17:16:05 -07:00
if err != nil {
2023-11-14 12:30:34 -08:00
return err
2023-08-03 17:16:05 -07:00
}
2023-11-14 12:30:34 -08:00
defer bin . Close ( )
2023-08-03 17:16:05 -07:00
2023-11-22 13:28:49 -08:00
layer , err := NewLayer ( bin , mediatype )
2023-08-03 17:16:05 -07:00
if err != nil {
2023-11-14 12:30:34 -08:00
return err
2023-08-03 17:16:05 -07:00
}
2023-08-08 16:56:48 -04:00
2023-11-22 13:28:49 -08:00
layers . Add ( layer )
2023-11-14 12:30:34 -08:00
case "license" :
fn ( api . ProgressResponse { Status : "creating license layer" } )
2023-11-22 13:28:49 -08:00
bin := strings . NewReader ( c . Args )
layer , err := NewLayer ( bin , mediatype )
2023-08-08 16:56:48 -04:00
if err != nil {
return err
}
2023-11-22 13:28:49 -08:00
layers . Add ( layer )
2023-11-14 12:30:34 -08:00
case "template" , "system" :
fn ( api . ProgressResponse { Status : fmt . Sprintf ( "creating %s layer" , c . Name ) } )
2023-11-22 13:28:49 -08:00
bin := strings . NewReader ( c . Args )
layer , err := NewLayer ( bin , mediatype )
2023-07-16 17:02:22 -07:00
if err != nil {
2023-07-17 14:21:27 -07:00
return err
2023-07-16 17:02:22 -07:00
}
2023-07-17 14:21:27 -07:00
2023-11-22 13:28:49 -08:00
layers . Replace ( layer )
2024-01-25 12:12:36 -08:00
case "message" :
messages = append ( messages , c . Args )
2023-07-16 17:02:22 -07:00
default :
2023-07-28 11:29:00 -04:00
params [ c . Name ] = append ( params [ c . Name ] , c . Args )
2023-07-16 17:02:22 -07:00
}
}
2024-01-25 12:12:36 -08:00
if len ( messages ) > 0 {
fn ( api . ProgressResponse { Status : "creating parameters layer" } )
msgs := make ( [ ] api . Message , 0 )
for _ , m := range messages {
// todo: handle images
msg := strings . SplitN ( m , ": " , 2 )
msgs = append ( msgs , api . Message { Role : msg [ 0 ] , Content : msg [ 1 ] } )
}
var b bytes . Buffer
if err := json . NewEncoder ( & b ) . Encode ( msgs ) ; err != nil {
return err
}
layer , err := NewLayer ( & b , "application/vnd.ollama.image.messages" )
if err != nil {
return err
}
layers . Replace ( layer )
}
2023-07-17 12:08:10 -07:00
if len ( params ) > 0 {
2023-11-14 12:30:34 -08:00
fn ( api . ProgressResponse { Status : "creating parameters layer" } )
2023-09-02 14:38:51 -04:00
2023-11-29 09:56:42 -08:00
formattedParams , err := api . FormatParams ( params )
2023-07-16 17:02:22 -07:00
if err != nil {
2023-11-14 12:30:34 -08:00
return err
2023-07-16 17:02:22 -07:00
}
2023-08-04 18:56:40 -04:00
2023-11-14 12:30:34 -08:00
for k , v := range fromParams {
2023-09-05 11:05:03 -07:00
if _ , ok := formattedParams [ k ] ; ! ok {
formattedParams [ k ] = v
}
}
2023-12-11 13:56:22 -08:00
// xxx - can this be removed?
2023-09-12 10:52:57 -07:00
if config . ModelType == "65B" {
2023-11-14 12:30:34 -08:00
if gqa , ok := formattedParams [ "gqa" ] . ( int ) ; ok && gqa == 8 {
2023-09-12 10:52:57 -07:00
config . ModelType = "70B"
}
}
2023-11-14 12:30:34 -08:00
var b bytes . Buffer
if err := json . NewEncoder ( & b ) . Encode ( formattedParams ) ; err != nil {
2023-08-04 18:56:40 -04:00
return err
}
2023-11-14 12:30:34 -08:00
fn ( api . ProgressResponse { Status : "creating config layer" } )
2023-11-22 13:28:49 -08:00
layer , err := NewLayer ( & b , "application/vnd.ollama.image.params" )
2023-07-16 17:02:22 -07:00
if err != nil {
2023-11-14 12:30:34 -08:00
return err
2023-07-16 17:02:22 -07:00
}
2023-11-14 12:30:34 -08:00
2023-11-22 13:28:49 -08:00
layers . Replace ( layer )
2023-08-04 18:56:40 -04:00
}
2023-11-22 13:28:49 -08:00
digests := make ( [ ] string , len ( layers . items ) )
for i , layer := range layers . items {
digests [ i ] = layer . Digest
2023-07-16 17:02:22 -07:00
}
2023-11-22 13:28:49 -08:00
config . RootFS . DiffIDs = digests
2023-11-14 12:30:34 -08:00
2023-11-22 13:28:49 -08:00
var b bytes . Buffer
if err := json . NewEncoder ( & b ) . Encode ( config ) ; err != nil {
2023-07-16 17:02:22 -07:00
return err
}
2023-11-22 13:28:49 -08:00
configLayer , err := NewLayer ( & b , "application/vnd.docker.container.image.v1+json" )
if err != nil {
2023-07-16 17:02:22 -07:00
return err
}
2023-11-22 13:28:49 -08:00
delete ( deleteMap , configLayer . Digest )
2023-07-16 17:02:22 -07:00
2023-11-22 13:28:49 -08:00
for _ , layer := range append ( layers . items , configLayer ) {
committed , err := layer . Commit ( )
2023-07-17 11:03:55 -07:00
if err != nil {
return err
}
2023-07-16 17:02:22 -07:00
2023-11-22 13:28:49 -08:00
status := "writing layer"
if ! committed {
status = "using already created layer"
2023-07-16 17:02:22 -07:00
}
2023-11-22 13:28:49 -08:00
fn ( api . ProgressResponse { Status : fmt . Sprintf ( "%s %s" , status , layer . Digest ) } )
2023-07-16 17:02:22 -07:00
2023-11-22 13:28:49 -08:00
delete ( deleteMap , layer . Digest )
2023-07-16 17:02:22 -07:00
}
2023-11-22 13:28:49 -08:00
fn ( api . ProgressResponse { Status : "writing manifest" } )
if err := WriteManifest ( name , configLayer , layers . items ) ; err != nil {
2023-10-27 10:19:59 -04:00
return err
}
2023-07-16 17:02:22 -07:00
2023-11-22 13:28:49 -08:00
if noprune := os . Getenv ( "OLLAMA_NOPRUNE" ) ; noprune == "" {
if err := deleteUnusedLayers ( nil , deleteMap , false ) ; err != nil {
return err
2023-07-16 17:02:22 -07:00
}
}
2023-11-22 13:28:49 -08:00
fn ( api . ProgressResponse { Status : "success" } )
return nil
2023-07-16 17:02:22 -07:00
}
2023-07-24 11:27:28 -04:00
func CopyModel ( src , dest string ) error {
2023-08-22 09:39:42 -07:00
srcModelPath := ParseModelPath ( src )
2023-10-27 10:19:59 -04:00
srcPath , err := srcModelPath . GetManifestPath ( )
2023-08-21 21:56:56 -07:00
if err != nil {
return err
}
2023-08-22 09:39:42 -07:00
destModelPath := ParseModelPath ( dest )
2023-10-27 10:19:59 -04:00
destPath , err := destModelPath . GetManifestPath ( )
2023-07-24 11:27:28 -04:00
if err != nil {
return err
}
2023-10-27 10:19:59 -04:00
if err := os . MkdirAll ( filepath . Dir ( destPath ) , 0 o755 ) ; err != nil {
return err
}
2023-07-24 11:27:28 -04:00
// copy the file
2023-07-28 10:38:15 -07:00
input , err := os . ReadFile ( srcPath )
2023-07-24 11:27:28 -04:00
if err != nil {
fmt . Println ( "Error reading file:" , err )
return err
}
2023-07-28 10:38:15 -07:00
err = os . WriteFile ( destPath , input , 0 o644 )
2023-07-24 11:27:28 -04:00
if err != nil {
fmt . Println ( "Error reading file:" , err )
return err
}
return nil
}
2023-11-14 12:30:34 -08:00
func deleteUnusedLayers ( skipModelPath * ModelPath , deleteMap map [ string ] struct { } , dryRun bool ) error {
2023-07-20 16:09:23 -07:00
fp , err := GetManifestPath ( )
if err != nil {
return err
}
2023-08-30 14:31:12 -04:00
walkFunc := func ( path string , info os . FileInfo , _ error ) error {
if info . IsDir ( ) {
return nil
2023-07-20 16:09:23 -07:00
}
2023-08-30 14:31:12 -04:00
dir , file := filepath . Split ( path )
dir = strings . Trim ( strings . TrimPrefix ( dir , fp ) , string ( os . PathSeparator ) )
tag := strings . Join ( [ ] string { dir , file } , ":" )
fmp := ParseModelPath ( tag )
2023-07-20 16:09:23 -07:00
2023-08-30 14:31:12 -04:00
// skip the manifest we're trying to delete
2023-09-11 11:46:35 -07:00
if skipModelPath != nil && skipModelPath . GetFullTagname ( ) == fmp . GetFullTagname ( ) {
2023-08-30 14:31:12 -04:00
return nil
2023-07-20 16:09:23 -07:00
}
2023-08-30 14:31:12 -04:00
// save (i.e. delete from the deleteMap) any files used in other manifests
manifest , _ , err := GetManifest ( fmp )
if err != nil {
2023-12-15 14:07:34 -08:00
// nolint: nilerr
2023-08-30 14:31:12 -04:00
return nil
}
for _ , layer := range manifest . Layers {
delete ( deleteMap , layer . Digest )
}
delete ( deleteMap , manifest . Config . Digest )
2023-07-20 16:09:23 -07:00
return nil
2023-08-30 14:31:12 -04:00
}
if err := filepath . Walk ( fp , walkFunc ) ; err != nil {
2023-07-31 15:26:18 -07:00
return err
}
2023-07-20 16:09:23 -07:00
// only delete the files which are still in the deleteMap
2023-11-14 12:30:34 -08:00
for k := range deleteMap {
fp , err := GetBlobsPath ( k )
if err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "couldn't get file path for '%s': %v" , k , err ) )
2023-11-14 12:30:34 -08:00
continue
}
if ! dryRun {
if err := os . Remove ( fp ) ; err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "couldn't remove file '%s': %v" , fp , err ) )
2023-07-21 17:30:40 -07:00
continue
}
2023-11-14 12:30:34 -08:00
} else {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "wanted to remove: %s" , fp ) )
2023-07-20 16:09:23 -07:00
}
}
2023-09-11 11:46:35 -07:00
return nil
}
func PruneLayers ( ) error {
2023-11-14 12:30:34 -08:00
deleteMap := make ( map [ string ] struct { } )
2023-09-11 11:46:35 -07:00
p , err := GetBlobsPath ( "" )
if err != nil {
return err
}
blobs , err := os . ReadDir ( p )
if err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "couldn't read dir '%s': %v" , p , err ) )
2023-09-11 11:46:35 -07:00
return err
}
for _ , blob := range blobs {
name := blob . Name ( )
if runtime . GOOS == "windows" {
name = strings . ReplaceAll ( name , "-" , ":" )
}
2023-11-14 14:27:51 -08:00
if strings . HasPrefix ( name , "sha256:" ) {
deleteMap [ name ] = struct { } { }
}
2023-09-11 11:46:35 -07:00
}
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "total blobs: %d" , len ( deleteMap ) ) )
2023-09-11 11:46:35 -07:00
err = deleteUnusedLayers ( nil , deleteMap , false )
if err != nil {
return err
}
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "total unused blobs removed: %d" , len ( deleteMap ) ) )
2023-09-11 11:46:35 -07:00
return nil
}
2023-09-26 17:28:14 -07:00
func PruneDirectory ( path string ) error {
info , err := os . Lstat ( path )
if err != nil {
return err
}
if info . IsDir ( ) && info . Mode ( ) & os . ModeSymlink == 0 {
entries , err := os . ReadDir ( path )
if err != nil {
return err
}
for _ , entry := range entries {
if err := PruneDirectory ( filepath . Join ( path , entry . Name ( ) ) ) ; err != nil {
return err
}
}
entries , err = os . ReadDir ( path )
if err != nil {
return err
}
if len ( entries ) > 0 {
return nil
}
return os . Remove ( path )
}
return nil
}
2023-09-11 11:46:35 -07:00
func DeleteModel ( name string ) error {
mp := ParseModelPath ( name )
manifest , _ , err := GetManifest ( mp )
if err != nil {
return err
}
2023-11-14 12:30:34 -08:00
deleteMap := make ( map [ string ] struct { } )
2023-09-11 11:46:35 -07:00
for _ , layer := range manifest . Layers {
2023-11-14 12:30:34 -08:00
deleteMap [ layer . Digest ] = struct { } { }
2023-09-11 11:46:35 -07:00
}
2023-11-14 12:30:34 -08:00
deleteMap [ manifest . Config . Digest ] = struct { } { }
2023-09-11 11:46:35 -07:00
err = deleteUnusedLayers ( & mp , deleteMap , false )
if err != nil {
return err
}
2023-10-27 10:19:59 -04:00
fp , err := mp . GetManifestPath ( )
2023-07-20 16:09:23 -07:00
if err != nil {
return err
}
err = os . Remove ( fp )
if err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "couldn't remove manifest file '%s': %v" , fp , err ) )
2023-07-20 16:09:23 -07:00
return err
}
return nil
}
2023-09-06 11:04:17 -07:00
func ShowModelfile ( model * Model ) ( string , error ) {
2023-10-17 15:40:06 -07:00
var mt struct {
2023-09-06 11:04:17 -07:00
* Model
2023-10-17 15:40:06 -07:00
From string
2023-10-17 15:53:46 -07:00
Parameters map [ string ] [ ] any
2023-09-06 11:04:17 -07:00
}
2023-10-17 15:53:46 -07:00
mt . Parameters = make ( map [ string ] [ ] any )
2023-09-06 11:04:17 -07:00
for k , v := range model . Options {
2023-10-17 15:53:46 -07:00
if s , ok := v . ( [ ] any ) ; ok {
mt . Parameters [ k ] = s
continue
2023-09-06 11:04:17 -07:00
}
2023-10-17 15:53:46 -07:00
mt . Parameters [ k ] = [ ] any { v }
2023-09-06 11:04:17 -07:00
}
2023-10-17 15:40:06 -07:00
mt . Model = model
mt . From = model . ModelPath
2023-09-06 11:04:17 -07:00
2024-01-25 12:12:36 -08:00
if model . ParentModel != "" {
mt . From = model . ParentModel
2023-09-06 11:04:17 -07:00
}
modelFile := ` # Modelfile generated by "ollama show"
# To build a new Modelfile based on this one , replace the FROM line with :
# FROM { { . ShortName } }
FROM { { . From } }
TEMPLATE "" "{{ .Template }}" ""
2023-10-17 15:25:43 -07:00
{ { - if . System } }
2023-09-06 11:04:17 -07:00
SYSTEM "" "{{ .System }}" ""
2023-10-17 15:25:43 -07:00
{ { - end } }
2023-10-17 15:28:38 -07:00
{ { - range $ adapter := . AdapterPaths } }
ADAPTER { { $ adapter } }
{ { - end } }
2023-10-17 15:40:06 -07:00
2023-10-17 15:53:46 -07:00
{ { - range $ k , $ v := . Parameters } }
{ { - range $ parameter := $ v } }
PARAMETER { { $ k } } { { printf "%#v" $ parameter } }
{ { - end } }
2023-10-17 15:40:06 -07:00
{ { - end } } `
2023-09-06 11:04:17 -07:00
tmpl , err := template . New ( "" ) . Parse ( modelFile )
if err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "error parsing template: %q" , err ) )
2023-09-06 11:04:17 -07:00
return "" , err
}
var buf bytes . Buffer
if err = tmpl . Execute ( & buf , mt ) ; err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "error executing template: %q" , err ) )
2023-09-06 11:04:17 -07:00
return "" , err
}
return buf . String ( ) , nil
}
2024-02-14 11:29:49 -08:00
func PushModel ( ctx context . Context , name string , regOpts * registryOptions , fn func ( api . ProgressResponse ) ) error {
2023-08-22 09:39:42 -07:00
mp := ParseModelPath ( name )
2023-07-18 18:51:30 -07:00
fn ( api . ProgressResponse { Status : "retrieving manifest" } )
2023-08-22 09:39:42 -07:00
if mp . ProtocolScheme == "http" && ! regOpts . Insecure {
return fmt . Errorf ( "insecure protocol http" )
}
2023-08-28 20:50:24 -07:00
manifest , _ , err := GetManifest ( mp )
2023-07-16 17:02:22 -07:00
if err != nil {
2023-07-18 18:51:30 -07:00
fn ( api . ProgressResponse { Status : "couldn't retrieve manifest" } )
2023-07-16 17:02:22 -07:00
return err
}
var layers [ ] * Layer
2023-07-31 21:37:40 -04:00
layers = append ( layers , manifest . Layers ... )
2023-11-22 13:28:49 -08:00
layers = append ( layers , manifest . Config )
2023-07-16 17:02:22 -07:00
for _ , layer := range layers {
2023-10-09 10:24:27 -07:00
if err := uploadBlob ( ctx , mp , layer , regOpts , fn ) ; err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "error uploading blob: %v" , err ) )
2023-11-16 16:44:18 -05:00
if errors . Is ( err , errUnauthorized ) {
return fmt . Errorf ( "unable to push %s, make sure this namespace exists and you are authorized to push to it" , ParseModelPath ( name ) . GetNamespaceRepository ( ) )
}
2023-07-16 17:02:22 -07:00
return err
}
2023-07-18 18:51:30 -07:00
}
2023-07-22 17:31:26 -07:00
fn ( api . ProgressResponse { Status : "pushing manifest" } )
2023-08-21 18:38:31 -07:00
requestURL := mp . BaseURL ( )
requestURL = requestURL . JoinPath ( "v2" , mp . GetNamespaceRepository ( ) , "manifests" , mp . Tag )
2023-07-16 17:02:22 -07:00
manifestJSON , err := json . Marshal ( manifest )
if err != nil {
return err
}
2023-08-21 18:24:42 -07:00
headers := make ( http . Header )
headers . Set ( "Content-Type" , "application/vnd.docker.distribution.manifest.v2+json" )
2023-11-02 13:10:58 -07:00
resp , err := makeRequestWithRetry ( ctx , http . MethodPut , requestURL , headers , bytes . NewReader ( manifestJSON ) , regOpts )
2023-07-16 17:02:22 -07:00
if err != nil {
return err
}
defer resp . Body . Close ( )
2023-07-22 17:31:26 -07:00
fn ( api . ProgressResponse { Status : "success" } )
2023-07-16 17:02:22 -07:00
return nil
}
2024-02-14 11:29:49 -08:00
func PullModel ( ctx context . Context , name string , regOpts * registryOptions , fn func ( api . ProgressResponse ) ) error {
2023-08-22 09:39:42 -07:00
mp := ParseModelPath ( name )
2023-09-11 11:46:35 -07:00
var manifest * ManifestV2
var err error
var noprune string
// build deleteMap to prune unused layers
2023-11-14 12:30:34 -08:00
deleteMap := make ( map [ string ] struct { } )
2023-09-11 11:46:35 -07:00
if noprune = os . Getenv ( "OLLAMA_NOPRUNE" ) ; noprune == "" {
manifest , _ , err = GetManifest ( mp )
if err != nil && ! errors . Is ( err , os . ErrNotExist ) {
return err
}
if manifest != nil {
for _ , l := range manifest . Layers {
2023-11-14 12:30:34 -08:00
deleteMap [ l . Digest ] = struct { } { }
2023-09-11 11:46:35 -07:00
}
2023-11-14 12:30:34 -08:00
deleteMap [ manifest . Config . Digest ] = struct { } { }
2023-09-11 11:46:35 -07:00
}
}
2023-08-22 09:39:42 -07:00
if mp . ProtocolScheme == "http" && ! regOpts . Insecure {
return fmt . Errorf ( "insecure protocol http" )
2023-08-21 21:56:56 -07:00
}
2023-07-16 17:02:22 -07:00
2023-07-18 18:51:30 -07:00
fn ( api . ProgressResponse { Status : "pulling manifest" } )
2023-07-16 17:02:22 -07:00
2023-09-11 11:46:35 -07:00
manifest , err = pullModelManifest ( ctx , mp , regOpts )
2023-07-16 17:02:22 -07:00
if err != nil {
2023-07-24 17:48:17 -04:00
return fmt . Errorf ( "pull model manifest: %s" , err )
2023-07-16 17:02:22 -07:00
}
var layers [ ] * Layer
2023-07-20 20:18:00 +02:00
layers = append ( layers , manifest . Layers ... )
2023-11-22 13:28:49 -08:00
layers = append ( layers , manifest . Config )
2023-07-16 17:02:22 -07:00
for _ , layer := range layers {
2023-08-15 15:07:19 -03:00
if err := downloadBlob (
ctx ,
downloadOpts {
mp : mp ,
digest : layer . Digest ,
regOpts : regOpts ,
fn : fn ,
} ) ; err != nil {
2023-07-16 17:02:22 -07:00
return err
}
2023-09-11 11:46:35 -07:00
delete ( deleteMap , layer . Digest )
2023-07-16 17:02:22 -07:00
}
2023-09-11 11:46:35 -07:00
delete ( deleteMap , manifest . Config . Digest )
2023-07-16 17:02:22 -07:00
2023-07-20 11:44:05 -07:00
fn ( api . ProgressResponse { Status : "verifying sha256 digest" } )
for _ , layer := range layers {
if err := verifyBlob ( layer . Digest ) ; err != nil {
2023-07-24 14:53:01 -04:00
if errors . Is ( err , errDigestMismatch ) {
// something went wrong, delete the blob
fp , err := GetBlobsPath ( layer . Digest )
if err != nil {
return err
}
if err := os . Remove ( fp ) ; err != nil {
// log this, but return the original error
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "couldn't remove file with digest mismatch '%s': %v" , fp , err ) )
2023-07-24 14:53:01 -04:00
}
}
2023-07-20 11:44:05 -07:00
return err
}
}
2023-07-18 18:51:30 -07:00
fn ( api . ProgressResponse { Status : "writing manifest" } )
2023-07-16 17:02:22 -07:00
2023-07-17 11:03:55 -07:00
manifestJSON , err := json . Marshal ( manifest )
2023-07-16 17:02:22 -07:00
if err != nil {
return err
}
2023-10-27 10:19:59 -04:00
fp , err := mp . GetManifestPath ( )
2023-07-16 17:02:22 -07:00
if err != nil {
return err
}
2023-10-27 10:19:59 -04:00
if err := os . MkdirAll ( filepath . Dir ( fp ) , 0 o755 ) ; err != nil {
return err
}
2023-07-16 17:02:22 -07:00
2023-07-20 20:18:00 +02:00
err = os . WriteFile ( fp , manifestJSON , 0 o644 )
2023-07-16 17:02:22 -07:00
if err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "couldn't write to %s" , fp ) )
2023-07-16 17:02:22 -07:00
return err
}
2023-09-11 11:46:35 -07:00
if noprune == "" {
fn ( api . ProgressResponse { Status : "removing any unused layers" } )
err = deleteUnusedLayers ( nil , deleteMap , false )
if err != nil {
return err
}
}
2023-07-18 18:51:30 -07:00
fn ( api . ProgressResponse { Status : "success" } )
2023-07-16 17:02:22 -07:00
return nil
}
2024-02-14 11:29:49 -08:00
func pullModelManifest ( ctx context . Context , mp ModelPath , regOpts * registryOptions ) ( * ManifestV2 , error ) {
2023-08-21 18:38:31 -07:00
requestURL := mp . BaseURL ( ) . JoinPath ( "v2" , mp . GetNamespaceRepository ( ) , "manifests" , mp . Tag )
2023-07-16 17:02:22 -07:00
2023-08-21 18:24:42 -07:00
headers := make ( http . Header )
headers . Set ( "Accept" , "application/vnd.docker.distribution.manifest.v2+json" )
2023-11-02 13:13:32 -07:00
resp , err := makeRequestWithRetry ( ctx , http . MethodGet , requestURL , headers , nil , regOpts )
2023-07-16 17:02:22 -07:00
if err != nil {
return nil , err
}
defer resp . Body . Close ( )
var m * ManifestV2
if err := json . NewDecoder ( resp . Body ) . Decode ( & m ) ; err != nil {
return nil , err
}
return m , err
}
// GetSHA256Digest returns the SHA256 hash of a given buffer and returns it, and the size of buffer
2023-09-28 10:00:34 -07:00
func GetSHA256Digest ( r io . Reader ) ( string , int64 ) {
2023-07-18 17:14:12 -07:00
h := sha256 . New ( )
n , err := io . Copy ( h , r )
if err != nil {
log . Fatal ( err )
}
2023-09-28 10:00:34 -07:00
return fmt . Sprintf ( "sha256:%x" , h . Sum ( nil ) ) , n
2023-07-16 17:02:22 -07:00
}
2023-11-16 16:44:18 -05:00
var errUnauthorized = fmt . Errorf ( "unauthorized" )
2024-02-14 11:29:49 -08:00
func makeRequestWithRetry ( ctx context . Context , method string , requestURL * url . URL , headers http . Header , body io . ReadSeeker , regOpts * registryOptions ) ( * http . Response , error ) {
2024-01-12 13:32:24 -08:00
for i := 0 ; i < 2 ; i ++ {
2024-02-14 11:29:49 -08:00
resp , err := makeRequest ( ctx , method , requestURL , headers , body , regOpts )
2023-08-17 12:35:29 -07:00
if err != nil {
2024-01-12 13:32:24 -08:00
if ! errors . Is ( err , context . Canceled ) {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "request failed: %v" , err ) )
2024-01-12 13:32:24 -08:00
}
2023-08-17 12:35:29 -07:00
return nil , err
}
2024-01-12 13:32:24 -08:00
switch {
case resp . StatusCode == http . StatusUnauthorized :
// Handle authentication error with one retry
2024-02-14 11:29:49 -08:00
challenge := parseRegistryChallenge ( resp . Header . Get ( "www-authenticate" ) )
token , err := getAuthorizationToken ( ctx , challenge )
2023-08-17 12:35:29 -07:00
if err != nil {
return nil , err
}
2024-01-12 13:32:24 -08:00
regOpts . Token = token
if body != nil {
_ , err = body . Seek ( 0 , io . SeekStart )
if err != nil {
return nil , err
}
}
case resp . StatusCode == http . StatusNotFound :
return nil , os . ErrNotExist
case resp . StatusCode >= http . StatusBadRequest :
responseBody , err := io . ReadAll ( resp . Body )
if err != nil {
return nil , fmt . Errorf ( "%d: %s" , resp . StatusCode , err )
}
return nil , fmt . Errorf ( "%d: %s" , resp . StatusCode , responseBody )
default :
return resp , nil
2023-08-17 12:35:29 -07:00
}
}
2024-01-12 13:32:24 -08:00
return nil , errUnauthorized
2023-08-17 12:35:29 -07:00
}
2024-02-14 11:29:49 -08:00
func makeRequest ( ctx context . Context , method string , requestURL * url . URL , headers http . Header , body io . Reader , regOpts * registryOptions ) ( * http . Response , error ) {
if requestURL . Scheme != "http" && regOpts != nil && regOpts . Insecure {
requestURL . Scheme = "http"
}
req , err := http . NewRequestWithContext ( ctx , method , requestURL . String ( ) , body )
if err != nil {
return nil , err
}
if headers != nil {
req . Header = headers
}
if regOpts != nil {
if regOpts . Token != "" {
req . Header . Set ( "Authorization" , "Bearer " + regOpts . Token )
} else if regOpts . Username != "" && regOpts . Password != "" {
req . SetBasicAuth ( regOpts . Username , regOpts . Password )
}
}
req . Header . Set ( "User-Agent" , fmt . Sprintf ( "ollama/%s (%s %s) Go/%s" , version . Version , runtime . GOARCH , runtime . GOOS , runtime . Version ( ) ) )
if s := req . Header . Get ( "Content-Length" ) ; s != "" {
contentLength , err := strconv . ParseInt ( s , 10 , 64 )
if err != nil {
return nil , err
}
req . ContentLength = contentLength
}
proxyURL , err := http . ProxyFromEnvironment ( req )
if err != nil {
return nil , err
}
client := http . Client {
Transport : & http . Transport {
Proxy : http . ProxyURL ( proxyURL ) ,
} ,
}
resp , err := client . Do ( req )
if err != nil {
return nil , err
}
return resp , nil
}
2023-08-10 11:34:25 -07:00
func getValue ( header , key string ) string {
startIdx := strings . Index ( header , key + "=" )
if startIdx == - 1 {
return ""
}
// Move the index to the starting quote after the key.
startIdx += len ( key ) + 2
endIdx := startIdx
for endIdx < len ( header ) {
if header [ endIdx ] == '"' {
if endIdx + 1 < len ( header ) && header [ endIdx + 1 ] != ',' { // If the next character isn't a comma, continue
endIdx ++
continue
}
break
}
endIdx ++
}
return header [ startIdx : endIdx ]
}
2024-02-14 11:29:49 -08:00
func parseRegistryChallenge ( authStr string ) registryChallenge {
2023-08-10 11:34:25 -07:00
authStr = strings . TrimPrefix ( authStr , "Bearer " )
2024-02-14 11:29:49 -08:00
return registryChallenge {
2023-08-10 11:34:25 -07:00
Realm : getValue ( authStr , "realm" ) ,
Service : getValue ( authStr , "service" ) ,
Scope : getValue ( authStr , "scope" ) ,
}
}
2023-07-24 14:53:01 -04:00
var errDigestMismatch = fmt . Errorf ( "digest mismatch, file must be downloaded again" )
2023-07-20 11:44:05 -07:00
func verifyBlob ( digest string ) error {
fp , err := GetBlobsPath ( digest )
if err != nil {
return err
}
f , err := os . Open ( fp )
if err != nil {
return err
}
defer f . Close ( )
fileDigest , _ := GetSHA256Digest ( f )
if digest != fileDigest {
2023-07-24 14:53:01 -04:00
return fmt . Errorf ( "%w: want %s, got %s" , errDigestMismatch , digest , fileDigest )
2023-07-20 11:44:05 -07:00
}
return nil
}