forked from francoismichel/ssh3
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathversion.go
302 lines (263 loc) · 10.4 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
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
package ssh3
import (
"fmt"
"strconv"
"strings"
"github.com/rs/zerolog/log"
)
// EXPERIMENTAL_SPEC_VERSION specifies which version of the protocol this software
// is implementing.
// The protocol version string format is:
//
// major + "." + minor[ + "_" + additional-version-information].
//
// It currently implements a first early version with no specification (alpha).
// Once IETF drafts get published, we plan on having versions such as
// 3.0_draft-michel-ssh3-XX when implementing the IETF specification from
// draft-michel-ssh3-XX.
const PROTOCOL_MAJOR int = 3
const PROTOCOL_MINOR int = 0
const PROTOCOL_EXPERIMENTAL_SPEC_VERSION string = "alpha-00"
const SOFTWARE_IMPLEMENTATION_NAME string = "francoismichel/ssh3"
const SOFTWARE_MAJOR int = 0
const SOFTWARE_MINOR int = 1
const SOFTWARE_PATCH int = 7
const SOFTWARE_RC int = 0
var AVAILABLE_CLIENT_VERSIONS []Version = []Version{
ThisVersion(),
NewVersion("SSH", NewProtocolVersion(3, 0, ""), NewSoftwareVersion(0, 1, 5, SOFTWARE_IMPLEMENTATION_NAME)),
NewVersion("SSH", NewProtocolVersion(3, 0, ""), NewSoftwareVersion(0, 1, 4, SOFTWARE_IMPLEMENTATION_NAME)),
NewVersion("SSH", NewProtocolVersion(3, 0, ""), NewSoftwareVersion(0, 1, 3, SOFTWARE_IMPLEMENTATION_NAME)),
}
func ThisVersion() Version {
return Version{
protocolName: "SSH",
protocolVersion: ProtocolVersion{
Major: PROTOCOL_MAJOR,
Minor: PROTOCOL_MINOR,
ExperimentalSpecVersion: PROTOCOL_EXPERIMENTAL_SPEC_VERSION,
},
softwareVersion: SoftwareVersion{
ImplementationName: SOFTWARE_IMPLEMENTATION_NAME,
Major: SOFTWARE_MAJOR,
Minor: SOFTWARE_MINOR,
Patch: SOFTWARE_PATCH,
},
}
}
// Tells if the this version (a.k.a. the version returned by ThisVersion())
// is compatible with `other`.
func IsVersionSupported(other Version) bool {
this := ThisVersion()
// right now, no check for protocol name as it is subject to change
// strict protocol version checking
if other.protocolVersion.Major != this.protocolVersion.Major || other.protocolVersion.Minor != this.protocolVersion.Minor {
return false
}
// special case: to our knowledge, experimental spec version older than alpha-00 are only implemented by us (i.e. francoismichel/ssh3)
// this should be removed in the near future, once we remove support for these legacy servers
if other.protocolVersion.ExperimentalSpecVersion == "" && other.softwareVersion.ImplementationName == "francoismichel/ssh3" &&
other.softwareVersion.Major == 0 && other.softwareVersion.Minor == 1 && other.softwareVersion.Patch <= 5 {
// then, only support software version >= 0.1.3
return other.softwareVersion.Patch >= 3
}
// Starting from here, we have proper experimental spec version signalling.
// this version is version alpha-00, other versions are not supported.
// If a server receives a request with an unsupported spec version, the client should
// start a new request with a compatible version.
return other.protocolVersion.ExperimentalSpecVersion == "alpha-00"
}
type SoftwareVersion struct {
ImplementationName string
Major int
Minor int
Patch int
}
func NewSoftwareVersion(major int, minor int, patch int, implementationName string) SoftwareVersion {
return SoftwareVersion{
Major: major,
Minor: minor,
Patch: patch,
ImplementationName: implementationName,
}
}
type InvalidSoftwareVersion struct {
softwareVersionString string
}
func (e InvalidSoftwareVersion) Error() string {
return fmt.Sprintf("invalid protocol version string: %s", e.softwareVersionString)
}
func ParseSoftwareVersion(implementationName string, versionString string) (SoftwareVersion, error) {
majorDotMinor := strings.Split(versionString, ".")
if len(majorDotMinor) != 3 {
log.Error().Msgf("bad SSH version major.minor.patch field")
return SoftwareVersion{}, InvalidSoftwareVersion{softwareVersionString: versionString}
}
major, err := strconv.Atoi(majorDotMinor[0])
if err != nil {
log.Error().Msgf("bad software version major value")
return SoftwareVersion{}, InvalidSoftwareVersion{softwareVersionString: versionString}
}
minor, err := strconv.Atoi(majorDotMinor[1])
if err != nil {
log.Error().Msgf("bad software version minor value")
return SoftwareVersion{}, InvalidSoftwareVersion{softwareVersionString: versionString}
}
patch, err := strconv.Atoi(majorDotMinor[2])
if err != nil {
log.Error().Msgf("bad software version patch value")
return SoftwareVersion{}, InvalidSoftwareVersion{softwareVersionString: versionString}
}
return SoftwareVersion{
ImplementationName: implementationName,
Major: major,
Minor: minor,
Patch: patch,
}, nil
}
func (v SoftwareVersion) String() string {
return fmt.Sprintf("%s %d.%d.%d", v.ImplementationName, v.Major, v.Minor, v.Patch)
}
type ProtocolVersion struct {
Major int
Minor int
ExperimentalSpecVersion string
}
func NewProtocolVersion(major int, minor int, experimentalspecversion string) ProtocolVersion {
return ProtocolVersion{
Major: major,
Minor: minor,
ExperimentalSpecVersion: experimentalspecversion,
}
}
func (v ProtocolVersion) String() string {
ret := fmt.Sprintf("%d.%d", v.Major, v.Minor)
if v.ExperimentalSpecVersion != "" {
ret = fmt.Sprintf("%s_%s", ret, v.ExperimentalSpecVersion)
}
return ret
}
type InvalidProtocolVersion struct {
protocolVersionString string
}
func (e InvalidProtocolVersion) Error() string {
return fmt.Sprintf("invalid protocol version string: %s", e.protocolVersionString)
}
func ParseProtocolVersion(versionString string) (ProtocolVersion, error) {
fields := strings.Split(versionString, "_")
if len(fields) == 0 || len(fields) > 2 {
return ProtocolVersion{}, InvalidProtocolVersion{protocolVersionString: versionString}
}
majorDotMinor := strings.Split(fields[0], ".")
if len(majorDotMinor) != 2 {
log.Error().Msgf("protocol version should be in format x.y, got: %s", fields[0])
return ProtocolVersion{}, InvalidProtocolVersion{protocolVersionString: versionString}
}
major, err := strconv.Atoi(majorDotMinor[0])
if err != nil {
log.Error().Msgf("bad protocol version major value %s: %s", majorDotMinor[0], err)
return ProtocolVersion{}, InvalidSSHVersion{versionString: versionString}
}
minor, err := strconv.Atoi(majorDotMinor[1])
if err != nil {
log.Error().Msgf("bad protocol version minor value %s: %s", majorDotMinor[1], err)
return ProtocolVersion{}, InvalidSSHVersion{versionString: versionString}
}
experimentalSpecVersion := ""
if len(fields) == 2 {
experimentalSpecVersion = fields[1]
}
return ProtocolVersion{
Major: major,
Minor: minor,
ExperimentalSpecVersion: experimentalSpecVersion,
}, nil
}
type Version struct {
protocolName string // having the protocol name here might sound silly but there are discussions about updating the name right now and we want to support a change
protocolVersion ProtocolVersion
softwareVersion SoftwareVersion
}
func (v Version) GetProtocolVersion() ProtocolVersion {
return v.protocolVersion
}
func (v Version) GetSoftwareVersion() SoftwareVersion {
return v.softwareVersion
}
// GetVersionString() returns the version string to be exchanged between two
// endpoints for version negotiation
func (v Version) GetVersionString() string {
// currently put the experimental spec version as an additional attribute for better compatibility with recent 0.1.5 software rcs
ret := fmt.Sprintf("%s %d.%d %s %d.%d.%d", v.protocolName, v.protocolVersion.Major, v.protocolVersion.Minor,
v.softwareVersion.ImplementationName, v.softwareVersion.Major, v.softwareVersion.Minor, v.softwareVersion.Patch)
if v.protocolVersion.ExperimentalSpecVersion != "" {
ret = fmt.Sprintf("%s experimental_spec_version=%s", ret, v.protocolVersion.ExperimentalSpecVersion)
}
return ret
}
func NewVersion(protocolName string, protocolVersion ProtocolVersion, softwareVersion SoftwareVersion) Version {
return Version{
protocolName: protocolName,
protocolVersion: protocolVersion,
softwareVersion: softwareVersion,
}
}
type InvalidSSHVersion struct {
versionString string
}
func (e InvalidSSHVersion) Error() string {
return fmt.Sprintf("invalid ssh version string: %s", e.versionString)
}
type UnsupportedSSHVersion struct {
versionString string
}
func (e UnsupportedSSHVersion) Error() string {
return fmt.Sprintf("unsupported ssh version: %s", e.versionString)
}
// GetCurrentVersionString() returns the version string to be exchanged between two
// endpoints for version negotiation
func GetCurrentVersionString() string {
return ThisVersion().GetVersionString()
}
func ParseVersionString(versionString string) (version Version, err error) {
fields := strings.Fields(versionString)
if len(fields) < 4 {
log.Error().Msgf("bad SSH version fields")
return Version{}, InvalidSSHVersion{versionString: versionString}
}
protocolName := fields[0]
protocolVersion, err := ParseProtocolVersion(fields[1])
if err != nil {
log.Error().Msgf("could not parse protocol version: %s", err)
return Version{}, err
}
softwareVersion, err := ParseSoftwareVersion(fields[2], fields[3])
if err != nil {
log.Error().Msgf("could not parse software version: %s", err)
return Version{}, err
}
// Temporary tweak to announce a spec version while keeping compatibility with alpha-00 and older versions,
// as alpha-00 and older versions do strict version checking and error as soon as the protocol version is not "3.0".
// This will likely disappear once we decide to remove support for alpha-00 and older versions.
// From that point onwards, the spec version will be announced as part of the version field.
if len(fields) > 4 {
for _, field := range fields[4:] {
subfields := strings.Split(field, "=")
if len(subfields) == 2 && subfields[0] == "experimental_spec_version" {
protocolVersion.ExperimentalSpecVersion = subfields[1]
} else {
log.Debug().Msgf("skipping custom version field %s", field)
}
}
}
return NewVersion(protocolName, protocolVersion, softwareVersion), nil
}
// GetCurrentSoftwareVersion() returns the current software version to be displayed to the user
// For version string to be communicated between endpoints, use GetCurrentVersionString() instead.
func GetCurrentSoftwareVersion() string {
versionStr := ThisVersion().softwareVersion.String()
if SOFTWARE_RC > 0 {
versionStr += fmt.Sprintf("-rc%d", SOFTWARE_RC)
}
return versionStr
}