summaryrefslogtreecommitdiffstats
path: root/pkg/artifactcache/storage.go
diff options
context:
space:
mode:
Diffstat (limited to 'pkg/artifactcache/storage.go')
-rw-r--r--pkg/artifactcache/storage.go130
1 files changed, 130 insertions, 0 deletions
diff --git a/pkg/artifactcache/storage.go b/pkg/artifactcache/storage.go
new file mode 100644
index 0000000..9a2609a
--- /dev/null
+++ b/pkg/artifactcache/storage.go
@@ -0,0 +1,130 @@
+package artifactcache
+
+import (
+ "fmt"
+ "io"
+ "net/http"
+ "os"
+ "path/filepath"
+)
+
+type Storage struct {
+ rootDir string
+}
+
+func NewStorage(rootDir string) (*Storage, error) {
+ if err := os.MkdirAll(rootDir, 0o755); err != nil {
+ return nil, err
+ }
+ return &Storage{
+ rootDir: rootDir,
+ }, nil
+}
+
+func (s *Storage) Exist(id uint64) (bool, error) {
+ name := s.filename(id)
+ if _, err := os.Stat(name); os.IsNotExist(err) {
+ return false, nil
+ } else if err != nil {
+ return false, err
+ }
+ return true, nil
+}
+
+func (s *Storage) Write(id uint64, offset int64, reader io.Reader) error {
+ name := s.tempName(id, offset)
+ if err := os.MkdirAll(filepath.Dir(name), 0o755); err != nil {
+ return err
+ }
+ file, err := os.Create(name)
+ if err != nil {
+ return err
+ }
+ defer file.Close()
+
+ _, err = io.Copy(file, reader)
+ return err
+}
+
+func (s *Storage) Commit(id uint64, size int64) (int64, error) {
+ defer func() {
+ _ = os.RemoveAll(s.tempDir(id))
+ }()
+
+ name := s.filename(id)
+ tempNames, err := s.tempNames(id)
+ if err != nil {
+ return 0, err
+ }
+
+ if err := os.MkdirAll(filepath.Dir(name), 0o755); err != nil {
+ return 0, err
+ }
+ file, err := os.Create(name)
+ if err != nil {
+ return 0, err
+ }
+ defer file.Close()
+
+ var written int64
+ for _, v := range tempNames {
+ f, err := os.Open(v)
+ if err != nil {
+ return 0, err
+ }
+ n, err := io.Copy(file, f)
+ _ = f.Close()
+ if err != nil {
+ return 0, err
+ }
+ written += n
+ }
+
+ // If size is less than 0, it means the size is unknown.
+ // We can't check the size of the file, just skip the check.
+ // It happens when the request comes from old versions of actions, like `actions/cache@v2`.
+ if size >= 0 && written != size {
+ _ = file.Close()
+ _ = os.Remove(name)
+ return 0, fmt.Errorf("broken file: %v != %v", written, size)
+ }
+
+ return written, nil
+}
+
+func (s *Storage) Serve(w http.ResponseWriter, r *http.Request, id uint64) {
+ name := s.filename(id)
+ http.ServeFile(w, r, name)
+}
+
+func (s *Storage) Remove(id uint64) {
+ _ = os.Remove(s.filename(id))
+ _ = os.RemoveAll(s.tempDir(id))
+}
+
+func (s *Storage) filename(id uint64) string {
+ return filepath.Join(s.rootDir, fmt.Sprintf("%02x", id%0xff), fmt.Sprint(id))
+}
+
+func (s *Storage) tempDir(id uint64) string {
+ return filepath.Join(s.rootDir, "tmp", fmt.Sprint(id))
+}
+
+func (s *Storage) tempName(id uint64, offset int64) string {
+ return filepath.Join(s.tempDir(id), fmt.Sprintf("%016x", offset))
+}
+
+func (s *Storage) tempNames(id uint64) ([]string, error) {
+ dir := s.tempDir(id)
+ files, err := os.ReadDir(dir)
+ if err != nil {
+ return nil, err
+ }
+ var names []string
+ for _, v := range files {
+ if !v.IsDir() {
+ names = append(names, filepath.Join(dir, v.Name()))
+ }
+ }
+ return names, nil
+}