manifest.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. package server
  2. import (
  3. "crypto/sha256"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "io"
  8. "log/slog"
  9. "os"
  10. "path/filepath"
  11. "github.com/Masterminds/semver/v3"
  12. "github.com/ollama/ollama/types/model"
  13. )
  14. type Manifest struct {
  15. SchemaVersion int `json:"schemaVersion"`
  16. MediaType string `json:"mediaType"`
  17. Config *Layer `json:"config"`
  18. Layers []*Layer `json:"layers"`
  19. Ollama *semver.Version `json:"ollama,omitempty"`
  20. filepath string
  21. fi os.FileInfo
  22. digest string
  23. }
  24. func (m *Manifest) Size() (size int64) {
  25. for _, layer := range append(m.Layers, m.Config) {
  26. size += layer.Size
  27. }
  28. return
  29. }
  30. func (m *Manifest) Remove() error {
  31. if err := os.Remove(m.filepath); err != nil {
  32. return err
  33. }
  34. manifests, err := GetManifestPath()
  35. if err != nil {
  36. return err
  37. }
  38. return PruneDirectory(manifests)
  39. }
  40. func (m *Manifest) RemoveLayers() error {
  41. for _, layer := range append(m.Layers, m.Config) {
  42. if err := layer.Remove(); errors.Is(err, os.ErrNotExist) {
  43. slog.Debug("layer does not exist", "digest", layer.Digest)
  44. } else if err != nil {
  45. return err
  46. }
  47. }
  48. return nil
  49. }
  50. func ParseNamedManifest(n model.Name) (*Manifest, error) {
  51. if !n.IsFullyQualified() {
  52. return nil, model.Unqualified(n)
  53. }
  54. manifests, err := GetManifestPath()
  55. if err != nil {
  56. return nil, err
  57. }
  58. p := filepath.Join(manifests, n.Filepath())
  59. var m Manifest
  60. f, err := os.Open(p)
  61. if err != nil {
  62. return nil, err
  63. }
  64. defer f.Close()
  65. fi, err := f.Stat()
  66. if err != nil {
  67. return nil, err
  68. }
  69. sha256sum := sha256.New()
  70. if err := json.NewDecoder(io.TeeReader(f, sha256sum)).Decode(&m); err != nil {
  71. return nil, err
  72. }
  73. m.filepath = p
  74. m.fi = fi
  75. m.digest = fmt.Sprintf("%x", sha256sum.Sum(nil))
  76. return &m, nil
  77. }
  78. func WriteManifest(name model.Name, ollama *semver.Version, config *Layer, layers []*Layer) error {
  79. manifests, err := GetManifestPath()
  80. if err != nil {
  81. return err
  82. }
  83. p := filepath.Join(manifests, name.Filepath())
  84. if err := os.MkdirAll(filepath.Dir(p), 0o755); err != nil {
  85. return err
  86. }
  87. f, err := os.Create(p)
  88. if err != nil {
  89. return err
  90. }
  91. defer f.Close()
  92. m := Manifest{
  93. SchemaVersion: 2,
  94. MediaType: "application/vnd.docker.distribution.manifest.v2+json",
  95. Config: config,
  96. Layers: layers,
  97. Ollama: ollama,
  98. }
  99. return json.NewEncoder(f).Encode(m)
  100. }
  101. func Manifests() (map[model.Name]*Manifest, error) {
  102. manifests, err := GetManifestPath()
  103. if err != nil {
  104. return nil, err
  105. }
  106. // TODO(mxyng): use something less brittle
  107. matches, err := filepath.Glob(filepath.Join(manifests, "*", "*", "*", "*"))
  108. if err != nil {
  109. return nil, err
  110. }
  111. ms := make(map[model.Name]*Manifest)
  112. for _, match := range matches {
  113. fi, err := os.Stat(match)
  114. if err != nil {
  115. return nil, err
  116. }
  117. if !fi.IsDir() {
  118. rel, err := filepath.Rel(manifests, match)
  119. if err != nil {
  120. slog.Warn("bad filepath", "path", match, "error", err)
  121. continue
  122. }
  123. n := model.ParseNameFromFilepath(rel)
  124. if !n.IsValid() {
  125. slog.Warn("bad manifest name", "path", rel, "error", err)
  126. continue
  127. }
  128. m, err := ParseNamedManifest(n)
  129. if err != nil {
  130. slog.Warn("bad manifest", "name", n, "error", err)
  131. continue
  132. }
  133. ms[n] = m
  134. }
  135. }
  136. return ms, nil
  137. }