upgrade_common.go 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. // Copyright (C) 2014 The Syncthing Authors.
  2. //
  3. // This program is free software: you can redistribute it and/or modify it
  4. // under the terms of the GNU General Public License as published by the Free
  5. // Software Foundation, either version 3 of the License, or (at your option)
  6. // any later version.
  7. //
  8. // This program is distributed in the hope that it will be useful, but WITHOUT
  9. // ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  10. // FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  11. // more details.
  12. //
  13. // You should have received a copy of the GNU General Public License along
  14. // with this program. If not, see <http://www.gnu.org/licenses/>.
  15. // Package upgrade downloads and compares releases, and upgrades the running binary.
  16. package upgrade
  17. import (
  18. "errors"
  19. "strconv"
  20. "strings"
  21. "github.com/calmh/osext"
  22. )
  23. type Release struct {
  24. Tag string `json:"tag_name"`
  25. Prerelease bool `json:"prerelease"`
  26. Assets []Asset `json:"assets"`
  27. }
  28. type Asset struct {
  29. URL string `json:"url"`
  30. Name string `json:"name"`
  31. }
  32. var (
  33. ErrVersionUpToDate = errors.New("current version is up to date")
  34. ErrVersionUnknown = errors.New("couldn't fetch release information")
  35. ErrUpgradeUnsupported = errors.New("upgrade unsupported")
  36. ErrUpgradeInProgress = errors.New("upgrade already in progress")
  37. upgradeUnlocked = make(chan bool, 1)
  38. )
  39. func init() {
  40. upgradeUnlocked <- true
  41. }
  42. // A wrapper around actual implementations
  43. func To(rel Release) error {
  44. select {
  45. case <-upgradeUnlocked:
  46. path, err := osext.Executable()
  47. if err != nil {
  48. upgradeUnlocked <- true
  49. return err
  50. }
  51. err = upgradeTo(path, rel)
  52. // If we've failed to upgrade, unlock so that another attempt could be made
  53. if err != nil {
  54. upgradeUnlocked <- true
  55. }
  56. return err
  57. default:
  58. return ErrUpgradeInProgress
  59. }
  60. }
  61. // A wrapper around actual implementations
  62. func ToURL(url string) error {
  63. select {
  64. case <-upgradeUnlocked:
  65. path, err := osext.Executable()
  66. if err != nil {
  67. upgradeUnlocked <- true
  68. return err
  69. }
  70. err = upgradeToURL(path, url)
  71. // If we've failed to upgrade, unlock so that another attempt could be made
  72. if err != nil {
  73. upgradeUnlocked <- true
  74. }
  75. return err
  76. default:
  77. return ErrUpgradeInProgress
  78. }
  79. }
  80. // Returns 1 if a>b, -1 if a<b and 0 if they are equal
  81. func CompareVersions(a, b string) int {
  82. arel, apre := versionParts(a)
  83. brel, bpre := versionParts(b)
  84. minlen := len(arel)
  85. if l := len(brel); l < minlen {
  86. minlen = l
  87. }
  88. // First compare major-minor-patch versions
  89. for i := 0; i < minlen; i++ {
  90. if arel[i] < brel[i] {
  91. return -1
  92. }
  93. if arel[i] > brel[i] {
  94. return 1
  95. }
  96. }
  97. // Longer version is newer, when the preceding parts are equal
  98. if len(arel) < len(brel) {
  99. return -1
  100. }
  101. if len(arel) > len(brel) {
  102. return 1
  103. }
  104. // Prerelease versions are older, if the versions are the same
  105. if len(apre) == 0 && len(bpre) > 0 {
  106. return 1
  107. }
  108. if len(apre) > 0 && len(bpre) == 0 {
  109. return -1
  110. }
  111. minlen = len(apre)
  112. if l := len(bpre); l < minlen {
  113. minlen = l
  114. }
  115. // Compare prerelease strings
  116. for i := 0; i < minlen; i++ {
  117. switch av := apre[i].(type) {
  118. case int:
  119. switch bv := bpre[i].(type) {
  120. case int:
  121. if av < bv {
  122. return -1
  123. }
  124. if av > bv {
  125. return 1
  126. }
  127. case string:
  128. return -1
  129. }
  130. case string:
  131. switch bv := bpre[i].(type) {
  132. case int:
  133. return 1
  134. case string:
  135. if av < bv {
  136. return -1
  137. }
  138. if av > bv {
  139. return 1
  140. }
  141. }
  142. }
  143. }
  144. // If all else is equal, longer prerelease string is newer
  145. if len(apre) < len(bpre) {
  146. return -1
  147. }
  148. if len(apre) > len(bpre) {
  149. return 1
  150. }
  151. // Looks like they're actually the same
  152. return 0
  153. }
  154. // Split a version into parts.
  155. // "1.2.3-beta.2" -> []int{1, 2, 3}, []interface{}{"beta", 2}
  156. func versionParts(v string) ([]int, []interface{}) {
  157. if strings.HasPrefix(v, "v") || strings.HasPrefix(v, "V") {
  158. // Strip initial 'v' or 'V' prefix if present.
  159. v = v[1:]
  160. }
  161. parts := strings.SplitN(v, "+", 2)
  162. parts = strings.SplitN(parts[0], "-", 2)
  163. fields := strings.Split(parts[0], ".")
  164. release := make([]int, len(fields))
  165. for i, s := range fields {
  166. v, _ := strconv.Atoi(s)
  167. release[i] = v
  168. }
  169. var prerelease []interface{}
  170. if len(parts) > 1 {
  171. fields = strings.Split(parts[1], ".")
  172. prerelease = make([]interface{}, len(fields))
  173. for i, s := range fields {
  174. v, err := strconv.Atoi(s)
  175. if err == nil {
  176. prerelease[i] = v
  177. } else {
  178. prerelease[i] = s
  179. }
  180. }
  181. }
  182. return release, prerelease
  183. }