2019-08-14 21:02:52 +02:00
|
|
|
// Copyright 2019 Google LLC
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
|
|
|
|
// use this file except in compliance with the License. You may obtain a copy of
|
|
|
|
// the License at
|
|
|
|
//
|
|
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
// License for the specific language governing permissions and limitations under
|
|
|
|
// the License.
|
|
|
|
package builder
|
|
|
|
|
|
|
|
import (
|
2019-09-30 15:25:55 +02:00
|
|
|
"bytes"
|
2019-09-09 17:41:52 +02:00
|
|
|
"context"
|
2019-09-30 15:25:55 +02:00
|
|
|
"encoding/json"
|
2019-09-09 17:41:52 +02:00
|
|
|
"io"
|
|
|
|
"log"
|
|
|
|
"os"
|
2019-08-14 21:02:52 +02:00
|
|
|
"sync"
|
2019-09-09 17:41:52 +02:00
|
|
|
|
|
|
|
"cloud.google.com/go/storage"
|
2019-08-14 21:02:52 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type void struct{}
|
|
|
|
|
2019-09-30 15:25:55 +02:00
|
|
|
type Build struct {
|
|
|
|
SHA256 string `json:"sha256"`
|
|
|
|
MD5 string `json:"md5"`
|
|
|
|
}
|
|
|
|
|
2019-09-09 17:41:52 +02:00
|
|
|
// LocalCache implements the structure used for local caching of
|
|
|
|
// manifests and layer uploads.
|
|
|
|
type LocalCache struct {
|
2019-09-30 15:25:55 +02:00
|
|
|
// Manifest cache
|
2019-08-14 21:02:52 +02:00
|
|
|
mmtx sync.RWMutex
|
2019-09-08 23:21:14 +02:00
|
|
|
mcache map[string]string
|
2019-08-14 21:02:52 +02:00
|
|
|
|
2019-09-30 15:25:55 +02:00
|
|
|
// Layer (tarball) cache
|
2019-08-14 21:02:52 +02:00
|
|
|
lmtx sync.RWMutex
|
|
|
|
lcache map[string]void
|
2019-09-30 15:25:55 +02:00
|
|
|
|
|
|
|
// Layer (build) cache
|
|
|
|
bmtx sync.RWMutex
|
|
|
|
bcache map[string]Build
|
2019-08-14 21:02:52 +02:00
|
|
|
}
|
|
|
|
|
2019-09-09 17:41:52 +02:00
|
|
|
func NewCache() LocalCache {
|
|
|
|
return LocalCache{
|
2019-09-08 23:21:14 +02:00
|
|
|
mcache: make(map[string]string),
|
2019-08-14 21:02:52 +02:00
|
|
|
lcache: make(map[string]void),
|
2019-10-02 00:26:09 +02:00
|
|
|
bcache: make(map[string]Build),
|
2019-08-14 21:02:52 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Has this layer hash already been seen by this Nixery instance? If
|
|
|
|
// yes, we can skip upload checking and such because it has already
|
|
|
|
// been done.
|
2019-09-09 17:41:52 +02:00
|
|
|
func (c *LocalCache) hasSeenLayer(hash string) bool {
|
2019-08-14 21:02:52 +02:00
|
|
|
c.lmtx.RLock()
|
|
|
|
defer c.lmtx.RUnlock()
|
|
|
|
_, seen := c.lcache[hash]
|
|
|
|
return seen
|
|
|
|
}
|
|
|
|
|
|
|
|
// Layer has now been seen and should be stored.
|
2019-09-09 17:41:52 +02:00
|
|
|
func (c *LocalCache) sawLayer(hash string) {
|
2019-08-14 21:02:52 +02:00
|
|
|
c.lmtx.Lock()
|
|
|
|
defer c.lmtx.Unlock()
|
|
|
|
c.lcache[hash] = void{}
|
|
|
|
}
|
|
|
|
|
2019-09-08 23:21:14 +02:00
|
|
|
// Retrieve a cached manifest if the build is cacheable and it exists.
|
2019-09-09 17:41:52 +02:00
|
|
|
func (c *LocalCache) manifestFromLocalCache(key string) (string, bool) {
|
2019-09-08 23:21:14 +02:00
|
|
|
c.mmtx.RLock()
|
|
|
|
path, ok := c.mcache[key]
|
2019-08-14 21:02:52 +02:00
|
|
|
c.mmtx.RUnlock()
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
|
2019-09-08 23:21:14 +02:00
|
|
|
return path, true
|
2019-08-14 21:02:52 +02:00
|
|
|
}
|
|
|
|
|
2019-09-09 17:41:52 +02:00
|
|
|
// Adds the result of a manifest build to the local cache, if the
|
|
|
|
// manifest is considered cacheable.
|
|
|
|
func (c *LocalCache) localCacheManifest(key, path string) {
|
|
|
|
c.mmtx.Lock()
|
|
|
|
c.mcache[key] = path
|
|
|
|
c.mmtx.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-09-30 15:25:55 +02:00
|
|
|
// Retrieve a cached build from the local cache.
|
|
|
|
func (c *LocalCache) buildFromLocalCache(key string) (*Build, bool) {
|
|
|
|
c.bmtx.RLock()
|
|
|
|
b, ok := c.bcache[key]
|
|
|
|
c.bmtx.RUnlock()
|
|
|
|
|
|
|
|
return &b, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add a build result to the local cache.
|
|
|
|
func (c *LocalCache) localCacheBuild(key string, b Build) {
|
|
|
|
c.bmtx.Lock()
|
|
|
|
c.bcache[key] = b
|
|
|
|
c.bmtx.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-09-09 17:41:52 +02:00
|
|
|
// Retrieve a manifest from the cache(s). First the local cache is
|
|
|
|
// checked, then the GCS-bucket cache.
|
2019-10-02 00:26:09 +02:00
|
|
|
func manifestFromCache(ctx context.Context, cache *LocalCache, bucket *storage.BucketHandle, key string) (string, bool) {
|
2019-09-09 17:41:52 +02:00
|
|
|
path, cached := cache.manifestFromLocalCache(key)
|
|
|
|
if cached {
|
|
|
|
return path, true
|
|
|
|
}
|
|
|
|
|
|
|
|
obj := bucket.Object("manifests/" + key)
|
|
|
|
|
|
|
|
// Probe whether the file exists before trying to fetch it.
|
2019-10-02 00:26:09 +02:00
|
|
|
_, err := obj.Attrs(ctx)
|
2019-09-09 17:41:52 +02:00
|
|
|
if err != nil {
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
|
2019-10-02 00:26:09 +02:00
|
|
|
r, err := obj.NewReader(ctx)
|
2019-09-09 17:41:52 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed to retrieve manifest '%s' from cache: %s\n", key, err)
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
defer r.Close()
|
|
|
|
|
|
|
|
path = os.TempDir() + "/" + key
|
|
|
|
f, _ := os.Create(path)
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
_, err = io.Copy(f, r)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed to read cached manifest for '%s': %s\n", key, err)
|
|
|
|
}
|
|
|
|
|
2019-09-10 12:13:10 +02:00
|
|
|
log.Printf("Retrieved manifest for sha1:%s from GCS\n", key)
|
2019-09-09 17:41:52 +02:00
|
|
|
cache.localCacheManifest(key, path)
|
|
|
|
|
|
|
|
return path, true
|
|
|
|
}
|
|
|
|
|
2019-09-30 15:25:55 +02:00
|
|
|
// Add a manifest to the bucket & local caches
|
2019-09-10 12:13:10 +02:00
|
|
|
func cacheManifest(ctx *context.Context, cache *LocalCache, bucket *storage.BucketHandle, key, path string) {
|
2019-09-09 17:41:52 +02:00
|
|
|
cache.localCacheManifest(key, path)
|
|
|
|
|
|
|
|
obj := bucket.Object("manifests/" + key)
|
|
|
|
w := obj.NewWriter(*ctx)
|
|
|
|
|
|
|
|
f, err := os.Open(path)
|
|
|
|
if err != nil {
|
2019-09-10 12:13:10 +02:00
|
|
|
log.Printf("failed to open manifest sha1:%s for cache upload: %s\n", key, err)
|
2019-09-09 17:41:52 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
size, err := io.Copy(w, f)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("failed to cache manifest sha1:%s: %s\n", key, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = w.Close(); err != nil {
|
|
|
|
log.Printf("failed to cache manifest sha1:%s: %s\n", key, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("Cached manifest sha1:%s (%v bytes written)\n", key, size)
|
2019-08-14 21:02:52 +02:00
|
|
|
}
|
2019-09-30 15:25:55 +02:00
|
|
|
|
|
|
|
// Retrieve a build from the cache, first checking the local cache
|
|
|
|
// followed by the bucket cache.
|
|
|
|
func buildFromCache(ctx *context.Context, cache *LocalCache, bucket *storage.BucketHandle, key string) (*Build, bool) {
|
|
|
|
build, cached := cache.buildFromLocalCache(key)
|
|
|
|
if cached {
|
|
|
|
return build, true
|
|
|
|
}
|
|
|
|
|
|
|
|
obj := bucket.Object("builds/" + key)
|
|
|
|
_, err := obj.Attrs(*ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
r, err := obj.NewReader(*ctx)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed to retrieve build '%s' from cache: %s\n", key, err)
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
defer r.Close()
|
|
|
|
|
|
|
|
jb := bytes.NewBuffer([]byte{})
|
|
|
|
_, err = io.Copy(jb, r)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed to read build '%s' from cache: %s\n", key, err)
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
var b Build
|
|
|
|
err = json.Unmarshal(jb.Bytes(), &build)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed to unmarshal build '%s' from cache: %s\n", key, err)
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
return &b, true
|
|
|
|
}
|
|
|
|
|
|
|
|
func cacheBuild(ctx context.Context, cache *LocalCache, bucket *storage.BucketHandle, key string, build Build) {
|
|
|
|
cache.localCacheBuild(key, build)
|
|
|
|
|
|
|
|
obj := bucket.Object("builds/" + key)
|
|
|
|
|
|
|
|
j, _ := json.Marshal(&build)
|
|
|
|
|
|
|
|
w := obj.NewWriter(ctx)
|
|
|
|
|
2019-10-02 00:26:09 +02:00
|
|
|
_, err := io.Copy(w, bytes.NewReader(j))
|
2019-09-30 15:25:55 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("failed to cache build '%s': %s\n", key, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|