mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-23 07:08:24 +00:00
d507126101
This is in preparation for future changes, but also improves the handling when talking to pre-v0.13 clients. It breaks out the Hello message and magic from the rest of the protocol implementation, with the intention that this small part of the protocol will survive future changes. To enable this, and future testing, the new ExchangeHello function takes an interface that can be implemented by future Hello versions and returns a version indendent result type. It correctly detects pre-v0.13 protocols and returns a "too old" error message which gets logged to the user at warning level: [I6KAH] 09:21:36 WARNING: Connecting to [...]: the remote device speaks an older version of the protocol (v0.12) not compatible with this version Conversely, something entirely unknown will generate: [I6KAH] 09:40:27 WARNING: Connecting to [...]: the remote device speaks an unknown (newer?) version of the protocol The intention is that in future iterations the Hello exchange will succeed on at least one side and ExchangeHello will return the actual data from the Hello together with ErrTooOld and an even more precise message can be generated. GitHub-Pull-Request: https://github.com/syncthing/syncthing/pull/3289
187 lines
3.9 KiB
Go
187 lines
3.9 KiB
Go
// Copyright (C) 2014 The Protocol Authors.
|
|
|
|
//go:generate -command genxdr go run ../../vendor/github.com/calmh/xdr/cmd/genxdr/main.go
|
|
//go:generate genxdr -o message_xdr.go message.go
|
|
|
|
package protocol
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/sha256"
|
|
"fmt"
|
|
)
|
|
|
|
var (
|
|
sha256OfEmptyBlock = sha256.Sum256(make([]byte, BlockSize))
|
|
)
|
|
|
|
type IndexMessage struct {
|
|
Folder string // max:256
|
|
Files []FileInfo // max:1000000
|
|
Flags uint32
|
|
Options []Option // max:64
|
|
}
|
|
|
|
type FileInfo struct {
|
|
Name string // max:8192
|
|
Flags uint32
|
|
Modified int64
|
|
Version Vector
|
|
LocalVersion int64
|
|
CachedSize int64 // noencode (cache only)
|
|
Blocks []BlockInfo // max:10000000
|
|
}
|
|
|
|
func (f FileInfo) String() string {
|
|
return fmt.Sprintf("File{Name:%q, Flags:0%o, Modified:%d, Version:%v, Size:%d, Blocks:%v}",
|
|
f.Name, f.Flags, f.Modified, f.Version, f.Size(), f.Blocks)
|
|
}
|
|
|
|
func (f FileInfo) Size() (bytes int64) {
|
|
if f.IsDeleted() || f.IsDirectory() {
|
|
return 128
|
|
}
|
|
if f.CachedSize > 0 {
|
|
return f.CachedSize
|
|
}
|
|
for _, b := range f.Blocks {
|
|
bytes += int64(b.Size)
|
|
}
|
|
f.CachedSize = bytes
|
|
return
|
|
}
|
|
|
|
func (f FileInfo) IsDeleted() bool {
|
|
return f.Flags&FlagDeleted != 0
|
|
}
|
|
|
|
func (f FileInfo) IsInvalid() bool {
|
|
return f.Flags&FlagInvalid != 0
|
|
}
|
|
|
|
func (f FileInfo) IsDirectory() bool {
|
|
return f.Flags&FlagDirectory != 0
|
|
}
|
|
|
|
func (f FileInfo) IsSymlink() bool {
|
|
return f.Flags&FlagSymlink != 0
|
|
}
|
|
|
|
func (f FileInfo) HasPermissionBits() bool {
|
|
return f.Flags&FlagNoPermBits == 0
|
|
}
|
|
|
|
// WinsConflict returns true if "f" is the one to choose when it is in
|
|
// conflict with "other".
|
|
func (f FileInfo) WinsConflict(other FileInfo) bool {
|
|
// If a modification is in conflict with a delete, we pick the
|
|
// modification.
|
|
if !f.IsDeleted() && other.IsDeleted() {
|
|
return true
|
|
}
|
|
if f.IsDeleted() && !other.IsDeleted() {
|
|
return false
|
|
}
|
|
|
|
// The one with the newer modification time wins.
|
|
if f.Modified > other.Modified {
|
|
return true
|
|
}
|
|
if f.Modified < other.Modified {
|
|
return false
|
|
}
|
|
|
|
// The modification times were equal. Use the device ID in the version
|
|
// vector as tie breaker.
|
|
return f.Version.Compare(other.Version) == ConcurrentGreater
|
|
}
|
|
|
|
type BlockInfo struct {
|
|
Offset int64 // noencode (cache only)
|
|
Size int32
|
|
Hash []byte // max:64
|
|
}
|
|
|
|
func (b BlockInfo) String() string {
|
|
return fmt.Sprintf("Block{%d/%d/%x}", b.Offset, b.Size, b.Hash)
|
|
}
|
|
|
|
// IsEmpty returns true if the block is a full block of zeroes.
|
|
func (b BlockInfo) IsEmpty() bool {
|
|
return b.Size == BlockSize && bytes.Equal(b.Hash, sha256OfEmptyBlock[:])
|
|
}
|
|
|
|
type RequestMessage struct {
|
|
Folder string // max:256
|
|
Name string // max:8192
|
|
Offset int64
|
|
Size int32
|
|
Hash []byte // max:64
|
|
Flags uint32
|
|
Options []Option // max:64
|
|
}
|
|
|
|
type ResponseMessage struct {
|
|
Data []byte
|
|
Code int32
|
|
}
|
|
|
|
type ClusterConfigMessage struct {
|
|
Folders []Folder // max:1000000
|
|
Options []Option // max:64
|
|
}
|
|
|
|
type DownloadProgressMessage struct {
|
|
Folder string // max:64
|
|
Updates []FileDownloadProgressUpdate // max:1000000
|
|
Flags uint32
|
|
Options []Option // max:64
|
|
}
|
|
|
|
func (o *ClusterConfigMessage) GetOption(key string) string {
|
|
for _, option := range o.Options {
|
|
if option.Key == key {
|
|
return option.Value
|
|
}
|
|
}
|
|
return ""
|
|
}
|
|
|
|
type Folder struct {
|
|
ID string // max:256
|
|
Label string // max:256
|
|
Devices []Device // max:1000000
|
|
Flags uint32
|
|
Options []Option // max:64
|
|
}
|
|
|
|
type Device struct {
|
|
ID []byte // max:32
|
|
Name string // max:64
|
|
Addresses []string // max:64,2083
|
|
Compression uint32
|
|
CertName string // max:64
|
|
MaxLocalVersion int64
|
|
Flags uint32
|
|
Options []Option // max:64
|
|
}
|
|
|
|
type FileDownloadProgressUpdate struct {
|
|
UpdateType uint32
|
|
Name string // max:8192
|
|
Version Vector
|
|
BlockIndexes []int32 // max:1000000
|
|
}
|
|
|
|
type Option struct {
|
|
Key string // max:64
|
|
Value string // max:1024
|
|
}
|
|
|
|
type CloseMessage struct {
|
|
Reason string // max:1024
|
|
Code int32
|
|
}
|
|
|
|
type EmptyMessage struct{}
|