|
|
@ -17,8 +17,9 @@ limitations under the License.
|
|
|
|
package driver // import "k8s.io/helm/pkg/storage/driver"
|
|
|
|
package driver // import "k8s.io/helm/pkg/storage/driver"
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
import (
|
|
|
|
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"fmt"
|
|
|
|
rapi "k8s.io/helm/api"
|
|
|
|
"io"
|
|
|
|
"strconv"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
"time"
|
|
|
@ -29,6 +30,8 @@ import (
|
|
|
|
kblabels "k8s.io/kubernetes/pkg/labels"
|
|
|
|
kblabels "k8s.io/kubernetes/pkg/labels"
|
|
|
|
"k8s.io/kubernetes/pkg/util/validation"
|
|
|
|
"k8s.io/kubernetes/pkg/util/validation"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
"github.com/graymeta/stow"
|
|
|
|
|
|
|
|
rapi "k8s.io/helm/api"
|
|
|
|
"k8s.io/helm/client/clientset"
|
|
|
|
"k8s.io/helm/client/clientset"
|
|
|
|
rspb "k8s.io/helm/pkg/proto/hapi/release"
|
|
|
|
rspb "k8s.io/helm/pkg/proto/hapi/release"
|
|
|
|
"k8s.io/kubernetes/pkg/api/unversioned"
|
|
|
|
"k8s.io/kubernetes/pkg/api/unversioned"
|
|
|
@ -42,7 +45,9 @@ const ReleasesDriverName = "helm.sh/Release"
|
|
|
|
// Releases is a wrapper around an implementation of a kubernetes
|
|
|
|
// Releases is a wrapper around an implementation of a kubernetes
|
|
|
|
// ReleasesInterface.
|
|
|
|
// ReleasesInterface.
|
|
|
|
type Releases struct {
|
|
|
|
type Releases struct {
|
|
|
|
impl clientset.ReleaseInterface
|
|
|
|
impl clientset.ReleaseInterface
|
|
|
|
|
|
|
|
container stow.Container
|
|
|
|
|
|
|
|
prefix string
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// NewReleases initializes a new Releases wrapping an implmenetation of
|
|
|
|
// NewReleases initializes a new Releases wrapping an implmenetation of
|
|
|
@ -51,6 +56,14 @@ func NewReleases(impl clientset.ReleaseInterface) *Releases {
|
|
|
|
return &Releases{impl: impl}
|
|
|
|
return &Releases{impl: impl}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func NewObjectStoreReleases(impl clientset.ReleaseInterface, c stow.Container, prefix string) *Releases {
|
|
|
|
|
|
|
|
p := prefix
|
|
|
|
|
|
|
|
if prefix == "" {
|
|
|
|
|
|
|
|
p = "tiller"
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Releases{impl: impl, container: c, prefix: p}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Name returns the name of the driver.
|
|
|
|
// Name returns the name of the driver.
|
|
|
|
func (releases *Releases) Name() string {
|
|
|
|
func (releases *Releases) Name() string {
|
|
|
|
return ReleasesDriverName
|
|
|
|
return ReleasesDriverName
|
|
|
@ -69,8 +82,13 @@ func (releases *Releases) Get(key string) (*rspb.Release, error) {
|
|
|
|
logerrf(err, "get: failed to get %q", key)
|
|
|
|
logerrf(err, "get: failed to get %q", key)
|
|
|
|
return nil, err
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// found the release, decode the base64 data string
|
|
|
|
// found the release, decode the base64 data string
|
|
|
|
r, err := decodeRelease(obj.Spec.Data.Inline)
|
|
|
|
data, err := releases.getReleaseData(obj)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
r, err := decodeRelease(data)
|
|
|
|
if err != nil {
|
|
|
|
if err != nil {
|
|
|
|
logerrf(err, "get: failed to decode data %q", key)
|
|
|
|
logerrf(err, "get: failed to decode data %q", key)
|
|
|
|
return nil, err
|
|
|
|
return nil, err
|
|
|
@ -97,7 +115,11 @@ func (releases *Releases) List(filter func(*rspb.Release) bool) ([]*rspb.Release
|
|
|
|
// iterate over the releases object list
|
|
|
|
// iterate over the releases object list
|
|
|
|
// and decode each release
|
|
|
|
// and decode each release
|
|
|
|
for _, item := range list.Items {
|
|
|
|
for _, item := range list.Items {
|
|
|
|
rls, err := decodeRelease(item.Spec.Data.Inline)
|
|
|
|
data, err := releases.getReleaseData(&item)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
rls, err := decodeRelease(data)
|
|
|
|
if err != nil {
|
|
|
|
if err != nil {
|
|
|
|
logerrf(err, "list: failed to decode release: %v", item)
|
|
|
|
logerrf(err, "list: failed to decode release: %v", item)
|
|
|
|
continue
|
|
|
|
continue
|
|
|
@ -134,7 +156,11 @@ func (releases *Releases) Query(labels map[string]string) ([]*rspb.Release, erro
|
|
|
|
|
|
|
|
|
|
|
|
var results []*rspb.Release
|
|
|
|
var results []*rspb.Release
|
|
|
|
for _, item := range list.Items {
|
|
|
|
for _, item := range list.Items {
|
|
|
|
rls, err := decodeRelease(item.Spec.Data.Inline)
|
|
|
|
data, err := releases.getReleaseData(&item)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
rls, err := decodeRelease(data)
|
|
|
|
if err != nil {
|
|
|
|
if err != nil {
|
|
|
|
logerrf(err, "query: failed to decode release: %s", err)
|
|
|
|
logerrf(err, "query: failed to decode release: %s", err)
|
|
|
|
continue
|
|
|
|
continue
|
|
|
@ -159,6 +185,12 @@ func (releases *Releases) Create(key string, rls *rspb.Release) error {
|
|
|
|
logerrf(err, "create: failed to encode release %q", rls.Name)
|
|
|
|
logerrf(err, "create: failed to encode release %q", rls.Name)
|
|
|
|
return err
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// push the release object data to object store if configured
|
|
|
|
|
|
|
|
err = releases.writeReleaseData(obj)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
logerrf(err, "create: failed to encode release %q", rls.Name)
|
|
|
|
|
|
|
|
return err
|
|
|
|
|
|
|
|
}
|
|
|
|
// push the release object out into the kubiverse
|
|
|
|
// push the release object out into the kubiverse
|
|
|
|
if _, err := releases.impl.Create(obj); err != nil {
|
|
|
|
if _, err := releases.impl.Create(obj); err != nil {
|
|
|
|
if kberrs.IsAlreadyExists(err) {
|
|
|
|
if kberrs.IsAlreadyExists(err) {
|
|
|
@ -186,6 +218,12 @@ func (releases *Releases) Update(key string, rls *rspb.Release) error {
|
|
|
|
logerrf(err, "update: failed to encode release %q", rls.Name)
|
|
|
|
logerrf(err, "update: failed to encode release %q", rls.Name)
|
|
|
|
return err
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// push the release object data to object store if configured
|
|
|
|
|
|
|
|
err = releases.writeReleaseData(obj)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
logerrf(err, "create: failed to encode release %q", rls.Name)
|
|
|
|
|
|
|
|
return err
|
|
|
|
|
|
|
|
}
|
|
|
|
// push the release object out into the kubiverse
|
|
|
|
// push the release object out into the kubiverse
|
|
|
|
_, err = releases.impl.Update(obj)
|
|
|
|
_, err = releases.impl.Update(obj)
|
|
|
|
if err != nil {
|
|
|
|
if err != nil {
|
|
|
@ -207,12 +245,99 @@ func (releases *Releases) Delete(key string) (rls *rspb.Release, err error) {
|
|
|
|
return nil, err
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// delete the release
|
|
|
|
// delete the release
|
|
|
|
|
|
|
|
if err = releases.deleteReleaseData(rls); err != nil {
|
|
|
|
|
|
|
|
return rls, err
|
|
|
|
|
|
|
|
}
|
|
|
|
if err = releases.impl.Delete(key); err != nil {
|
|
|
|
if err = releases.impl.Delete(key); err != nil {
|
|
|
|
return rls, err
|
|
|
|
return rls, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rls, nil
|
|
|
|
return rls, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func (releases *Releases) itemIDFromTPR(rls *rapi.Release) string {
|
|
|
|
|
|
|
|
return fmt.Sprintf("%v/releases/%v/versions%v", releases.prefix, rls.Name, rls.Spec.Version)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func (releases *Releases) itemIDFromProto(rls *rspb.Release) string {
|
|
|
|
|
|
|
|
return fmt.Sprintf("%v/releases/%v/versions%v", releases.prefix, rls.Name, rls.Version)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func (releases *Releases) deleteReleaseData(rls *rspb.Release) error {
|
|
|
|
|
|
|
|
if releases.container != nil {
|
|
|
|
|
|
|
|
return releases.container.RemoveItem(releases.itemIDFromProto(rls))
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func (releases *Releases) writeReleaseData(rls *rapi.Release) error {
|
|
|
|
|
|
|
|
if releases.container != nil {
|
|
|
|
|
|
|
|
b := bytes.NewBufferString(rls.Spec.Data)
|
|
|
|
|
|
|
|
sz := len(rls.Spec.Data)
|
|
|
|
|
|
|
|
rls.Spec.Data = ""
|
|
|
|
|
|
|
|
_, err := releases.container.Put(releases.itemIDFromTPR(rls), b, int64(sz), nil)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
return err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func (releases *Releases) getReleaseData(rls *rapi.Release) (string, error) {
|
|
|
|
|
|
|
|
if rls.Spec.Data != "" {
|
|
|
|
|
|
|
|
return rls.Spec.Data, nil
|
|
|
|
|
|
|
|
} else if releases.container != nil {
|
|
|
|
|
|
|
|
item, err := releases.container.Item(releases.itemIDFromTPR(rls))
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
f, err := item.Open()
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
// It's a good but not certain bet that FileInfo will tell us exactly how much to
|
|
|
|
|
|
|
|
// read, so let's try it but be prepared for the answer to be wrong.
|
|
|
|
|
|
|
|
var n int64
|
|
|
|
|
|
|
|
// Don't preallocate a huge buffer, just in case.
|
|
|
|
|
|
|
|
if size, err := item.Size(); err != nil && size < 1e9 {
|
|
|
|
|
|
|
|
n = size
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// As initial capacity for readAll, use n + a little extra in case Size is zero,
|
|
|
|
|
|
|
|
// and to avoid another allocation after Read has filled the buffer. The readAll
|
|
|
|
|
|
|
|
// call will read into its allocated internal buffer cheaply. If the size was
|
|
|
|
|
|
|
|
// wrong, we'll either waste some space off the end or reallocate as needed, but
|
|
|
|
|
|
|
|
// in the overwhelmingly common case we'll get it just right.
|
|
|
|
|
|
|
|
b, err := readAll(f, n+bytes.MinRead)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return string(b), nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return "", fmt.Errorf("Missing release data for %v", rls.Name)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// readAll reads from r until an error or EOF and returns the data it read
|
|
|
|
|
|
|
|
// from the internal buffer allocated with a specified capacity.
|
|
|
|
|
|
|
|
func readAll(r io.Reader, capacity int64) (b []byte, err error) {
|
|
|
|
|
|
|
|
buf := bytes.NewBuffer(make([]byte, 0, capacity))
|
|
|
|
|
|
|
|
// If the buffer overflows, we will get bytes.ErrTooLarge.
|
|
|
|
|
|
|
|
// Return that as an error. Any other panic remains.
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
|
|
|
|
e := recover()
|
|
|
|
|
|
|
|
if e == nil {
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if panicErr, ok := e.(error); ok && panicErr == bytes.ErrTooLarge {
|
|
|
|
|
|
|
|
err = panicErr
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
panic(e)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
_, err = buf.ReadFrom(r)
|
|
|
|
|
|
|
|
return buf.Bytes(), err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// newReleasesObject constructs a kubernetes Release object
|
|
|
|
// newReleasesObject constructs a kubernetes Release object
|
|
|
|
// to store a release. Each release data entry is the base64
|
|
|
|
// to store a release. Each release data entry is the base64
|
|
|
|
// encoded string of a release's binary protobuf encoding.
|
|
|
|
// encoded string of a release's binary protobuf encoding.
|
|
|
@ -254,9 +379,7 @@ func newReleasesObject(key string, rls *rspb.Release, lbs labels) (*rapi.Release
|
|
|
|
Spec: rapi.ReleaseSpec{
|
|
|
|
Spec: rapi.ReleaseSpec{
|
|
|
|
Config: rls.Config,
|
|
|
|
Config: rls.Config,
|
|
|
|
Version: rls.Version,
|
|
|
|
Version: rls.Version,
|
|
|
|
Data: rapi.ReleaseData{
|
|
|
|
Data: s,
|
|
|
|
Inline: s,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Status: rapi.ReleaseStatus{
|
|
|
|
Status: rapi.ReleaseStatus{
|
|
|
|
FirstDeployed: toKubeTime(rls.Info.FirstDeployed),
|
|
|
|
FirstDeployed: toKubeTime(rls.Info.FirstDeployed),
|
|
|
|