manifest.go 2.92 KB
Newer Older
1
2
3
4
5
package server

import (
	"crypto/sha256"
	"encoding/json"
6
	"errors"
7
8
	"fmt"
	"io"
9
	"log/slog"
10
11
12
13
14
15
16
17
	"os"
	"path/filepath"

	"github.com/ollama/ollama/types/model"
)

type Manifest struct {
	ManifestV2
18
19

	filepath string
20
	fi       os.FileInfo
21
	digest   string
22
23
24
25
26
27
28
29
30
31
}

func (m *Manifest) Size() (size int64) {
	for _, layer := range append(m.Layers, m.Config) {
		size += layer.Size
	}

	return
}

32
33
34
35
36
37
38
39
40
41
42
43
44
func (m *Manifest) Remove() error {
	if err := os.Remove(m.filepath); err != nil {
		return err
	}

	manifests, err := GetManifestPath()
	if err != nil {
		return err
	}

	return PruneDirectory(manifests)
}

45
46
func (m *Manifest) RemoveLayers() error {
	for _, layer := range append(m.Layers, m.Config) {
47
48
49
		if err := layer.Remove(); errors.Is(err, os.ErrNotExist) {
			slog.Debug("layer does not exist", "digest", layer.Digest)
		} else if err != nil {
50
51
52
53
54
55
56
			return err
		}
	}

	return nil
}

57
58
59
func ParseNamedManifest(n model.Name) (*Manifest, error) {
	if !n.IsFullyQualified() {
		return nil, model.Unqualified(n)
60
61
62
63
64
65
66
	}

	manifests, err := GetManifestPath()
	if err != nil {
		return nil, err
	}

67
68
69
70
	p := filepath.Join(manifests, n.Filepath())

	var m ManifestV2
	f, err := os.Open(p)
71
72
73
	if err != nil {
		return nil, err
	}
74
	defer f.Close()
75

76
77
78
79
80
	fi, err := f.Stat()
	if err != nil {
		return nil, err
	}

81
	sha256sum := sha256.New()
82
	if err := json.NewDecoder(io.TeeReader(f, sha256sum)).Decode(&m); err != nil {
83
84
85
86
		return nil, err
	}

	return &Manifest{
87
88
		ManifestV2: m,
		filepath:   p,
89
		fi:         fi,
90
		digest:     fmt.Sprintf("%x", sha256sum.Sum(nil)),
91
92
93
	}, nil
}

94
95
96
97
func WriteManifest(name model.Name, config *Layer, layers []*Layer) error {
	manifests, err := GetManifestPath()
	if err != nil {
		return err
98
99
	}

100
101
	p := filepath.Join(manifests, name.Filepath())
	if err := os.MkdirAll(filepath.Dir(p), 0o755); err != nil {
102
103
104
		return err
	}

105
	f, err := os.Create(p)
106
107
108
	if err != nil {
		return err
	}
109
	defer f.Close()
110

111
112
113
114
115
	m := ManifestV2{
		SchemaVersion: 2,
		MediaType:     "application/vnd.docker.distribution.manifest.v2+json",
		Config:        config,
		Layers:        layers,
116
117
	}

118
	return json.NewEncoder(f).Encode(m)
119
}
120
121
122
123
124
125
126
127

func Manifests() (map[model.Name]*Manifest, error) {
	manifests, err := GetManifestPath()
	if err != nil {
		return nil, err
	}

	// TODO(mxyng): use something less brittle
Michael Yang's avatar
Michael Yang committed
128
	matches, err := filepath.Glob(filepath.Join(manifests, "*", "*", "*", "*"))
129
130
131
132
133
134
	if err != nil {
		return nil, err
	}

	ms := make(map[model.Name]*Manifest)
	for _, match := range matches {
Michael Yang's avatar
Michael Yang committed
135
		fi, err := os.Stat(match)
136
137
138
139
		if err != nil {
			return nil, err
		}

Michael Yang's avatar
Michael Yang committed
140
141
142
143
144
145
146
147
148
149
150
151
152
		if !fi.IsDir() {
			rel, err := filepath.Rel(manifests, match)
			if err != nil {
				slog.Warn("bad filepath", "path", match, "error", err)
				continue
			}

			n := model.ParseNameFromFilepath(rel)
			if !n.IsValid() {
				slog.Warn("bad manifest name", "path", rel, "error", err)
				continue
			}

153
154
			m, err := ParseNamedManifest(n)
			if err != nil {
155
156
				slog.Warn("bad manifest", "name", n, "error", err)
				continue
157
158
159
160
161
162
163
164
			}

			ms[n] = m
		}
	}

	return ms, nil
}