forked from lightninglabs/taproot-assets
-
Notifications
You must be signed in to change notification settings - Fork 0
/
version.go
215 lines (181 loc) · 7.05 KB
/
version.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
// Copyright (c) 2013-2017 The btcsuite developers
// Copyright (c) 2015-2016 The Decred developers
// Heavily inspired by https://github.com/btcsuite/btcd/blob/master/version.go
// Copyright (C) 2015-2022 The Lightning Network Developers
package taprootassets
import (
"bytes"
"fmt"
"math"
"runtime/debug"
"strings"
)
var (
// Commit stores the current commit of this build, which includes the
// most recent tag, the number of commits since that tag (if non-zero),
// the commit hash, and a dirty marker. This should be set using the
// -ldflags during compilation.
Commit string
// CommitHash stores the current commit hash of this build.
CommitHash string
// RawTags contains the raw set of build tags, separated by commas.
RawTags string
// GoVersion stores the go version that the executable was compiled
// with.
GoVersion string
)
// versionFieldsAlphabet is the set of characters that are permitted for use in
// a version string field.
const versionFieldsAlphabet = "0123456789abcdefghijklmnopqrstuvwxyz"
// These constants define the application version and follow the semantic
// versioning 2.0.0 spec (http://semver.org/).
const (
// AppMajor defines the major version of this binary.
AppMajor uint = 0
// AppMinor defines the minor version of this binary.
AppMinor uint = 4
// AppPatch defines the application patch for this binary.
AppPatch uint = 0
// AppStatus defines the release status of this binary (e.g. beta).
AppStatus = "alpha"
// AppPreRelease defines the pre-release version of this binary.
// It MUST only contain characters from the semantic versioning spec.
AppPreRelease = ""
// GitTagIncludeStatus indicates whether the status should be included
// in the git tag name.
//
// Including the app version status in the git tag may be problematic
// for golang projects when importing them as dependencies. We therefore
// include this flag to allow toggling the status on and off in a
// standardised way across our projects.
GitTagIncludeStatus = false
// defaultAgentName is the default name of the software that is added as
// the first part of the user agent string.
defaultAgentName = "tapd"
)
// agentName stores the name of the software that is added as the first part of
// the user agent string. This defaults to the value "tapd" when being run as
// a standalone component but can be overwritten by LiT for example when tapd
// is integrated into the UI.
var agentName = defaultAgentName
// SetAgentName overwrites the default agent name which can be used to identify
// the software tapd is bundled in (for example LiT). This function panics if
// the agent name contains characters outside of the allowed semantic alphabet.
func SetAgentName(newAgentName string) {
agentNameAlphabet := versionFieldsAlphabet + "-. "
for _, r := range newAgentName {
if !strings.ContainsRune(agentNameAlphabet, r) {
panic(fmt.Errorf("rune: %v is not in the semantic "+
"alphabet", r))
}
}
agentName = newAgentName
}
// UserAgent returns the full user agent string that identifies the software
// that is submitting swaps to the loop server.
func UserAgent(initiator string) string {
// We'll only allow "safe" characters in the initiator portion of the
// user agent string and spaces only if surrounded by other characters.
initiatorAlphabet := versionFieldsAlphabet + "-. "
cleanInitiator := normalizeVerString(
strings.TrimSpace(initiator), initiatorAlphabet,
)
if len(cleanInitiator) > 0 {
cleanInitiator = fmt.Sprintf(",initiator=%s", cleanInitiator)
}
// The whole user agent string is limited to 255 characters server side
// and also consists of the agent name, version and commit. So we only
// want to take up at most 150 characters for the initiator. Anything
// more will just be dropped.
strLen := len(cleanInitiator)
cleanInitiator = cleanInitiator[:int(math.Min(float64(strLen), 150))]
// Assemble full string, including the commit hash of current build.
return fmt.Sprintf(
"%s/v%s/commit=%s%s", agentName, semanticVersion(), Commit,
cleanInitiator,
)
}
func init() {
// Assert that AppStatus is valid according to the semantic versioning
// guidelines for pre-release version and build metadata strings. In
// particular, it MUST only contain characters in versionFieldsAlphabet.
for _, r := range AppStatus {
if !strings.ContainsRune(versionFieldsAlphabet, r) {
panic(fmt.Errorf("rune: %v is not in the semantic "+
"alphabet", r))
}
}
// Assert that AppPreRelease is valid according to the semantic
// versioning guidelines for pre-release version and build metadata
// strings. In particular, it MUST only contain characters in
// versionFieldsAlphabet.
for _, r := range AppPreRelease {
if !strings.ContainsRune(versionFieldsAlphabet, r) {
panic(fmt.Errorf("rune: %v is not in the semantic "+
"alphabet", r))
}
}
// Get build information from the runtime.
if info, ok := debug.ReadBuildInfo(); ok {
GoVersion = info.GoVersion
for _, setting := range info.Settings {
switch setting.Key {
case "vcs.revision":
CommitHash = setting.Value
case "-tags":
RawTags = setting.Value
}
}
}
}
// Version returns the application version as a properly formed string per the
// semantic versioning 2.0.0 spec (http://semver.org/).
func Version() string {
return fmt.Sprintf("%s commit=%s", semanticVersion(), Commit)
}
// Tags returns the list of build tags that were compiled into the executable.
func Tags() []string {
if len(RawTags) == 0 {
return nil
}
return strings.Split(RawTags, ",")
}
// normalizeVerString returns the passed string stripped of all characters
// which are not valid according to the given alphabet.
func normalizeVerString(str, alphabet string) string {
var result bytes.Buffer
for _, r := range str {
if strings.ContainsRune(alphabet, r) {
result.WriteRune(r)
}
}
return result.String()
}
// semanticVersion returns the SemVer part of the version.
func semanticVersion() string {
// Start with the major, minor, and patch versions.
version := fmt.Sprintf("%d.%d.%d", AppMajor, AppMinor, AppPatch)
// If defined, we will now sanitise the release status string. The
// hyphen called for by the semantic versioning spec is automatically
// appended and should not be contained in the status string. The status
// is not appended if it contains invalid characters.
appStatus := normalizeVerString(AppStatus, versionFieldsAlphabet)
// If defined, we will now sanitise the pre-release version string. The
// hyphen called for by the semantic versioning spec is automatically
// appended and should not be contained in the pre-release string.
// The pre-release version is not appended if it contains invalid
// characters.
preRelease := normalizeVerString(AppPreRelease, versionFieldsAlphabet)
// Append any status and pre-release strings to the version string.
switch {
case appStatus != "" && preRelease != "":
version = fmt.Sprintf(
"%s-%s.%s", version, appStatus, preRelease,
)
case appStatus != "":
version = fmt.Sprintf("%s-%s", version, appStatus)
case preRelease != "":
version = fmt.Sprintf("%s-%s", version, preRelease)
}
return version
}