forked from tmaxmax/go-sse
-
Notifications
You must be signed in to change notification settings - Fork 0
/
message.go
355 lines (315 loc) · 10.1 KB
/
message.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
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
package sse
import (
"bytes"
"fmt"
"io"
"reflect"
"strconv"
"strings"
"time"
"unicode/utf8"
"unsafe"
"github.com/tmaxmax/go-sse/internal/parser"
)
func isSingleLine(p string) bool {
_, newlineLen := parser.NewlineIndex(p)
return newlineLen == 0
}
// fieldBytes holds the byte representation of each field type along with a colon at the end.
var (
fieldBytesData = []byte(parser.FieldNameData + ": ")
fieldBytesEvent = []byte(parser.FieldNameEvent + ": ")
fieldBytesRetry = []byte(parser.FieldNameRetry + ": ")
fieldBytesID = []byte(parser.FieldNameID + ": ")
fieldBytesComment = []byte(": ")
)
type chunk struct {
content string
isComment bool
}
var newline = []byte{'\n'}
func (c *chunk) WriteTo(w io.Writer) (int64, error) {
name := fieldBytesData
if c.isComment {
name = fieldBytesComment
}
n, err := w.Write(name)
if err != nil {
return int64(n), err
}
m, err := writeString(w, c.content)
n += m
if err != nil {
return int64(n), err
}
m, err = w.Write(newline)
return int64(n + m), err
}
// Message is the representation of an event sent from the server to its clients.
type Message struct {
chunks []chunk
ID EventID
Type EventType
Retry time.Duration
}
func (e *Message) appendText(isComment bool, chunks ...string) {
for _, c := range chunks {
var content string
for c != "" {
content, c, _ = parser.NextChunk(c)
e.chunks = append(e.chunks, chunk{content: content, isComment: isComment})
}
}
}
// AppendData adds multiple data fields on the message's event from the given strings.
// Each string will be a distinct data field, and if the strings themselves span multiple lines
// they will be broken into multiple fields.
//
// Server-sent events are not suited for binary data: the event fields are delimited by newlines,
// where a newline can be a LF, CR or CRLF sequence. When the client interprets the fields,
// it joins multiple data fields using LF, so information is altered. Here's an example:
//
// initial payload: This is a\r\nmultiline\rtext.\nIt has multiple\nnewline\r\nvariations.
// data sent over the wire:
// data: This is a
// data: multiline
// data: text.
// data: It has multiple
// data: newline
// data: variations
// data received by client: This is a\nmultiline\ntext.\nIt has multiple\nnewline\nvariations.
//
// Each line prepended with "data:" is a field; multiple data fields are joined together using LF as the delimiter.
// If you attempted to send the same payload without prepending the "data:" prefix, like so:
//
// data: This is a
// multiline
// text.
// It has multiple
// newline
// variations
//
// there would be only one data field (the first one). The rest would be different fields, named "multiline", "text.",
// "It has multiple" etc., which are invalid fields according to the protocol.
//
// Besides, the protocol explicitly states that event streams must always be UTF-8 encoded:
// https://html.spec.whatwg.org/multipage/server-sent-events.html#parsing-an-event-stream.
//
// If you need to send binary data, you can use a Base64 encoder or any other encoder that does not output
// any newline characters (\r or \n) and then append the resulted data.
//
// Given that clients treat all newlines the same and replace the original newlines with LF,
// for internal code simplicity AppendData replaces them aswell.
func (e *Message) AppendData(chunks ...string) {
e.appendText(false, chunks...)
}
// AppendComment adds comment fields to the message's event.
// If the comments span multiple lines, they are broken into multiple comment fields.
func (e *Message) AppendComment(comments ...string) {
e.appendText(true, comments...)
}
func (e *Message) writeMessageField(w io.Writer, f messageField, fieldBytes []byte) (int64, error) {
if !f.IsSet() {
return 0, nil
}
n, err := w.Write(fieldBytes)
if err != nil {
return int64(n), err
}
m, err := writeString(w, f.String())
n += m
if err != nil {
return int64(n), err
}
m, err = w.Write(newline)
return int64(n + m), err
}
func (e *Message) writeID(w io.Writer) (int64, error) {
return e.writeMessageField(w, e.ID.messageField, fieldBytesID)
}
func (e *Message) writeType(w io.Writer) (int64, error) {
return e.writeMessageField(w, e.Type.messageField, fieldBytesEvent)
}
func (e *Message) writeRetry(w io.Writer) (int64, error) {
millis := e.Retry.Milliseconds()
if millis <= 0 {
return 0, nil
}
n, err := w.Write(fieldBytesRetry)
if err != nil {
return int64(n), err
}
var buf [13]byte // log10(INT64_MAX / 1e6) ~= 13
i := len(buf) - 1
for millis != 0 {
buf[i] = '0' + byte(millis%10)
i--
millis /= 10
}
m, err := w.Write(buf[i+1:])
n += m
if err != nil {
return int64(n), err
}
m, err = w.Write(newline)
return int64(n + m), err
}
// WriteTo writes the standard textual representation of the message's event to an io.Writer.
// This operation is heavily optimized, so it is strongly preferred over MarshalText or String.
func (e *Message) WriteTo(w io.Writer) (int64, error) {
n, err := e.writeID(w)
if err != nil {
return n, err
}
m, err := e.writeType(w)
n += m
if err != nil {
return n, err
}
m, err = e.writeRetry(w)
n += m
if err != nil {
return n, err
}
for i := range e.chunks {
m, err = e.chunks[i].WriteTo(w)
n += m
if err != nil {
return n, err
}
}
if n == 0 {
return 0, nil
}
o, err := w.Write(newline)
return int64(o) + n, err
}
// MarshalText writes the standard textual representation of the message's event. Marshalling and unmarshalling will
// result in a message with an event that has the same fields; topic will be lost.
//
// If you want to preserve everything, create your own custom marshalling logic.
// For an example using encoding/json, see the top-level MessageCustomJSONMarshal example.
//
// Use the WriteTo method if you don't need the byte representation.
//
// The representation is written to a bytes.Buffer, which means the error is always nil.
// If the buffer grows to a size bigger than the maximum allowed, MarshalText will panic.
// See the bytes.Buffer documentation for more info.
func (e *Message) MarshalText() ([]byte, error) {
b := bytes.Buffer{}
_, err := e.WriteTo(&b)
return b.Bytes(), err
}
// String writes the message's event standard textual representation to a strings.Builder and returns the resulted string.
// It may panic if the representation is too long to be buffered.
//
// Use the WriteTo method if you don't actually need the string representation.
func (e *Message) String() string {
s := strings.Builder{}
_, _ = e.WriteTo(&s)
return s.String()
}
// UnmarshalError is the error returned by the Message's UnmarshalText method.
// If the error is related to a specific field, FieldName will be a non-empty string.
// If no fields were found in the target text or any other errors occurred, only
// a Reason will be provided. Reason is always present.
type UnmarshalError struct {
Reason error
FieldName string
// The value of the invalid field.
FieldValue string
}
func (u *UnmarshalError) Error() string {
if u.FieldName == "" {
return fmt.Sprintf("unmarshal event error: %s", u.Reason.Error())
}
return fmt.Sprintf("unmarshal event error, %s field invalid: %s. contents: %s", u.FieldName, u.Reason.Error(), u.FieldValue)
}
func (u *UnmarshalError) Unwrap() error {
return u.Reason
}
// ErrUnexpectedEOF is returned when unmarshaling a Message from an input that doesn't end in a newline.
//
// If it returned from a Connection, it means that the data from the server has reached EOF
// in the middle of an incomplete event and retries are disabled (normally the client retries
// the connection in this situation).
var ErrUnexpectedEOF = parser.ErrUnexpectedEOF
func (e *Message) reset() {
e.chunks = nil
e.Type = EventType{}
e.ID = EventID{}
e.Retry = 0
}
// UnmarshalText extracts the first event found in the given byte slice into the
// receiver. The input is expected to be a wire format event, as defined by the spec.
// Therefore, previous fields present on the Message will be overwritten
// (i.e. event, ID, comments, data, retry).
//
// Unmarshaling ignores fields with invalid names. If no valid fields are found,
// an error is returned. For a field to be valid it must end in a newline - if the last
// field of the event doesn't end in one, an error is returned.
//
// All returned errors are of type UnmarshalError.
func (e *Message) UnmarshalText(p []byte) error {
e.reset()
s := parser.NewFieldParser(string(p))
s.KeepComments(true)
s.RemoveBOM(true)
loop:
for f := (parser.Field{}); s.Next(&f); {
switch f.Name {
case parser.FieldNameRetry:
if i := strings.IndexFunc(f.Value, func(r rune) bool {
return r < '0' || r > '9'
}); i != -1 {
r, _ := utf8.DecodeRuneInString(f.Value[i:])
return &UnmarshalError{
FieldName: string(f.Name),
FieldValue: f.Value,
Reason: fmt.Errorf("contains character %q, which is not an ASCII digit", r),
}
}
milli, err := strconv.ParseInt(f.Value, 10, 64)
if err != nil {
return &UnmarshalError{
FieldName: string(f.Name),
FieldValue: f.Value,
Reason: fmt.Errorf("invalid retry value: %w", err),
}
}
e.Retry = time.Duration(milli) * time.Millisecond
case parser.FieldNameData, parser.FieldNameComment:
e.chunks = append(e.chunks, chunk{content: f.Value, isComment: f.Name == parser.FieldNameComment})
case parser.FieldNameEvent:
e.Type.value = f.Value
e.Type.set = true
case parser.FieldNameID:
if strings.IndexByte(f.Value, 0) != -1 {
break
}
e.ID.value = f.Value
e.ID.set = true
default: // event end
break loop
}
}
if len(e.chunks) == 0 && !e.Type.IsSet() && e.Retry == 0 && !e.ID.IsSet() || s.Err() != nil {
e.reset()
return &UnmarshalError{Reason: ErrUnexpectedEOF}
}
return nil
}
// Clone returns a copy of the message.
func (e *Message) Clone() *Message {
return &Message{
// The first AppendData will trigger a reallocation.
// Already appended chunks cannot be modified/removed, so this is safe.
chunks: e.chunks[:len(e.chunks):len(e.chunks)],
Retry: e.Retry,
Type: e.Type,
ID: e.ID,
}
}
func writeString(w io.Writer, s string) (int, error) {
return w.Write(unsafe.Slice((*byte)(unsafe.Pointer((*reflect.StringHeader)(unsafe.Pointer(&s)).Data)), len(s)))
}