-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathpayload.go
167 lines (139 loc) · 4.54 KB
/
payload.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
// SPDX-FileCopyrightText: 2021 The go-metafeed Authors
//
// SPDX-License-Identifier: MIT
// Package metafeed implements the SSB metafeed spec to enable partial replication.
package metafeed
import (
"bytes"
"fmt"
"time"
"github.com/ssbc/go-metafeed/internal/bencodeext"
refs "github.com/ssbc/go-ssb-refs"
"github.com/ssbc/go-ssb-refs/tfk"
"github.com/zeebo/bencode"
)
// Payload represents a single Payload on a metafeed.
type Payload struct {
Author refs.FeedRef
Sequence int
Previous *refs.MessageRef
Timestamp time.Time
Content bencode.RawMessage
}
var (
_ bencode.Marshaler = (*Payload)(nil)
_ bencode.Unmarshaler = (*Payload)(nil)
)
// MarshalBencode turns the payload into an array of 5 elements:
// author as tfk, sequence, previous as tfk, timestamp as unix ts and content as a bencode entity (usually object or byte string for box2)
func (p *Payload) MarshalBencode() ([]byte, error) {
authorAsTFK, err := tfk.FeedFromRef(p.Author)
if err != nil {
return nil, fmt.Errorf("metafeed/payload: failed to turn author into a tfk: %w", err)
}
autherAsBytes, err := authorAsTFK.MarshalBinary()
if err != nil {
return nil, fmt.Errorf("metafeed/payload: failed to encode author tfk: %w", err)
}
var prevAsBytes []byte
if p.Sequence > 1 {
if p.Previous == nil {
return nil, fmt.Errorf("metafeed/payload: previous nil on seq %d", p.Sequence)
}
prevMsg, err := tfk.MessageFromRef(*p.Previous)
if err != nil {
return nil, fmt.Errorf("metafeed/payload: failed to turn previous into a tfk: %w", err)
}
prevAsBytes, err = prevMsg.MarshalBinary()
if err != nil {
return nil, fmt.Errorf("metafeed/payload: failed to encode previous tfk: %w", err)
}
} else {
prevAsBytes = bencodeext.Null
}
output, err := bencode.EncodeBytes([]interface{}{
autherAsBytes,
int32(p.Sequence),
prevAsBytes,
p.Timestamp.Unix(),
p.Content,
})
if err != nil {
return nil, fmt.Errorf("metafeed/payload: failed to encode payload: %w", err)
}
if n := len(output); n > maxMessageSize {
return nil, fmt.Errorf("metafeed/payload: message is too large (%d bytes)", n)
}
return output, nil
}
const maxMessageSize = 8192
// UnmarshalBencode does the reverse of MarshalBencode. It expects the input to be a bencoded array of 5 entries.
func (p *Payload) UnmarshalBencode(input []byte) error {
if n := len(input); n > maxMessageSize {
return fmt.Errorf("metafeed/payload: message is too large (%d bytes)", n)
}
// first, split up the array in raw parts (decodeing to []interface{} is annoying if we know the types anyhow)
var raw []bencode.RawMessage
err := bencode.DecodeBytes(input, &raw)
if err != nil {
return fmt.Errorf("metafeed/payload: failed to decode raw slices: %w", err)
}
if n := len(raw); n != 5 {
return fmt.Errorf("metafeed/payload: expected at least 5 parts, got %d", n)
}
// elem 1: author
var authorBytes []byte
err = bencode.DecodeBytes(raw[0], &authorBytes)
if err != nil {
return fmt.Errorf("metafeed/payload: failed to get bytes from author position: %w", err)
}
var author tfk.Feed
err = author.UnmarshalBinary(authorBytes)
if err != nil {
return fmt.Errorf("metafeed/payload: failed to decode author tfk: %w", err)
}
p.Author, err = author.Feed()
if err != nil {
return fmt.Errorf("metafeed/payload: invalid author tfk: %w", err)
}
if p.Author.Algo() != refs.RefAlgoFeedBendyButt {
return fmt.Errorf("metafeed/payload: invalid author type: %w", err)
}
// elem 2: sequence
err = bencode.DecodeBytes(raw[1], &p.Sequence)
if err != nil {
return fmt.Errorf("metafeed/payload: expected squence: %w", err)
}
// elem 3: previous
var previousBytes []byte
err = bencode.DecodeBytes(raw[2], &previousBytes)
if err != nil {
return fmt.Errorf("metafeed/payload: failed to decode previous bytes: %w", err)
}
if p.Sequence == 1 {
if !bytes.Equal(previousBytes, bencodeext.Null) {
return fmt.Errorf("metafeed/payload: invalid first message previous entry")
}
} else {
var prev tfk.Message
err = prev.UnmarshalBinary(previousBytes)
if err != nil {
return fmt.Errorf("metafeed/payload: failed to decode previous tfk: %w", err)
}
prevMsg, err := prev.Message()
if err != nil {
return fmt.Errorf("metafeed/payload: failed to turn previous tfk into a message: %w", err)
}
p.Previous = &prevMsg
}
// elem 4: timestamp
var tsInSeconds int64
err = bencode.DecodeBytes(raw[3], &tsInSeconds)
if err != nil {
return fmt.Errorf("metafeed/payload: failed to decode timestamp integer: %w", err)
}
p.Timestamp = time.Unix(tsInSeconds, 0)
// elem 5: content
p.Content = raw[4]
return nil
}