mirror of https://github.com/helm/helm
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
202 lines
5.6 KiB
202 lines
5.6 KiB
/*
|
|
Copyright The Helm Authors.
|
|
|
|
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
|
|
|
|
http://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 main
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/gosuri/uitable"
|
|
"github.com/spf13/cobra"
|
|
|
|
"helm.sh/helm/v3/cmd/helm/require"
|
|
"helm.sh/helm/v3/internal/completion"
|
|
"helm.sh/helm/v3/pkg/action"
|
|
"helm.sh/helm/v3/pkg/chart"
|
|
"helm.sh/helm/v3/pkg/cli/output"
|
|
"helm.sh/helm/v3/pkg/release"
|
|
"helm.sh/helm/v3/pkg/releaseutil"
|
|
helmtime "helm.sh/helm/v3/pkg/time"
|
|
)
|
|
|
|
var historyHelp = `
|
|
History prints historical revisions for a given release.
|
|
|
|
A default maximum of 256 revisions will be returned. Setting '--max'
|
|
configures the maximum length of the revision list returned.
|
|
|
|
The historical release set is printed as a formatted table, e.g:
|
|
|
|
$ helm history angry-bird
|
|
REVISION UPDATED STATUS CHART APP VERSION DESCRIPTION
|
|
1 Mon Oct 3 10:15:13 2016 superseded alpine-0.1.0 1.0 Initial install
|
|
2 Mon Oct 3 10:15:13 2016 superseded alpine-0.1.0 1.0 Upgraded successfully
|
|
3 Mon Oct 3 10:15:13 2016 superseded alpine-0.1.0 1.0 Rolled back to 2
|
|
4 Mon Oct 3 10:15:13 2016 deployed alpine-0.1.0 1.0 Upgraded successfully
|
|
`
|
|
|
|
func newHistoryCmd(cfg *action.Configuration, out io.Writer) *cobra.Command {
|
|
client := action.NewHistory(cfg)
|
|
var outfmt output.Format
|
|
|
|
cmd := &cobra.Command{
|
|
Use: "history RELEASE_NAME",
|
|
Long: historyHelp,
|
|
Short: "fetch release history",
|
|
Aliases: []string{"hist"},
|
|
Args: require.ExactArgs(1),
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
history, err := getHistory(client, args[0])
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return outfmt.Write(out, history)
|
|
},
|
|
}
|
|
|
|
// Function providing dynamic auto-completion
|
|
completion.RegisterValidArgsFunc(cmd, func(cmd *cobra.Command, args []string, toComplete string) ([]string, completion.BashCompDirective) {
|
|
if len(args) != 0 {
|
|
return nil, completion.BashCompDirectiveNoFileComp
|
|
}
|
|
return compListReleases(toComplete, cfg)
|
|
})
|
|
|
|
f := cmd.Flags()
|
|
f.IntVar(&client.Max, "max", 256, "maximum number of revision to include in history")
|
|
bindOutputFlag(cmd, &outfmt)
|
|
|
|
return cmd
|
|
}
|
|
|
|
type releaseInfo struct {
|
|
Revision int `json:"revision"`
|
|
Updated helmtime.Time `json:"updated"`
|
|
Status string `json:"status"`
|
|
Chart string `json:"chart"`
|
|
AppVersion string `json:"app_version"`
|
|
Description string `json:"description"`
|
|
}
|
|
|
|
type releaseHistory []releaseInfo
|
|
|
|
func (r releaseHistory) WriteJSON(out io.Writer) error {
|
|
return output.EncodeJSON(out, r)
|
|
}
|
|
|
|
func (r releaseHistory) WriteYAML(out io.Writer) error {
|
|
return output.EncodeYAML(out, r)
|
|
}
|
|
|
|
func (r releaseHistory) WriteTable(out io.Writer) error {
|
|
tbl := uitable.New()
|
|
tbl.AddRow("REVISION", "UPDATED", "STATUS", "CHART", "APP VERSION", "DESCRIPTION")
|
|
for _, item := range r {
|
|
tbl.AddRow(item.Revision, item.Updated.Format(time.ANSIC), item.Status, item.Chart, item.AppVersion, item.Description)
|
|
}
|
|
return output.EncodeTable(out, tbl)
|
|
}
|
|
|
|
func getHistory(client *action.History, name string) (releaseHistory, error) {
|
|
hist, err := client.Run(name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
releaseutil.Reverse(hist, releaseutil.SortByRevision)
|
|
|
|
var rels []*release.Release
|
|
for i := 0; i < min(len(hist), client.Max); i++ {
|
|
rels = append(rels, hist[i])
|
|
}
|
|
|
|
if len(rels) == 0 {
|
|
return releaseHistory{}, nil
|
|
}
|
|
|
|
releaseHistory := getReleaseHistory(rels)
|
|
|
|
return releaseHistory, nil
|
|
}
|
|
|
|
func getReleaseHistory(rls []*release.Release) (history releaseHistory) {
|
|
for i := len(rls) - 1; i >= 0; i-- {
|
|
r := rls[i]
|
|
c := formatChartname(r.Chart)
|
|
s := r.Info.Status.String()
|
|
v := r.Version
|
|
d := r.Info.Description
|
|
a := formatAppVersion(r.Chart)
|
|
|
|
rInfo := releaseInfo{
|
|
Revision: v,
|
|
Status: s,
|
|
Chart: c,
|
|
AppVersion: a,
|
|
Description: d,
|
|
}
|
|
if !r.Info.LastDeployed.IsZero() {
|
|
rInfo.Updated = r.Info.LastDeployed
|
|
|
|
}
|
|
history = append(history, rInfo)
|
|
}
|
|
|
|
return history
|
|
}
|
|
|
|
func formatChartname(c *chart.Chart) string {
|
|
if c == nil || c.Metadata == nil {
|
|
// This is an edge case that has happened in prod, though we don't
|
|
// know how: https://github.com/helm/helm/issues/1347
|
|
return "MISSING"
|
|
}
|
|
return fmt.Sprintf("%s-%s", c.Name(), c.Metadata.Version)
|
|
}
|
|
|
|
func formatAppVersion(c *chart.Chart) string {
|
|
if c == nil || c.Metadata == nil {
|
|
// This is an edge case that has happened in prod, though we don't
|
|
// know how: https://github.com/helm/helm/issues/1347
|
|
return "MISSING"
|
|
}
|
|
return c.AppVersion()
|
|
}
|
|
|
|
func min(x, y int) int {
|
|
if x < y {
|
|
return x
|
|
}
|
|
return y
|
|
}
|
|
|
|
func compListRevisions(cfg *action.Configuration, releaseName string) ([]string, completion.BashCompDirective) {
|
|
client := action.NewHistory(cfg)
|
|
|
|
var revisions []string
|
|
if hist, err := client.Run(releaseName); err == nil {
|
|
for _, release := range hist {
|
|
revisions = append(revisions, strconv.Itoa(release.Version))
|
|
}
|
|
return revisions, completion.BashCompDirectiveDefault
|
|
}
|
|
return nil, completion.BashCompDirectiveError
|
|
}
|