-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcache.go
116 lines (100 loc) · 2.71 KB
/
cache.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
111
112
113
114
115
116
package main
import (
"fmt"
"io/ioutil"
"os"
"path/filepath"
"github.com/NickHackman/dots/config"
"golang.org/x/tools/go/vcs"
)
// Cache is a wrapper around the path to the cache directory.
//
// The default case the Dir will be `XDG_CACHE_HOME/dots` or `~/.cache/dots`
type Cache struct {
Dir string
}
// DefaultCache creates a Cache instance using the default value of
// `XDG_CACHE_HOME/dots`
func DefaultCache() (*Cache, error) {
cacheDir, err := os.UserCacheDir()
if err != nil {
return nil, fmt.Errorf("failed to find cache directory: %w", err)
}
dir := filepath.Join(cacheDir, "dots")
if err = os.MkdirAll(dir, os.ModeDir); err != nil {
return nil, fmt.Errorf("failed to mkdir %s: %w", cacheDir, err)
}
return &Cache{dir}, nil
}
// IsHit checks to see if the cache already has a repository downloaded locally
//
// path is expected to be of the form `$domain/$username/$repoName`
// for example: `github.com/NickHackman/dotfiles`
func (cache *Cache) IsHit(path string) bool {
repoPath := filepath.Join(cache.Dir, path)
file, err := os.Stat(repoPath)
if os.IsNotExist(err) || err != nil {
return false
}
if !file.IsDir() {
return false
}
dotsPath, err := config.FindConfig(repoPath)
if err != nil {
return false
}
if _, err = os.Stat(dotsPath); os.IsNotExist(err) || err != nil {
return false
}
return true
}
// Config gets the dots configuration file at a given repository path,
// repository path is expected to be of the form `$domain/$username/$repoName`.
//
// Config cannot on its own determine if the cache is present, prior to calling `Config`
// one should call `IsHit` to verify the existence of the cache.
func (cache *Cache) Config() (*config.DotsConfig, error) {
dotsConfig, err := config.Parse(cache.Dir)
if err != nil {
return nil, err
}
return dotsConfig, nil
}
// Clean completely removes all sub directories of `Cache.Dir`
func (cache *Cache) Clean() error {
return os.RemoveAll(cache.Dir)
}
// UpgradeAll upgrades all repositories it finds inside of Cache
// in a breadth first fashion.
func (cache *Cache) UpgradeAll() error {
queue := []string{cache.Dir}
for len(queue) != 0 {
front := queue[0]
info, err := os.Stat(front)
if err != nil {
return err
}
if info.IsDir() {
files, err := ioutil.ReadDir(front)
if err != nil {
return err
}
for _, file := range files {
path := filepath.Join(front, file.Name())
queue = append(queue, path)
}
}
if err = cache.Upgrade(front); err != nil {
return err
}
}
return nil
}
// Upgrade upgrades a specific repository expected to exist at path
func (cache *Cache) Upgrade(path string) error {
vcs, root, err := vcs.FromDir(path, path)
if err != nil {
return err
}
return vcs.Download(root)
}