-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathgc.go
110 lines (99 loc) · 2.54 KB
/
gc.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
package lib
import (
"log"
"os"
"syscall"
dtypes "github.com/docker/engine-api/types"
"github.com/docker/engine-api/types/filters"
"github.com/dustin/go-humanize"
"golang.org/x/net/context"
)
const (
emptyTagString = "<none>:<none>"
)
type GCCleaner interface {
GC()
}
type DockerImageGC struct {
log *log.Logger
dc ImageBuildClient
mc MetricsCollector
dockerPath string
}
func NewDockerImageGC(log *log.Logger, dc ImageBuildClient, mc MetricsCollector, dockerPath string) *DockerImageGC {
return &DockerImageGC{
log: log,
dc: dc,
mc: mc,
dockerPath: dockerPath,
}
}
func (dgc *DockerImageGC) reportDiskMetrics() error {
if _, err := os.Stat(dgc.dockerPath); err != nil {
if os.IsNotExist(err) {
dgc.log.Printf("gc: %v: not found, not calculating free space", dgc.dockerPath)
return nil
}
return err
}
fs := syscall.Statfs_t{}
err := syscall.Statfs(dgc.dockerPath, &fs)
if err != nil {
return err
}
freeBytes := fs.Bfree * uint64(fs.Bsize) // bytes
freeFileNodes := fs.Ffree
dgc.log.Printf("gc: %v free space: %v", dgc.dockerPath, humanize.Bytes(freeBytes))
dgc.log.Printf("gc: %v free file nodes: %v", dgc.dockerPath, freeFileNodes)
dgc.mc.DiskFree(freeBytes)
dgc.mc.FileNodesFree(freeFileNodes)
return nil
}
func (dgc *DockerImageGC) cleanUntaggedImages() error {
ctx := context.Background()
f, err := filters.ParseFlag("dangling=true", filters.NewArgs())
if err != nil {
return err
}
ops := dtypes.ImageListOptions{
All: true,
Filters: f,
}
il, err := dgc.dc.ImageList(ctx, ops)
if err != nil {
return err
}
ropts := dtypes.ImageRemoveOptions{
Force: true,
PruneChildren: true,
}
i := 0
isz := int64(0)
for _, img := range il {
if len(img.RepoTags) == 1 && img.RepoTags[0] == emptyTagString { // redundant, only untagged images should be returned
dgc.log.Printf("gc: removing untagged image: %v (%v)", img.ID, humanize.Bytes(uint64(img.Size)))
_, err = dgc.dc.ImageRemove(ctx, img.ID, ropts)
if err != nil {
return err
}
dgc.mc.GCUntaggedImageRemoved()
i++
isz += img.Size
}
}
dgc.log.Printf("gc: removed %v untagged images (%v total)", i, humanize.Bytes(uint64(isz)))
dgc.mc.GCBytesReclaimed(uint64(isz))
return nil
}
func (dgc *DockerImageGC) GC() {
err := dgc.cleanUntaggedImages()
if err != nil {
dgc.log.Printf("error cleaning untagged images: %v", err)
dgc.mc.GCFailure()
}
err = dgc.reportDiskMetrics()
if err != nil {
dgc.log.Printf("error calculating free disk space: %v", err)
dgc.mc.GCFailure()
}
}