69 lines
2.4 KiB
Go
Raw Normal View History

2018-06-09 18:44:59 -05:00
package yggdrasil
// This file contains the version metadata struct
2019-11-29 11:45:02 +02:00
// Used in the initial connection setup and key exchange
2018-06-09 18:44:59 -05:00
// Some of this could arguably go in wire.go instead
import "crypto/ed25519"
// This is the version-specific metadata exchanged at the start of a connection.
2019-11-29 11:45:02 +02:00
// It must always begin with the 4 bytes "meta" and a wire formatted uint64 major version number.
// The current version also includes a minor version number, and the box/sig/link keys that need to be exchanged to open a connection.
2018-06-09 18:44:59 -05:00
type version_metadata struct {
meta [4]byte
ver uint8 // 1 byte in this version
2018-06-09 18:44:59 -05:00
// Everything after this point potentially depends on the version number, and is subject to change in future versions
minorVer uint8 // 1 byte in this version
key ed25519.PublicKey
2018-06-09 18:44:59 -05:00
}
// Gets a base metadata with no keys set, but with the correct version numbers.
2018-06-09 18:44:59 -05:00
func version_getBaseMetadata() version_metadata {
return version_metadata{
meta: [4]byte{'m', 'e', 't', 'a'},
ver: 0,
minorVer: 0,
2018-06-09 18:44:59 -05:00
}
}
// Gets the length of the metadata for this version, used to know how many bytes to read from the start of a connection.
2018-06-09 18:44:59 -05:00
func version_getMetaLength() (mlen int) {
mlen += 4 // meta
mlen++ // ver, as long as it's < 127, which it is in this version
mlen++ // minorVer, as long as it's < 127, which it is in this version
mlen += ed25519.PublicKeySize // key
2018-06-09 18:44:59 -05:00
return
}
// Encodes version metadata into its wire format.
2018-06-09 18:44:59 -05:00
func (m *version_metadata) encode() []byte {
bs := make([]byte, 0, version_getMetaLength())
bs = append(bs, m.meta[:]...)
bs = append(bs, m.ver)
bs = append(bs, m.minorVer)
bs = append(bs, m.key[:]...)
2018-06-09 18:44:59 -05:00
if len(bs) != version_getMetaLength() {
panic("Inconsistent metadata length")
}
return bs
}
// Decodes version metadata from its wire format into the struct.
2018-06-09 18:44:59 -05:00
func (m *version_metadata) decode(bs []byte) bool {
if len(bs) != version_getMetaLength() {
2018-06-09 18:44:59 -05:00
return false
}
offset := 0
offset += copy(m.meta[:], bs[offset:])
m.ver, offset = bs[offset], offset+1
m.minorVer, offset = bs[offset], offset+1
m.key = append([]byte(nil), bs[offset:]...)
2018-06-09 18:44:59 -05:00
return true
}
// Checks that the "meta" bytes and the version numbers are the expected values.
2018-06-09 18:44:59 -05:00
func (m *version_metadata) check() bool {
base := version_getBaseMetadata()
return base.meta == m.meta && base.ver == m.ver && base.minorVer == m.minorVer
}