mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-24 23:48:26 +00:00
2111386ee4
I'm working through linter complaints, these are some fixes. Broad categories: 1) Ignore errors where we can ignore errors: add "_ = ..." construct. you can argue that this is annoying noise, but apart from silencing the linter it *does* serve the purpose of highlighting that an error is being ignored. I think this is OK, because the linter highlighted some error cases I wasn't aware of (starting CPU profiles, for example). 2) Untyped constants where we though we had set the type. 3) A real bug where we ineffectually assigned to a shadowed err. 4) Some dead code removed. There'll be more of these, because not all packages are fixed, but the diff was already large enough.
244 lines
5.5 KiB
Go
244 lines
5.5 KiB
Go
// Copyright (C) 2014 The Syncthing Authors.
|
|
//
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
// Package upgrade downloads and compares releases, and upgrades the running binary.
|
|
package upgrade
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"path"
|
|
"runtime"
|
|
"strconv"
|
|
"strings"
|
|
)
|
|
|
|
type Release struct {
|
|
Tag string `json:"tag_name"`
|
|
Prerelease bool `json:"prerelease"`
|
|
Assets []Asset `json:"assets"`
|
|
}
|
|
|
|
type Asset struct {
|
|
URL string `json:"url"`
|
|
Name string `json:"name"`
|
|
}
|
|
|
|
var (
|
|
ErrVersionUpToDate = errors.New("current version is up to date")
|
|
ErrNoReleaseDownload = errors.New("couldn't find a release to download")
|
|
ErrNoVersionToSelect = errors.New("no version to select")
|
|
ErrUpgradeUnsupported = errors.New("upgrade unsupported")
|
|
ErrUpgradeInProgress = errors.New("upgrade already in progress")
|
|
upgradeUnlocked = make(chan bool, 1)
|
|
)
|
|
|
|
func init() {
|
|
upgradeUnlocked <- true
|
|
}
|
|
|
|
func To(rel Release) error {
|
|
select {
|
|
case <-upgradeUnlocked:
|
|
path, err := os.Executable()
|
|
if err != nil {
|
|
upgradeUnlocked <- true
|
|
return err
|
|
}
|
|
err = upgradeTo(path, rel)
|
|
// If we've failed to upgrade, unlock so that another attempt could be made
|
|
if err != nil {
|
|
upgradeUnlocked <- true
|
|
}
|
|
return err
|
|
default:
|
|
return ErrUpgradeInProgress
|
|
}
|
|
}
|
|
|
|
func ToURL(url string) error {
|
|
select {
|
|
case <-upgradeUnlocked:
|
|
binary, err := os.Executable()
|
|
if err != nil {
|
|
upgradeUnlocked <- true
|
|
return err
|
|
}
|
|
err = upgradeToURL(path.Base(url), binary, url)
|
|
// If we've failed to upgrade, unlock so that another attempt could be made
|
|
if err != nil {
|
|
upgradeUnlocked <- true
|
|
}
|
|
return err
|
|
default:
|
|
return ErrUpgradeInProgress
|
|
}
|
|
}
|
|
|
|
type Relation int
|
|
|
|
const (
|
|
MajorOlder Relation = -2 // Older by a major version (x in x.y.z or 0.x.y).
|
|
Older Relation = -1 // Older by a minor version (y or z in x.y.z, or y in 0.x.y)
|
|
Equal Relation = 0 // Versions are semantically equal
|
|
Newer Relation = 1 // Newer by a minor version (y or z in x.y.z, or y in 0.x.y)
|
|
MajorNewer Relation = 2 // Newer by a major version (x in x.y.z or 0.x.y).
|
|
)
|
|
|
|
// CompareVersions returns a relation describing how a compares to b.
|
|
func CompareVersions(a, b string) Relation {
|
|
arel, apre := versionParts(a)
|
|
brel, bpre := versionParts(b)
|
|
|
|
minlen := len(arel)
|
|
if l := len(brel); l < minlen {
|
|
minlen = l
|
|
}
|
|
|
|
// First compare major-minor-patch versions
|
|
for i := 0; i < minlen; i++ {
|
|
if arel[i] < brel[i] {
|
|
if i == 0 {
|
|
// major version difference
|
|
if arel[0] == 0 && brel[0] == 1 {
|
|
// special case, v0.x is equivalent in majorness to v1.x.
|
|
return Older
|
|
}
|
|
return MajorOlder
|
|
}
|
|
// minor or patch version difference
|
|
return Older
|
|
}
|
|
if arel[i] > brel[i] {
|
|
if i == 0 {
|
|
// major version difference
|
|
if arel[0] == 1 && brel[0] == 0 {
|
|
// special case, v0.x is equivalent in majorness to v1.x.
|
|
return Newer
|
|
}
|
|
return MajorNewer
|
|
}
|
|
// minor or patch version difference
|
|
return Newer
|
|
}
|
|
}
|
|
|
|
// Longer version is newer, when the preceding parts are equal
|
|
if len(arel) < len(brel) {
|
|
return Older
|
|
}
|
|
if len(arel) > len(brel) {
|
|
return Newer
|
|
}
|
|
|
|
// Prerelease versions are older, if the versions are the same
|
|
if len(apre) == 0 && len(bpre) > 0 {
|
|
return Newer
|
|
}
|
|
if len(apre) > 0 && len(bpre) == 0 {
|
|
return Older
|
|
}
|
|
|
|
minlen = len(apre)
|
|
if l := len(bpre); l < minlen {
|
|
minlen = l
|
|
}
|
|
|
|
// Compare prerelease strings
|
|
for i := 0; i < minlen; i++ {
|
|
switch av := apre[i].(type) {
|
|
case int:
|
|
switch bv := bpre[i].(type) {
|
|
case int:
|
|
if av < bv {
|
|
return Older
|
|
}
|
|
if av > bv {
|
|
return Newer
|
|
}
|
|
case string:
|
|
return Older
|
|
}
|
|
case string:
|
|
switch bv := bpre[i].(type) {
|
|
case int:
|
|
return Newer
|
|
case string:
|
|
if av < bv {
|
|
return Older
|
|
}
|
|
if av > bv {
|
|
return Newer
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// If all else is equal, longer prerelease string is newer
|
|
if len(apre) < len(bpre) {
|
|
return Older
|
|
}
|
|
if len(apre) > len(bpre) {
|
|
return Newer
|
|
}
|
|
|
|
// Looks like they're actually the same
|
|
return Equal
|
|
}
|
|
|
|
// Split a version into parts.
|
|
// "1.2.3-beta.2" -> []int{1, 2, 3}, []interface{}{"beta", 2}
|
|
func versionParts(v string) ([]int, []interface{}) {
|
|
if strings.HasPrefix(v, "v") || strings.HasPrefix(v, "V") {
|
|
// Strip initial 'v' or 'V' prefix if present.
|
|
v = v[1:]
|
|
}
|
|
parts := strings.SplitN(v, "+", 2)
|
|
parts = strings.SplitN(parts[0], "-", 2)
|
|
fields := strings.Split(parts[0], ".")
|
|
|
|
release := make([]int, len(fields))
|
|
for i, s := range fields {
|
|
v, _ := strconv.Atoi(s)
|
|
release[i] = v
|
|
}
|
|
|
|
var prerelease []interface{}
|
|
if len(parts) > 1 {
|
|
fields = strings.Split(parts[1], ".")
|
|
prerelease = make([]interface{}, len(fields))
|
|
for i, s := range fields {
|
|
v, err := strconv.Atoi(s)
|
|
if err == nil {
|
|
prerelease[i] = v
|
|
} else {
|
|
prerelease[i] = s
|
|
}
|
|
}
|
|
}
|
|
|
|
return release, prerelease
|
|
}
|
|
|
|
func releaseNames(tag string) []string {
|
|
// We must ensure that the release asset matches the expected naming
|
|
// standard, containing both the architecture/OS and the tag name we
|
|
// expect. This protects against malformed release data potentially
|
|
// tricking us into doing a downgrade.
|
|
switch runtime.GOOS {
|
|
case "darwin":
|
|
return []string{
|
|
fmt.Sprintf("syncthing-macos-%s-%s.", runtime.GOARCH, tag),
|
|
fmt.Sprintf("syncthing-macosx-%s-%s.", runtime.GOARCH, tag),
|
|
}
|
|
default:
|
|
return []string{
|
|
fmt.Sprintf("syncthing-%s-%s-%s.", runtime.GOOS, runtime.GOARCH, tag),
|
|
}
|
|
}
|
|
}
|