Skip to content
GitLab
Menu
Projects
Groups
Snippets
Loading...
Help
Help
Support
Community forum
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
Menu
Open sidebar
OpenDAS
ollama
Commits
69eb06c4
Unverified
Commit
69eb06c4
authored
Aug 07, 2024
by
Jesse Gross
Committed by
GitHub
Aug 07, 2024
Browse files
Merge pull request #6145 from ollama/jessegross/bug5840
Fix crash on startup when trying to clean up unused files (#5840)
parents
de4fc297
1829fb61
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
64 additions
and
37 deletions
+64
-37
server/images.go
server/images.go
+27
-18
server/layer.go
server/layer.go
+14
-1
server/manifest.go
server/manifest.go
+10
-8
server/routes.go
server/routes.go
+13
-10
No files found.
server/images.go
View file @
69eb06c4
...
@@ -250,19 +250,21 @@ func GetModel(name string) (*Model, error) {
...
@@ -250,19 +250,21 @@ func GetModel(name string) (*Model, error) {
Template
:
template
.
DefaultTemplate
,
Template
:
template
.
DefaultTemplate
,
}
}
filename
,
err
:=
GetBlobsPath
(
manifest
.
Config
.
Digest
)
if
manifest
.
Config
.
Digest
!=
""
{
if
err
!=
nil
{
filename
,
err
:=
GetBlobsPath
(
manifest
.
Config
.
Digest
)
return
nil
,
err
if
err
!=
nil
{
}
return
nil
,
err
}
configFile
,
err
:=
os
.
Open
(
filename
)
configFile
,
err
:=
os
.
Open
(
filename
)
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
}
}
defer
configFile
.
Close
()
defer
configFile
.
Close
()
if
err
:=
json
.
NewDecoder
(
configFile
)
.
Decode
(
&
model
.
Config
);
err
!=
nil
{
if
err
:=
json
.
NewDecoder
(
configFile
)
.
Decode
(
&
model
.
Config
);
err
!=
nil
{
return
nil
,
err
return
nil
,
err
}
}
}
for
_
,
layer
:=
range
manifest
.
Layers
{
for
_
,
layer
:=
range
manifest
.
Layers
{
...
@@ -714,8 +716,7 @@ func deleteUnusedLayers(skipModelPath *ModelPath, deleteMap map[string]struct{})
...
@@ -714,8 +716,7 @@ func deleteUnusedLayers(skipModelPath *ModelPath, deleteMap map[string]struct{})
// save (i.e. delete from the deleteMap) any files used in other manifests
// save (i.e. delete from the deleteMap) any files used in other manifests
manifest
,
_
,
err
:=
GetManifest
(
fmp
)
manifest
,
_
,
err
:=
GetManifest
(
fmp
)
if
err
!=
nil
{
if
err
!=
nil
{
//nolint:nilerr
return
err
return
nil
}
}
for
_
,
layer
:=
range
manifest
.
Layers
{
for
_
,
layer
:=
range
manifest
.
Layers
{
...
@@ -782,7 +783,8 @@ func PruneLayers() error {
...
@@ -782,7 +783,8 @@ func PruneLayers() error {
err
=
deleteUnusedLayers
(
nil
,
deleteMap
)
err
=
deleteUnusedLayers
(
nil
,
deleteMap
)
if
err
!=
nil
{
if
err
!=
nil
{
return
err
slog
.
Error
(
fmt
.
Sprintf
(
"couldn't remove unused layers: %v"
,
err
))
return
nil
}
}
slog
.
Info
(
fmt
.
Sprintf
(
"total unused blobs removed: %d"
,
len
(
deleteMap
)))
slog
.
Info
(
fmt
.
Sprintf
(
"total unused blobs removed: %d"
,
len
(
deleteMap
)))
...
@@ -839,7 +841,9 @@ func PushModel(ctx context.Context, name string, regOpts *registryOptions, fn fu
...
@@ -839,7 +841,9 @@ func PushModel(ctx context.Context, name string, regOpts *registryOptions, fn fu
var
layers
[]
*
Layer
var
layers
[]
*
Layer
layers
=
append
(
layers
,
manifest
.
Layers
...
)
layers
=
append
(
layers
,
manifest
.
Layers
...
)
layers
=
append
(
layers
,
manifest
.
Config
)
if
manifest
.
Config
.
Digest
!=
""
{
layers
=
append
(
layers
,
&
manifest
.
Config
)
}
for
_
,
layer
:=
range
layers
{
for
_
,
layer
:=
range
layers
{
if
err
:=
uploadBlob
(
ctx
,
mp
,
layer
,
regOpts
,
fn
);
err
!=
nil
{
if
err
:=
uploadBlob
(
ctx
,
mp
,
layer
,
regOpts
,
fn
);
err
!=
nil
{
...
@@ -890,7 +894,9 @@ func PullModel(ctx context.Context, name string, regOpts *registryOptions, fn fu
...
@@ -890,7 +894,9 @@ func PullModel(ctx context.Context, name string, regOpts *registryOptions, fn fu
for
_
,
l
:=
range
manifest
.
Layers
{
for
_
,
l
:=
range
manifest
.
Layers
{
deleteMap
[
l
.
Digest
]
=
struct
{}{}
deleteMap
[
l
.
Digest
]
=
struct
{}{}
}
}
deleteMap
[
manifest
.
Config
.
Digest
]
=
struct
{}{}
if
manifest
.
Config
.
Digest
!=
""
{
deleteMap
[
manifest
.
Config
.
Digest
]
=
struct
{}{}
}
}
}
}
}
...
@@ -907,7 +913,9 @@ func PullModel(ctx context.Context, name string, regOpts *registryOptions, fn fu
...
@@ -907,7 +913,9 @@ func PullModel(ctx context.Context, name string, regOpts *registryOptions, fn fu
var
layers
[]
*
Layer
var
layers
[]
*
Layer
layers
=
append
(
layers
,
manifest
.
Layers
...
)
layers
=
append
(
layers
,
manifest
.
Layers
...
)
layers
=
append
(
layers
,
manifest
.
Config
)
if
manifest
.
Config
.
Digest
!=
""
{
layers
=
append
(
layers
,
&
manifest
.
Config
)
}
skipVerify
:=
make
(
map
[
string
]
bool
)
skipVerify
:=
make
(
map
[
string
]
bool
)
for
_
,
layer
:=
range
layers
{
for
_
,
layer
:=
range
layers
{
...
@@ -971,7 +979,8 @@ func PullModel(ctx context.Context, name string, regOpts *registryOptions, fn fu
...
@@ -971,7 +979,8 @@ func PullModel(ctx context.Context, name string, regOpts *registryOptions, fn fu
fn
(
api
.
ProgressResponse
{
Status
:
"removing any unused layers"
})
fn
(
api
.
ProgressResponse
{
Status
:
"removing any unused layers"
})
err
=
deleteUnusedLayers
(
nil
,
deleteMap
)
err
=
deleteUnusedLayers
(
nil
,
deleteMap
)
if
err
!=
nil
{
if
err
!=
nil
{
return
err
slog
.
Error
(
fmt
.
Sprintf
(
"couldn't remove unused layers: %v"
,
err
))
fn
(
api
.
ProgressResponse
{
Status
:
fmt
.
Sprintf
(
"couldn't remove unused layers: %v"
,
err
)})
}
}
}
}
...
...
server/layer.go
View file @
69eb06c4
...
@@ -2,6 +2,7 @@ package server
...
@@ -2,6 +2,7 @@ package server
import
(
import
(
"crypto/sha256"
"crypto/sha256"
"errors"
"fmt"
"fmt"
"io"
"io"
"os"
"os"
...
@@ -61,6 +62,10 @@ func NewLayer(r io.Reader, mediatype string) (*Layer, error) {
...
@@ -61,6 +62,10 @@ func NewLayer(r io.Reader, mediatype string) (*Layer, error) {
}
}
func
NewLayerFromLayer
(
digest
,
mediatype
,
from
string
)
(
*
Layer
,
error
)
{
func
NewLayerFromLayer
(
digest
,
mediatype
,
from
string
)
(
*
Layer
,
error
)
{
if
digest
==
""
{
return
nil
,
errors
.
New
(
"creating new layer from layer with empty digest"
)
}
blob
,
err
:=
GetBlobsPath
(
digest
)
blob
,
err
:=
GetBlobsPath
(
digest
)
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
...
@@ -81,6 +86,10 @@ func NewLayerFromLayer(digest, mediatype, from string) (*Layer, error) {
...
@@ -81,6 +86,10 @@ func NewLayerFromLayer(digest, mediatype, from string) (*Layer, error) {
}
}
func
(
l
*
Layer
)
Open
()
(
io
.
ReadSeekCloser
,
error
)
{
func
(
l
*
Layer
)
Open
()
(
io
.
ReadSeekCloser
,
error
)
{
if
l
.
Digest
==
""
{
return
nil
,
errors
.
New
(
"opening layer with empty digest"
)
}
blob
,
err
:=
GetBlobsPath
(
l
.
Digest
)
blob
,
err
:=
GetBlobsPath
(
l
.
Digest
)
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
...
@@ -90,13 +99,17 @@ func (l *Layer) Open() (io.ReadSeekCloser, error) {
...
@@ -90,13 +99,17 @@ func (l *Layer) Open() (io.ReadSeekCloser, error) {
}
}
func
(
l
*
Layer
)
Remove
()
error
{
func
(
l
*
Layer
)
Remove
()
error
{
if
l
.
Digest
==
""
{
return
nil
}
ms
,
err
:=
Manifests
()
ms
,
err
:=
Manifests
()
if
err
!=
nil
{
if
err
!=
nil
{
return
err
return
err
}
}
for
_
,
m
:=
range
ms
{
for
_
,
m
:=
range
ms
{
for
_
,
layer
:=
range
append
(
m
.
Layers
,
m
.
Config
)
{
for
_
,
layer
:=
range
append
(
m
.
Layers
,
&
m
.
Config
)
{
if
layer
.
Digest
==
l
.
Digest
{
if
layer
.
Digest
==
l
.
Digest
{
// something is using this layer
// something is using this layer
return
nil
return
nil
...
...
server/manifest.go
View file @
69eb06c4
...
@@ -16,7 +16,7 @@ import (
...
@@ -16,7 +16,7 @@ import (
type
Manifest
struct
{
type
Manifest
struct
{
SchemaVersion
int
`json:"schemaVersion"`
SchemaVersion
int
`json:"schemaVersion"`
MediaType
string
`json:"mediaType"`
MediaType
string
`json:"mediaType"`
Config
*
Layer
`json:"config"`
Config
Layer
`json:"config"`
Layers
[]
*
Layer
`json:"layers"`
Layers
[]
*
Layer
`json:"layers"`
filepath
string
filepath
string
...
@@ -25,7 +25,7 @@ type Manifest struct {
...
@@ -25,7 +25,7 @@ type Manifest struct {
}
}
func
(
m
*
Manifest
)
Size
()
(
size
int64
)
{
func
(
m
*
Manifest
)
Size
()
(
size
int64
)
{
for
_
,
layer
:=
range
append
(
m
.
Layers
,
m
.
Config
)
{
for
_
,
layer
:=
range
append
(
m
.
Layers
,
&
m
.
Config
)
{
size
+=
layer
.
Size
size
+=
layer
.
Size
}
}
...
@@ -46,11 +46,13 @@ func (m *Manifest) Remove() error {
...
@@ -46,11 +46,13 @@ func (m *Manifest) Remove() error {
}
}
func
(
m
*
Manifest
)
RemoveLayers
()
error
{
func
(
m
*
Manifest
)
RemoveLayers
()
error
{
for
_
,
layer
:=
range
append
(
m
.
Layers
,
m
.
Config
)
{
for
_
,
layer
:=
range
append
(
m
.
Layers
,
&
m
.
Config
)
{
if
err
:=
layer
.
Remove
();
errors
.
Is
(
err
,
os
.
ErrNotExist
)
{
if
layer
.
Digest
!=
""
{
slog
.
Debug
(
"layer does not exist"
,
"digest"
,
layer
.
Digest
)
if
err
:=
layer
.
Remove
();
errors
.
Is
(
err
,
os
.
ErrNotExist
)
{
}
else
if
err
!=
nil
{
slog
.
Debug
(
"layer does not exist"
,
"digest"
,
layer
.
Digest
)
return
err
}
else
if
err
!=
nil
{
return
err
}
}
}
}
}
...
@@ -113,7 +115,7 @@ func WriteManifest(name model.Name, config *Layer, layers []*Layer) error {
...
@@ -113,7 +115,7 @@ func WriteManifest(name model.Name, config *Layer, layers []*Layer) error {
m
:=
Manifest
{
m
:=
Manifest
{
SchemaVersion
:
2
,
SchemaVersion
:
2
,
MediaType
:
"application/vnd.docker.distribution.manifest.v2+json"
,
MediaType
:
"application/vnd.docker.distribution.manifest.v2+json"
,
Config
:
config
,
Config
:
*
config
,
Layers
:
layers
,
Layers
:
layers
,
}
}
...
...
server/routes.go
View file @
69eb06c4
...
@@ -824,17 +824,20 @@ func (s *Server) ListModelsHandler(c *gin.Context) {
...
@@ -824,17 +824,20 @@ func (s *Server) ListModelsHandler(c *gin.Context) {
models
:=
[]
api
.
ListModelResponse
{}
models
:=
[]
api
.
ListModelResponse
{}
for
n
,
m
:=
range
ms
{
for
n
,
m
:=
range
ms
{
f
,
err
:=
m
.
Config
.
Open
()
if
err
!=
nil
{
slog
.
Warn
(
"bad manifest filepath"
,
"name"
,
n
,
"error"
,
err
)
continue
}
defer
f
.
Close
()
var
cf
ConfigV2
var
cf
ConfigV2
if
err
:=
json
.
NewDecoder
(
f
)
.
Decode
(
&
cf
);
err
!=
nil
{
slog
.
Warn
(
"bad manifest config"
,
"name"
,
n
,
"error"
,
err
)
if
m
.
Config
.
Digest
!=
""
{
continue
f
,
err
:=
m
.
Config
.
Open
()
if
err
!=
nil
{
slog
.
Warn
(
"bad manifest filepath"
,
"name"
,
n
,
"error"
,
err
)
continue
}
defer
f
.
Close
()
if
err
:=
json
.
NewDecoder
(
f
)
.
Decode
(
&
cf
);
err
!=
nil
{
slog
.
Warn
(
"bad manifest config"
,
"name"
,
n
,
"error"
,
err
)
continue
}
}
}
// tag should never be masked
// tag should never be masked
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
.
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment