-
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
// Copyright 2025 Shota FUJI
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
// SPDX-License-Identifier: Apache-2.0
import Foundation
enum SoodDecodeError: Error {
case signatureMismatch
case missingMessageKind
case propertyKeySizeIsZero
case incompletePropertyKey
case missingPropertyValueSize
case incompletePropertyValue
case invalidPropertyKey
case invalidPropertyValue
}
enum SoodMessageKind: Equatable, LosslessStringConvertible {
/// Unknown, or unsupported message kind. Sorely exists for forward compatibility.
case unknown(String)
/// Discovery query. Client (Roon Extension) sends this via UDP multicast and/or broadcast.
case query
/// Response to a query. Roon Server sends this in response to a query SOOD message.
case response
var description: String {
switch self {
case .query:
return "Q"
case .response:
return "R"
case .unknown(let original):
return original
}
}
init(_ from: String) {
switch from {
case "R":
self = .response
case "Q":
self = .query
default:
self = .unknown(from)
}
}
func data() -> Data {
return String(self).data(using: .utf8)!
}
}
/// SOOD message is a binary data format used for Roon Server Discovery, a process for finding
/// Roon Servers in a network. SOOD messages are sent over UDP via multicast and broadcast.
struct Sood {
private static let signature = "SOOD\u{2}"
let kind: SoodMessageKind
let properties: [String: String]
init(kind: SoodMessageKind, properties: [String: String] = [:]) {
self.kind = kind
self.properties = properties
}
init?(_ from: String) {
try? self.init(from: from.data(using: .utf8)!)
}
private enum PropertyParseState {
case keySize
case key(read: [UInt8], remainingBytes: UInt8)
case valueSizeHi(key: String)
case valueSizeLo(key: String, hi: UInt8)
case value(key: String, read: [UInt8], remainingBytes: UInt16)
}
private enum ParseState {
case signature(expecting: Character, remaining: Substring)
case kind
case properties(
SoodMessageKind,
parsing: PropertyParseState,
parsed: [String: String]
)
}
init(from: Data) throws(SoodDecodeError) {
var state: ParseState = .signature(
expecting: Sood.signature.first!,
remaining: Sood.signature.dropFirst()
)
for byte in from {
switch state {
case .signature(let expecting, let remaining):
if byte != expecting.asciiValue {
throw .signatureMismatch
}
guard let next = remaining.first else {
state = .kind
break
}
state = .signature(expecting: next, remaining: remaining.dropFirst())
break
case .kind:
guard let char = String(bytes: [byte], encoding: .ascii) else {
throw .missingMessageKind
}
state = .properties(.init(char), parsing: .keySize, parsed: [:])
break
case .properties(let kind, let parsing, let parsed):
switch parsing {
case .keySize:
if byte == 0 {
throw .propertyKeySizeIsZero
}
state = .properties(
kind,
parsing: .key(read: [], remainingBytes: byte),
parsed: parsed
)
break
case .key(let read, let remainingBytes):
let read = read + [byte]
if remainingBytes > 1 {
state = .properties(
kind,
parsing: .key(read: read, remainingBytes: remainingBytes - 1),
parsed: parsed
)
break
}
guard let key = String(bytes: read, encoding: .utf8) else {
throw .invalidPropertyKey
}
state = .properties(
kind,
parsing: .valueSizeHi(key: key),
parsed: parsed
)
break
case .valueSizeHi(let key):
state = .properties(
kind,
parsing: .valueSizeLo(key: key, hi: byte),
parsed: parsed
)
break
case .valueSizeLo(let key, let hi):
let hi = UInt16(hi)
let lo = UInt16(byte)
let size = UInt16(bigEndian: (hi.bigEndian << 8) | lo.bigEndian)
state = .properties(
kind,
parsing: .value(key: key, read: [], remainingBytes: size),
parsed: parsed
)
break
case .value(let key, let read, let remainingBytes):
let read = read + [byte]
if remainingBytes > 1 {
state = .properties(
kind,
parsing: .value(
key: key,
read: read,
remainingBytes: remainingBytes - 1
),
parsed: parsed
)
break
}
guard let value = String(bytes: read, encoding: .utf8) else {
throw .invalidPropertyValue
}
var parsed = parsed
_ = parsed.updateValue(value, forKey: key)
state = .properties(kind, parsing: .keySize, parsed: parsed)
break
}
}
}
switch state {
case .signature(_, _):
throw .signatureMismatch
case .kind:
throw .missingMessageKind
case .properties(let kind, parsing: .keySize, let parsed):
self.kind = kind
self.properties = parsed
return
case .properties(_, parsing: .key(_, _), parsed: _):
throw .invalidPropertyKey
case .properties(_, parsing: .valueSizeHi(_), parsed: _):
throw .missingPropertyValueSize
case .properties(_, parsing: .valueSizeLo(_, _), parsed: _):
throw .missingPropertyValueSize
case .properties(_, parsing: .value(_, _, _), parsed: _):
throw .incompletePropertyValue
}
}
func data() -> Data {
var data = Sood.signature.data(using: .utf8)!
data.append(self.kind.data())
for (key, value) in self.properties {
let key = key.data(using: .utf8)!
data.append(contentsOf: [UInt8(key.count)])
data.append(key)
let value = value.data(using: .utf8)!
let hi = UInt8((UInt16(value.count).bigEndian & 0b11111111).littleEndian)
let lo = UInt8(
((UInt16(value.count).bigEndian & 0b11111111_00000000) >> 8)
.littleEndian
)
data.append(contentsOf: [hi, lo])
data.append(value)
}
return data
}
}