mirror of
https://github.com/mudler/luet.git
synced 2025-09-04 00:34:41 +00:00
Update gomod and vendor
This commit is contained in:
500
vendor/github.com/moby/buildkit/source/http/httpsource.go
generated
vendored
Normal file
500
vendor/github.com/moby/buildkit/source/http/httpsource.go
generated
vendored
Normal file
@@ -0,0 +1,500 @@
|
||||
package http
|
||||
|
||||
import (
|
||||
"context"
|
||||
"crypto/sha256"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"io"
|
||||
"mime"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"os"
|
||||
"path"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/docker/docker/pkg/idtools"
|
||||
"github.com/docker/docker/pkg/locker"
|
||||
"github.com/moby/buildkit/cache"
|
||||
"github.com/moby/buildkit/cache/metadata"
|
||||
"github.com/moby/buildkit/session"
|
||||
"github.com/moby/buildkit/snapshot"
|
||||
"github.com/moby/buildkit/source"
|
||||
"github.com/moby/buildkit/util/tracing"
|
||||
digest "github.com/opencontainers/go-digest"
|
||||
"github.com/pkg/errors"
|
||||
bolt "go.etcd.io/bbolt"
|
||||
)
|
||||
|
||||
type Opt struct {
|
||||
CacheAccessor cache.Accessor
|
||||
MetadataStore *metadata.Store
|
||||
Transport http.RoundTripper
|
||||
}
|
||||
|
||||
type httpSource struct {
|
||||
md *metadata.Store
|
||||
cache cache.Accessor
|
||||
locker *locker.Locker
|
||||
transport http.RoundTripper
|
||||
}
|
||||
|
||||
func NewSource(opt Opt) (source.Source, error) {
|
||||
transport := opt.Transport
|
||||
if transport == nil {
|
||||
transport = tracing.DefaultTransport
|
||||
}
|
||||
hs := &httpSource{
|
||||
md: opt.MetadataStore,
|
||||
cache: opt.CacheAccessor,
|
||||
locker: locker.New(),
|
||||
transport: transport,
|
||||
}
|
||||
return hs, nil
|
||||
}
|
||||
|
||||
func (hs *httpSource) ID() string {
|
||||
return source.HttpsScheme
|
||||
}
|
||||
|
||||
type httpSourceHandler struct {
|
||||
*httpSource
|
||||
src source.HttpIdentifier
|
||||
refID string
|
||||
cacheKey digest.Digest
|
||||
client *http.Client
|
||||
}
|
||||
|
||||
func (hs *httpSource) Resolve(ctx context.Context, id source.Identifier, sm *session.Manager) (source.SourceInstance, error) {
|
||||
httpIdentifier, ok := id.(*source.HttpIdentifier)
|
||||
if !ok {
|
||||
return nil, errors.Errorf("invalid http identifier %v", id)
|
||||
}
|
||||
|
||||
sessionID := session.FromContext(ctx)
|
||||
|
||||
return &httpSourceHandler{
|
||||
src: *httpIdentifier,
|
||||
httpSource: hs,
|
||||
client: &http.Client{Transport: newTransport(hs.transport, sm, sessionID)},
|
||||
}, nil
|
||||
}
|
||||
|
||||
// urlHash is internal hash the etag is stored by that doesn't leak outside
|
||||
// this package.
|
||||
func (hs *httpSourceHandler) urlHash() (digest.Digest, error) {
|
||||
dt, err := json.Marshal(struct {
|
||||
Filename string
|
||||
Perm, UID, GID int
|
||||
}{
|
||||
Filename: getFileName(hs.src.URL, hs.src.Filename, nil),
|
||||
Perm: hs.src.Perm,
|
||||
UID: hs.src.UID,
|
||||
GID: hs.src.GID,
|
||||
})
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
return digest.FromBytes(dt), nil
|
||||
}
|
||||
|
||||
func (hs *httpSourceHandler) formatCacheKey(filename string, dgst digest.Digest, lastModTime string) digest.Digest {
|
||||
dt, err := json.Marshal(struct {
|
||||
Filename string
|
||||
Perm, UID, GID int
|
||||
Checksum digest.Digest
|
||||
LastModTime string `json:",omitempty"`
|
||||
}{
|
||||
Filename: filename,
|
||||
Perm: hs.src.Perm,
|
||||
UID: hs.src.UID,
|
||||
GID: hs.src.GID,
|
||||
Checksum: dgst,
|
||||
LastModTime: lastModTime,
|
||||
})
|
||||
if err != nil {
|
||||
return dgst
|
||||
}
|
||||
return digest.FromBytes(dt)
|
||||
}
|
||||
|
||||
func (hs *httpSourceHandler) CacheKey(ctx context.Context, index int) (string, bool, error) {
|
||||
if hs.src.Checksum != "" {
|
||||
hs.cacheKey = hs.src.Checksum
|
||||
return hs.formatCacheKey(getFileName(hs.src.URL, hs.src.Filename, nil), hs.src.Checksum, "").String(), true, nil
|
||||
}
|
||||
|
||||
uh, err := hs.urlHash()
|
||||
if err != nil {
|
||||
return "", false, nil
|
||||
}
|
||||
|
||||
// look up metadata(previously stored headers) for that URL
|
||||
sis, err := hs.md.Search(uh.String())
|
||||
if err != nil {
|
||||
return "", false, errors.Wrapf(err, "failed to search metadata for %s", uh)
|
||||
}
|
||||
|
||||
req, err := http.NewRequest("GET", hs.src.URL, nil)
|
||||
if err != nil {
|
||||
return "", false, err
|
||||
}
|
||||
req = req.WithContext(ctx)
|
||||
m := map[string]*metadata.StorageItem{}
|
||||
|
||||
// If we request a single ETag in 'If-None-Match', some servers omit the
|
||||
// unambiguous ETag in their response.
|
||||
// See: https://github.com/moby/buildkit/issues/905
|
||||
var onlyETag string
|
||||
|
||||
if len(sis) > 0 {
|
||||
for _, si := range sis {
|
||||
// if metaDigest := getMetaDigest(si); metaDigest == hs.formatCacheKey("") {
|
||||
if etag := getETag(si); etag != "" {
|
||||
if dgst := getChecksum(si); dgst != "" {
|
||||
m[etag] = si
|
||||
}
|
||||
}
|
||||
// }
|
||||
}
|
||||
if len(m) > 0 {
|
||||
etags := make([]string, 0, len(m))
|
||||
for t := range m {
|
||||
etags = append(etags, t)
|
||||
}
|
||||
req.Header.Add("If-None-Match", strings.Join(etags, ", "))
|
||||
|
||||
if len(etags) == 1 {
|
||||
onlyETag = etags[0]
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Some servers seem to have trouble supporting If-None-Match properly even
|
||||
// though they return ETag-s. So first, optionally try a HEAD request with
|
||||
// manual ETag value comparison.
|
||||
if len(m) > 0 {
|
||||
req.Method = "HEAD"
|
||||
resp, err := hs.client.Do(req)
|
||||
if err == nil {
|
||||
if resp.StatusCode == http.StatusOK || resp.StatusCode == http.StatusNotModified {
|
||||
respETag := resp.Header.Get("ETag")
|
||||
|
||||
// If a 304 is returned without an ETag and we had only sent one ETag,
|
||||
// the response refers to the ETag we asked about.
|
||||
if respETag == "" && onlyETag != "" && resp.StatusCode == http.StatusNotModified {
|
||||
respETag = onlyETag
|
||||
}
|
||||
si, ok := m[respETag]
|
||||
if ok {
|
||||
hs.refID = si.ID()
|
||||
dgst := getChecksum(si)
|
||||
if dgst != "" {
|
||||
modTime := getModTime(si)
|
||||
resp.Body.Close()
|
||||
return hs.formatCacheKey(getFileName(hs.src.URL, hs.src.Filename, resp), dgst, modTime).String(), true, nil
|
||||
}
|
||||
}
|
||||
}
|
||||
resp.Body.Close()
|
||||
}
|
||||
req.Method = "GET"
|
||||
}
|
||||
|
||||
resp, err := hs.client.Do(req)
|
||||
if err != nil {
|
||||
return "", false, err
|
||||
}
|
||||
if resp.StatusCode < 200 || resp.StatusCode >= 400 {
|
||||
return "", false, errors.Errorf("invalid response status %d", resp.StatusCode)
|
||||
}
|
||||
if resp.StatusCode == http.StatusNotModified {
|
||||
respETag := resp.Header.Get("ETag")
|
||||
if respETag == "" && onlyETag != "" {
|
||||
respETag = onlyETag
|
||||
|
||||
// Set the missing ETag header on the response so that it's available
|
||||
// to .save()
|
||||
resp.Header.Set("ETag", onlyETag)
|
||||
}
|
||||
si, ok := m[respETag]
|
||||
if !ok {
|
||||
return "", false, errors.Errorf("invalid not-modified ETag: %v", respETag)
|
||||
}
|
||||
hs.refID = si.ID()
|
||||
dgst := getChecksum(si)
|
||||
if dgst == "" {
|
||||
return "", false, errors.Errorf("invalid metadata change")
|
||||
}
|
||||
modTime := getModTime(si)
|
||||
resp.Body.Close()
|
||||
return hs.formatCacheKey(getFileName(hs.src.URL, hs.src.Filename, resp), dgst, modTime).String(), true, nil
|
||||
}
|
||||
|
||||
ref, dgst, err := hs.save(ctx, resp)
|
||||
if err != nil {
|
||||
return "", false, err
|
||||
}
|
||||
ref.Release(context.TODO())
|
||||
|
||||
hs.cacheKey = dgst
|
||||
|
||||
return hs.formatCacheKey(getFileName(hs.src.URL, hs.src.Filename, resp), dgst, resp.Header.Get("Last-Modified")).String(), true, nil
|
||||
}
|
||||
|
||||
func (hs *httpSourceHandler) save(ctx context.Context, resp *http.Response) (ref cache.ImmutableRef, dgst digest.Digest, retErr error) {
|
||||
newRef, err := hs.cache.New(ctx, nil, cache.CachePolicyRetain, cache.WithDescription(fmt.Sprintf("http url %s", hs.src.URL)))
|
||||
if err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
|
||||
releaseRef := func() {
|
||||
newRef.Release(context.TODO())
|
||||
}
|
||||
|
||||
defer func() {
|
||||
if retErr != nil && newRef != nil {
|
||||
releaseRef()
|
||||
}
|
||||
}()
|
||||
|
||||
mount, err := newRef.Mount(ctx, false)
|
||||
if err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
|
||||
lm := snapshot.LocalMounter(mount)
|
||||
dir, err := lm.Mount()
|
||||
if err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
|
||||
defer func() {
|
||||
if retErr != nil && lm != nil {
|
||||
lm.Unmount()
|
||||
}
|
||||
}()
|
||||
perm := 0600
|
||||
if hs.src.Perm != 0 {
|
||||
perm = hs.src.Perm
|
||||
}
|
||||
fp := filepath.Join(dir, getFileName(hs.src.URL, hs.src.Filename, resp))
|
||||
|
||||
f, err := os.OpenFile(fp, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, os.FileMode(perm))
|
||||
if err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
defer func() {
|
||||
if f != nil {
|
||||
f.Close()
|
||||
}
|
||||
}()
|
||||
|
||||
h := sha256.New()
|
||||
|
||||
if _, err := io.Copy(io.MultiWriter(f, h), resp.Body); err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
|
||||
if err := f.Close(); err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
f = nil
|
||||
|
||||
uid := hs.src.UID
|
||||
gid := hs.src.GID
|
||||
if idmap := mount.IdentityMapping(); idmap != nil {
|
||||
identity, err := idmap.ToHost(idtools.Identity{
|
||||
UID: int(uid),
|
||||
GID: int(gid),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
uid = identity.UID
|
||||
gid = identity.GID
|
||||
}
|
||||
|
||||
if gid != 0 || uid != 0 {
|
||||
if err := os.Chown(fp, uid, gid); err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
}
|
||||
|
||||
mTime := time.Unix(0, 0)
|
||||
lastMod := resp.Header.Get("Last-Modified")
|
||||
if lastMod != "" {
|
||||
if parsedMTime, err := http.ParseTime(lastMod); err == nil {
|
||||
mTime = parsedMTime
|
||||
}
|
||||
}
|
||||
|
||||
if err := os.Chtimes(fp, mTime, mTime); err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
|
||||
lm.Unmount()
|
||||
lm = nil
|
||||
|
||||
ref, err = newRef.Commit(ctx)
|
||||
if err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
newRef = nil
|
||||
|
||||
hs.refID = ref.ID()
|
||||
dgst = digest.NewDigest(digest.SHA256, h)
|
||||
|
||||
if respETag := resp.Header.Get("ETag"); respETag != "" {
|
||||
setETag(ref.Metadata(), respETag)
|
||||
uh, err := hs.urlHash()
|
||||
if err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
setChecksum(ref.Metadata(), uh.String(), dgst)
|
||||
if err := ref.Metadata().Commit(); err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
}
|
||||
|
||||
if modTime := resp.Header.Get("Last-Modified"); modTime != "" {
|
||||
setModTime(ref.Metadata(), modTime)
|
||||
}
|
||||
|
||||
return ref, dgst, nil
|
||||
}
|
||||
|
||||
func (hs *httpSourceHandler) Snapshot(ctx context.Context) (cache.ImmutableRef, error) {
|
||||
if hs.refID != "" {
|
||||
ref, err := hs.cache.Get(ctx, hs.refID)
|
||||
if err == nil {
|
||||
return ref, nil
|
||||
}
|
||||
}
|
||||
|
||||
req, err := http.NewRequest("GET", hs.src.URL, nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
req = req.WithContext(ctx)
|
||||
|
||||
resp, err := hs.client.Do(req)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
ref, dgst, err := hs.save(ctx, resp)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if dgst != hs.cacheKey {
|
||||
ref.Release(context.TODO())
|
||||
return nil, errors.Errorf("digest mismatch %s: %s", dgst, hs.cacheKey)
|
||||
}
|
||||
|
||||
return ref, nil
|
||||
}
|
||||
|
||||
const keyETag = "etag"
|
||||
const keyChecksum = "http.checksum"
|
||||
const keyModTime = "http.modtime"
|
||||
|
||||
func setETag(si *metadata.StorageItem, s string) error {
|
||||
v, err := metadata.NewValue(s)
|
||||
if err != nil {
|
||||
return errors.Wrap(err, "failed to create etag value")
|
||||
}
|
||||
si.Queue(func(b *bolt.Bucket) error {
|
||||
return si.SetValue(b, keyETag, v)
|
||||
})
|
||||
return nil
|
||||
}
|
||||
|
||||
func getETag(si *metadata.StorageItem) string {
|
||||
v := si.Get(keyETag)
|
||||
if v == nil {
|
||||
return ""
|
||||
}
|
||||
var etag string
|
||||
if err := v.Unmarshal(&etag); err != nil {
|
||||
return ""
|
||||
}
|
||||
return etag
|
||||
}
|
||||
|
||||
func setModTime(si *metadata.StorageItem, s string) error {
|
||||
v, err := metadata.NewValue(s)
|
||||
if err != nil {
|
||||
return errors.Wrap(err, "failed to create modtime value")
|
||||
}
|
||||
si.Queue(func(b *bolt.Bucket) error {
|
||||
return si.SetValue(b, keyModTime, v)
|
||||
})
|
||||
return nil
|
||||
}
|
||||
|
||||
func getModTime(si *metadata.StorageItem) string {
|
||||
v := si.Get(keyModTime)
|
||||
if v == nil {
|
||||
return ""
|
||||
}
|
||||
var modTime string
|
||||
if err := v.Unmarshal(&modTime); err != nil {
|
||||
return ""
|
||||
}
|
||||
return modTime
|
||||
}
|
||||
|
||||
func setChecksum(si *metadata.StorageItem, url string, d digest.Digest) error {
|
||||
v, err := metadata.NewValue(d)
|
||||
if err != nil {
|
||||
return errors.Wrap(err, "failed to create checksum value")
|
||||
}
|
||||
v.Index = url
|
||||
si.Queue(func(b *bolt.Bucket) error {
|
||||
return si.SetValue(b, keyChecksum, v)
|
||||
})
|
||||
return nil
|
||||
}
|
||||
|
||||
func getChecksum(si *metadata.StorageItem) digest.Digest {
|
||||
v := si.Get(keyChecksum)
|
||||
if v == nil {
|
||||
return ""
|
||||
}
|
||||
var dgstStr string
|
||||
if err := v.Unmarshal(&dgstStr); err != nil {
|
||||
return ""
|
||||
}
|
||||
dgst, err := digest.Parse(dgstStr)
|
||||
if err != nil {
|
||||
return ""
|
||||
}
|
||||
return dgst
|
||||
}
|
||||
|
||||
func getFileName(urlStr, manualFilename string, resp *http.Response) string {
|
||||
if manualFilename != "" {
|
||||
return manualFilename
|
||||
}
|
||||
if resp != nil {
|
||||
if contentDisposition := resp.Header.Get("Content-Disposition"); contentDisposition != "" {
|
||||
if _, params, err := mime.ParseMediaType(contentDisposition); err == nil {
|
||||
if params["filename"] != "" && !strings.HasSuffix(params["filename"], "/") {
|
||||
if filename := filepath.Base(filepath.FromSlash(params["filename"])); filename != "" {
|
||||
return filename
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
u, err := url.Parse(urlStr)
|
||||
if err == nil {
|
||||
if base := path.Base(u.Path); base != "." && base != "/" {
|
||||
return base
|
||||
}
|
||||
}
|
||||
return "download"
|
||||
}
|
Reference in New Issue
Block a user