mirror of
https://github.com/minio/minio.git
synced 2025-01-27 06:33:18 -05:00
Merge pull request #983 from harshavardhana/improve-update
update: Re-write update command.
This commit is contained in:
commit
f35128eec5
@ -47,14 +47,12 @@ func mainConfigVersion(ctx *cli.Context) {
|
|||||||
|
|
||||||
// convert interface{} back to its original struct
|
// convert interface{} back to its original struct
|
||||||
newConf := config
|
newConf := config
|
||||||
type Version struct {
|
type Version string
|
||||||
Value string `json:"value"`
|
|
||||||
}
|
|
||||||
if globalJSONFlag {
|
if globalJSONFlag {
|
||||||
tB, e := json.Marshal(
|
tB, e := json.Marshal(
|
||||||
struct {
|
struct {
|
||||||
Version Version `json:"version"`
|
Version Version `json:"version"`
|
||||||
}{Version: Version{newConf.Version}},
|
}{Version: Version(newConf.Version)},
|
||||||
)
|
)
|
||||||
fatalIf(probe.NewError(e), "Unable to construct version string.", nil)
|
fatalIf(probe.NewError(e), "Unable to construct version string.", nil)
|
||||||
Println(string(tB))
|
Println(string(tB))
|
||||||
|
@ -81,6 +81,6 @@ func colorizeUpdateMessage(updateString string) (string, *probe.Error) {
|
|||||||
sideBar + line2InColor + spacePaddingLine2 + sideBar + "\n" +
|
sideBar + line2InColor + spacePaddingLine2 + sideBar + "\n" +
|
||||||
bottom + "\n"
|
bottom + "\n"
|
||||||
}
|
}
|
||||||
// finally print the message
|
// return the final message
|
||||||
return message, nil
|
return message, nil
|
||||||
}
|
}
|
||||||
|
200
update-main.go
200
update-main.go
@ -22,7 +22,6 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/fatih/color"
|
"github.com/fatih/color"
|
||||||
@ -30,177 +29,152 @@ import (
|
|||||||
"github.com/minio/minio-xl/pkg/probe"
|
"github.com/minio/minio-xl/pkg/probe"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// command specific flags.
|
||||||
|
var (
|
||||||
|
updateFlags = []cli.Flag{
|
||||||
|
cli.BoolFlag{
|
||||||
|
Name: "help, h",
|
||||||
|
Usage: "Help for update.",
|
||||||
|
},
|
||||||
|
cli.BoolFlag{
|
||||||
|
Name: "experimental, E",
|
||||||
|
Usage: "Check experimental update.",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
// Check for new software updates.
|
// Check for new software updates.
|
||||||
var updateCmd = cli.Command{
|
var updateCmd = cli.Command{
|
||||||
Name: "update",
|
Name: "update",
|
||||||
Usage: "Check for new software updates.",
|
Usage: "Check for a new software update.",
|
||||||
Action: mainUpdate,
|
Action: mainUpdate,
|
||||||
|
Flags: updateFlags,
|
||||||
CustomHelpTemplate: `Name:
|
CustomHelpTemplate: `Name:
|
||||||
minio {{.Name}} - {{.Usage}}
|
minio {{.Name}} - {{.Usage}}
|
||||||
|
|
||||||
USAGE:
|
USAGE:
|
||||||
minio {{.Name}} release
|
minio {{.Name}} [FLAGS]
|
||||||
minio {{.Name}} experimental
|
|
||||||
|
|
||||||
|
FLAGS:
|
||||||
|
{{range .Flags}}{{.}}
|
||||||
|
{{end}}
|
||||||
EXAMPLES:
|
EXAMPLES:
|
||||||
1. Check for new official releases
|
1. Check for any new official release.
|
||||||
$ minio {{.Name}} release
|
$ minio {{.Name}}
|
||||||
|
|
||||||
2. Check for new experimental releases
|
2. Check for any new experimental release.
|
||||||
$ minio {{.Name}} experimental
|
$ minio {{.Name}} --experimental
|
||||||
`,
|
`,
|
||||||
}
|
}
|
||||||
|
|
||||||
// updates container to hold updates json
|
// update URL endpoints.
|
||||||
type updates struct {
|
const (
|
||||||
|
minioUpdateStableURL = "https://dl.minio.io:9000/updates/updates.json"
|
||||||
|
minioUpdateExperimentalURL = "https://dl.minio.io:9000/updates/experimental.json"
|
||||||
|
)
|
||||||
|
|
||||||
|
// minioUpdates container to hold updates json.
|
||||||
|
type minioUpdates struct {
|
||||||
BuildDate string
|
BuildDate string
|
||||||
Platforms map[string]string
|
Platforms map[string]string
|
||||||
}
|
}
|
||||||
|
|
||||||
// updateMessage container to hold update messages
|
// updateMessage container to hold update messages.
|
||||||
type updateMessage struct {
|
type updateMessage struct {
|
||||||
|
Status string `json:"status"`
|
||||||
Update bool `json:"update"`
|
Update bool `json:"update"`
|
||||||
Download string `json:"downloadURL"`
|
Download string `json:"downloadURL"`
|
||||||
Version string `json:"version"`
|
Version string `json:"version"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// String colorized update message
|
// String colorized update message.
|
||||||
func (u updateMessage) String() string {
|
func (u updateMessage) String() string {
|
||||||
if u.Update {
|
if !u.Update {
|
||||||
var msg string
|
updateMessage := color.New(color.FgGreen, color.Bold).SprintfFunc()
|
||||||
if runtime.GOOS == "windows" {
|
return updateMessage("You are already running the most recent version of ‘minio’.")
|
||||||
msg = "Download " + u.Download
|
|
||||||
} else {
|
|
||||||
msg = "Download " + u.Download
|
|
||||||
}
|
|
||||||
msg, err := colorizeUpdateMessage(msg)
|
|
||||||
fatalIf(err.Trace(msg), "Unable to colorize experimental update notification string ‘"+msg+"’.", nil)
|
|
||||||
return msg
|
|
||||||
}
|
}
|
||||||
updateMessage := color.New(color.FgGreen, color.Bold).SprintfFunc()
|
var msg string
|
||||||
return updateMessage("You are already running the most recent version of ‘minio’.")
|
if runtime.GOOS == "windows" {
|
||||||
|
msg = "Download " + u.Download
|
||||||
|
} else {
|
||||||
|
msg = "Download " + u.Download
|
||||||
|
}
|
||||||
|
msg, err := colorizeUpdateMessage(msg)
|
||||||
|
fatalIf(err.Trace(msg), "Unable to colorize experimental update notification string ‘"+msg+"’.", nil)
|
||||||
|
return msg
|
||||||
}
|
}
|
||||||
|
|
||||||
// JSON jsonified update message
|
// JSON jsonified update message.
|
||||||
func (u updateMessage) JSON() string {
|
func (u updateMessage) JSON() string {
|
||||||
|
u.Status = "success"
|
||||||
updateMessageJSONBytes, err := json.Marshal(u)
|
updateMessageJSONBytes, err := json.Marshal(u)
|
||||||
fatalIf(probe.NewError(err), "Unable to marshal into JSON.", nil)
|
fatalIf(probe.NewError(err), "Unable to marshal into JSON.", nil)
|
||||||
|
|
||||||
return string(updateMessageJSONBytes)
|
return string(updateMessageJSONBytes)
|
||||||
}
|
}
|
||||||
|
|
||||||
func getExperimentalUpdate() {
|
// verify updates for releases.
|
||||||
|
func getReleaseUpdate(updateURL string) {
|
||||||
|
data, e := http.Get(updateURL)
|
||||||
|
fatalIf(probe.NewError(e), "Unable to read from update URL ‘"+updateURL+"’.", nil)
|
||||||
|
|
||||||
|
if minioVersion == "UNOFFICIAL.GOGET" {
|
||||||
|
fatalIf(probe.NewError(errors.New("")),
|
||||||
|
"Update mechanism is not supported for ‘go get’ based binary builds. Please download official releases from https://minio.io/#minio", nil)
|
||||||
|
}
|
||||||
|
|
||||||
current, e := time.Parse(time.RFC3339, minioVersion)
|
current, e := time.Parse(time.RFC3339, minioVersion)
|
||||||
fatalIf(probe.NewError(e), "Unable to parse Version string as time.", nil)
|
fatalIf(probe.NewError(e), "Unable to parse version string as time.", nil)
|
||||||
|
|
||||||
if current.IsZero() {
|
if current.IsZero() {
|
||||||
fatalIf(probe.NewError(errors.New("")), "Experimental updates are not supported for custom build. Version field is empty. Please download official releases from https://dl.minio.io:9000", nil)
|
fatalIf(probe.NewError(errors.New("")),
|
||||||
|
"Updates not supported for custom builds. Version field is empty. Please download official releases from https://minio.io/#minio", nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
resp, err := http.Get(minioExperimentalURL)
|
var updates minioUpdates
|
||||||
fatalIf(probe.NewError(err), "Unable to initalize experimental URL.", nil)
|
decoder := json.NewDecoder(data.Body)
|
||||||
|
e = decoder.Decode(&updates)
|
||||||
var experimentals updates
|
|
||||||
decoder := json.NewDecoder(resp.Body)
|
|
||||||
defer resp.Body.Close()
|
|
||||||
e = decoder.Decode(&experimentals)
|
|
||||||
fatalIf(probe.NewError(e), "Unable to decode experimental update notification.", nil)
|
|
||||||
|
|
||||||
latest, e := time.Parse(time.RFC3339, experimentals.BuildDate)
|
|
||||||
fatalIf(probe.NewError(e), "Unable to parse BuildDate.", nil)
|
|
||||||
|
|
||||||
if latest.IsZero() {
|
|
||||||
fatalIf(probe.NewError(errors.New("")), "Unable to validate any experimental update available at this time. Please open an issue at https://github.com/minio/minio/issues", nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
minioExperimentalURLParse, err := url.Parse(minioExperimentalURL)
|
|
||||||
if err != nil {
|
|
||||||
fatalIf(probe.NewError(err), "Unable to parse URL: "+minioExperimentalURL, nil)
|
|
||||||
}
|
|
||||||
downloadURL := minioExperimentalURLParse.Scheme + "://" +
|
|
||||||
minioExperimentalURLParse.Host + "/" + experimentals.Platforms[runtime.GOOS+"-"+runtime.GOARCH]
|
|
||||||
|
|
||||||
updateMessage := updateMessage{
|
|
||||||
Download: downloadURL,
|
|
||||||
Version: minioVersion,
|
|
||||||
}
|
|
||||||
if latest.After(current) {
|
|
||||||
updateMessage.Update = true
|
|
||||||
}
|
|
||||||
if globalJSONFlag {
|
|
||||||
Println(updateMessage.JSON())
|
|
||||||
} else {
|
|
||||||
Println(updateMessage)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func getReleaseUpdate() {
|
|
||||||
current, e := time.Parse(time.RFC3339, minioVersion)
|
|
||||||
fatalIf(probe.NewError(e), "Unable to parse Version string as time.", nil)
|
|
||||||
|
|
||||||
if current.IsZero() {
|
|
||||||
fatalIf(probe.NewError(errors.New("")), "Updates not supported for custom build. Version field is empty. Please download official releases from https://dl.minio.io:9000", nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
resp, err := http.Get(minioUpdateURL)
|
|
||||||
fatalIf(probe.NewError(err), "Unable to initalize experimental URL.", nil)
|
|
||||||
|
|
||||||
var releases updates
|
|
||||||
decoder := json.NewDecoder(resp.Body)
|
|
||||||
e = decoder.Decode(&releases)
|
|
||||||
fatalIf(probe.NewError(e), "Unable to decode update notification.", nil)
|
fatalIf(probe.NewError(e), "Unable to decode update notification.", nil)
|
||||||
|
|
||||||
latest, e := time.Parse(time.RFC3339, releases.BuildDate)
|
latest, e := time.Parse(time.RFC3339, updates.BuildDate)
|
||||||
fatalIf(probe.NewError(e), "Unable to parse BuildDate.", nil)
|
if e != nil {
|
||||||
|
latest, e = time.Parse(http.TimeFormat, updates.BuildDate)
|
||||||
|
fatalIf(probe.NewError(e), "Unable to parse BuildDate.", nil)
|
||||||
|
}
|
||||||
|
|
||||||
if latest.IsZero() {
|
if latest.IsZero() {
|
||||||
fatalIf(probe.NewError(errors.New("")), "Unable to validate any update available at this time. Please open an issue at https://github.com/minio/minio/issues", nil)
|
fatalIf(probe.NewError(errors.New("")),
|
||||||
|
"Unable to validate any update available at this time. Please open an issue at https://github.com/minio/minio/issues", nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
minioUpdateURLParse, err := url.Parse(minioUpdateURL)
|
updateURLParse, err := url.Parse(updateURL)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fatalIf(probe.NewError(err), "Unable to parse URL: "+minioUpdateURL, nil)
|
fatalIf(probe.NewError(err), "Unable to parse URL: "+updateURL, nil)
|
||||||
}
|
}
|
||||||
downloadURL := minioUpdateURLParse.Scheme +
|
downloadURL := updateURLParse.Scheme + "://" +
|
||||||
"://" + minioUpdateURLParse.Host + "/" + releases.Platforms[runtime.GOOS+"-"+runtime.GOARCH]
|
updateURLParse.Host + "/" + updates.Platforms[runtime.GOOS+"-"+runtime.GOARCH]
|
||||||
updateMessage := updateMessage{
|
|
||||||
|
updateMsg := updateMessage{
|
||||||
Download: downloadURL,
|
Download: downloadURL,
|
||||||
Version: minioVersion,
|
Version: minioVersion,
|
||||||
}
|
}
|
||||||
if latest.After(current) {
|
if latest.After(current) {
|
||||||
updateMessage.Update = true
|
updateMsg.Update = true
|
||||||
}
|
}
|
||||||
|
|
||||||
if globalJSONFlag {
|
if globalJSONFlag {
|
||||||
Println(updateMessage.JSON())
|
Println(updateMsg.JSON())
|
||||||
} else {
|
} else {
|
||||||
Println(updateMessage)
|
Println(updateMsg)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const (
|
// main entry point for update command.
|
||||||
minioUpdateURL = "https://dl.minio.io:9000/updates/minio/updates.json"
|
|
||||||
minioExperimentalURL = "https://dl.minio.io:9000/updates/minio/experimental.json"
|
|
||||||
)
|
|
||||||
|
|
||||||
func checkUpdateSyntax(ctx *cli.Context) {
|
|
||||||
if ctx.Args().First() == "help" || !ctx.Args().Present() {
|
|
||||||
cli.ShowCommandHelpAndExit(ctx, "update", 1) // last argument is exit code
|
|
||||||
}
|
|
||||||
arg := strings.TrimSpace(ctx.Args().First())
|
|
||||||
if arg != "release" && arg != "experimental" {
|
|
||||||
fatalIf(probe.NewError(errInvalidArgument), "Unrecognized argument provided.", nil)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// mainUpdate -
|
|
||||||
func mainUpdate(ctx *cli.Context) {
|
func mainUpdate(ctx *cli.Context) {
|
||||||
checkUpdateSyntax(ctx)
|
// Check for update.
|
||||||
arg := strings.TrimSpace(ctx.Args().First())
|
if ctx.Bool("experimental") {
|
||||||
switch arg {
|
getReleaseUpdate(minioUpdateExperimentalURL)
|
||||||
case "release":
|
} else {
|
||||||
getReleaseUpdate()
|
getReleaseUpdate(minioUpdateStableURL)
|
||||||
case "experimental":
|
|
||||||
getExperimentalUpdate()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user