Open Sound Control library for macOS, iOS and tvOS written in Swift.
- OSC address pattern matching and dispatch
- Convenient OSC message value type masking, validation and strong-typing
- Modular: use the provided UDP network layer by default, or use your own
- Support for custom OSC types
- Thread-safe
- Fully unit tested
-
Add OSCKit as a dependency using Swift Package Manager.
-
In an app project or framework, in Xcode:
- Select the menu: File โ Swift Packages โ Add Package Dependency...
- Enter this URL:
https://github.com/orchetect/OSCKit
-
In a Swift Package, add it to the Package.swift dependencies:
.package(url: "https://github.com/orchetect/OSCKit", from: "0.4.0")
-
-
Import the library:
import OSCKit
Or to import OSCKit without networking I/O in order to implement your own UDP sockets:
import OSCKitCore
-
The Examples folder contains projects to get started.
A single global OSC client is all that is needed to send OSC packets. It can be used to send OSC messages to any receiver.
let oscClient = OSCClient()
To send a single message, construct an OSCMessage
and send it using a global OSCClient
instance.
let msg = OSCMessage("/msg2", values: ["string", 123])
oscClient.send(msg, to: "192.168.1.2", port: 8000)
To send multiple OSC messages or nested OSC bundles to the same destination at the same time, pack them in an OSCBundle
and send it using a global OSCClient
instance.
// Option 1: build elements separately
let msg1 = OSCMessage("/msg1")
let msg2 = OSCMessage("/msg2", values: ["string", 123])
let bundle = OSCBundle([msg1, msg2])
// Option 2: build elements inline
let bundle = OSCBundle([
.message("/msg1"),
.message("/msg2", values: ["string", 123])
])
// send the bundle
oscClient.send(bundle, to: "192.168.1.2", port: 8000)
OSC bundles carry a time tag. If not specified, by default a time tag equivalent to "immediate" is used, which indicates to receivers that they should handle the bundle and the message(s) it contains immediately upon receiving them.
It is possible to specify a future time tag. When present, a receiver which adheres to the OSC 1.0 spec will hold the bundle in memory and handle it at the future time specified in the time tag.
// by default, bundles use an immediate time tag; these two lines are identical:
OSCBundle([ ... ])
OSCBundle(timeTag: .immediate(), [ ... ])
// specify a non-immediate time tag of the current time
OSCBundle(timeTag: .now(), [ ... ])
// 5 seconds in the future
OSCBundle(timeTag: .timeIntervalSinceNow(5.0), [ ... ])
// at the specified time as a Date instance
let date = Date( ... )
OSCBundle(timeTag: .future(date), [ ... ])
// a raw time tag can also be supplied
let timeTag: UInt64 = 16535555370123264000
OSCBundle(timeTag: .init(timeTag), [ ... ])
Create a server instance. A single global instance is often created once at app startup to receive OSC messages on a specific port. The default OSC port is 8000 but it may be set to any open port if desired.
let oscServer = OSCServer(port: 8000)
Set the receiver handler.
oscServer.setHandler { [weak self] oscMessage, timeTag in
// Note: handler is called on the main thread
// and is thread-safe if it causes UI updates
do {
try self?.handle(received: oscMessage)
} catch {
print(error)
}
}
private func handle(received oscMessage: OSCMessage) throws {
// handle received messages here
}
Then start the server to begin listening for inbound OSC packets.
// call this once, usually during your app's startup
try oscServer.start()
If received OSC bundles contain a future time tag and the OSCServer
is set to .osc1_0
mode, these bundles will be held in memory automatically and scheduled to be dispatched to the handler at the future time.
Note that as per the OSC 1.1 proposal, this behavior has largely been deprecated. OSCServer
will default to .ignore
and not perform any scheduling unless explicitly set to .osc1_0
mode.
// example: received OSC message with address "/{some,other}/address/*"
private func handle(received message: OSCMessage) throws {
if message.addressPattern.matches(localAddress: "/some/address/methodA") { // will match
// perform methodA action using message.values
}
if message.addressPattern.matches(localAddress: "/some/address/methodB") { // will match
// perform methodB action using message.values
}
if message.addressPattern.matches(localAddress: "/different/methodC") { // won't match
// perform methodC action using message.values
}
}
OSCKit provides an abstraction called OSCAddressSpace
. This object is generally instanced once and stored globally.
Each local OSC address (OSC Method) is registered once with this object in order to enable it to perform matching against received OSC message address patterns. Each method is assigned an ID, and can optionally store a closure.
Method IDs, method closures, or a combination of both may be used for maximum flexibility.
- Registration will return a unique ID token to correspond to each method that is registered. This can be stored and used to identify methods that
OSCAddressSpace
matches for you. - When an OSC message is received:
- Pass its address pattern to the
methods(matching:)
method of theOSCAddressSpace
instance. - This method will pattern-match it against all registered local addresses and return an array of local method IDs that match.
- You can then compare the IDs to ones you stored while registering the local methods.
- Pass its address pattern to the
// instance address space and register methods only once, usually at app startup.
let addressSpace = OSCAddressSpace()
let idMethodA = addressSpace.register(localAddress: "/methodA")
let idMethodB = addressSpace.register(localAddress: "/some/address/methodB")
func handle(message: OSCMessage) throws {
let ids = addressSpace.methods(matching: message.addressPattern)
try ids.forEach { id in
switch id {
case idMethodA:
let str = try message.values.masked(String.self)
performMethodA(str)
case idMethodB:
let (str, int) = try message.values.masked(String.self, Int?.self)
performMethodB(str, int)
default:
print("Received unhandled OSC message:", message)
}
}
}
func performMethodA(_ str: String) { }
func performMethodB(_ str: String, _ int: Int?) { }
- When registering a local method, it can also store a closure. This closure can be executed automatically when matching against a received OSC message's address pattern.
- When an OSC message is received:
- Pass its address pattern to the
dispatch(_:)
method of theOSCAddressSpace
instance. - This method will pattern-match it against all registered local addresses and execute their closures, optionally on a specified queue.
- It also returns an array of local method IDs that match exactly like
methods(matching:)
(which may be discarded if handling of unregistered/unrecognized methods is not needed). - If the returned method ID array is empty, that indicates that no methods matched the address pattern. In this case you may want to handle the unhandled message in a special way.
- Pass its address pattern to the
// instance address space and register methods only once, usually at app startup.
let addressSpace = OSCAddressSpace()
addressSpace.register(localAddress: "/methodA") { values in
guard let str = try? message.values.masked(String.self) else { return }
performMethodA(str)
}
addressSpace.register(localAddress: "/some/address/methodB") { values in
guard let (str, int) = try message.values.masked(String.self, Int?.self) else { return }
performMethodB(str, int)
}
func handle(message: OSCMessage) throws {
let ids = addressSpace.dispatch(message)
if ids.isEmpty {
print("Received unhandled OSC message:", message)
}
}
func performMethodA(_ str: String) { }
func performMethodB(_ str: String, _ int: Int?) { }
Since local OSC "addresses" (OSC Methods) are generally considered methods (akin to functions) which take parameters (OSC values/arguments), in most use cases an OSC Method will have a defined type mask. OSCKit provides a powerful and flexible API to both validate and strongly type an OSC value array.
Validate and unwrap value array with expected member String
:
let str = try oscMessage.values.masked(String.self)
print("string: \(str)")
The special wrapper type AnyOSCNumberValue
is able to match any number and provides easy type-erased access to its contents, converting value types if necessary automatically.
Validate and unwrap value array with expected members String, Int, <number>?
:
let (str, int, num) = try oscMessage.values.masked(String.self,
Int.self,
AnyOSCNumberValue?.self)
print(str, int, num.intValue)
print(str, int, num.doubleValue)
print(str, int, num.base) // access to the strongly typed integer or floating-point value
It is generally easier to use masked()
as demonstrated above, since it handles masking, strongly typing, as well as translation of interpolated (Int8
, Int16
, etc.) and opaque (AnyOSCNumberValue
, etc.) types.
Validate and unwrap value array with expected member String
:
guard oscMessage.values.count == 1 else { ... }
guard let str = oscMessage.values[0] as? String else { ... } // compulsory
print(str) // String
Validate and unwrap value array with expected members String, Int32?, Double?
:
guard (1...3).contains(oscMessage.values.count) else { ... }
guard let str = oscMessage.values[0] as? String else { ... } // compulsory
let int: Int32? = oscMessage.count > 1 ? oscMessage.values[1] as? Int32 : nil // optional
let dbl: Double? = oscMessage.count > 2 ? oscMessage.values[2] as? Double : nil // optional
print(str, int, dbl) // String, Int32?, Double?
It may be desired to imperatively validate and cast values when their expected mask may be unknown.
oscMessage.values.forEach { oscValue
switch oscValue {
case let val as String:
print(val)
case let val as Int32:
print(val)
default:
// unhandled
}
}
The following OSC value types are available, conforming to the Open Sound Control 1.0 specification.
Core OSC Type | Swift Concrete Type | Standard Invocation | Convenience Invocation |
---|---|---|---|
int32, big-endian | Int32 |
Int32( ... ) |
- |
float32, big-endian | Float32 |
Float32( ... ) |
- |
string, null-terminated | String |
String( ... ) |
String literal |
blob, null-terminated | Data |
Data( ... ) |
- |
Extended OSC Type | Swift Concrete Type | Standard Invocation | Convenience Invocation |
---|---|---|---|
bool | Bool |
true , false |
- |
int64, big-endian | Int64 |
Int64( ... ) |
- |
double, big-endian | Double |
Double( ... ) |
- |
ASCII char | Character |
Character( ... ) |
Character literal |
[ ... ] |
OSCArrayValue |
OSCArrayValue([ ... ]) |
.array([ ... ]) |
uint64, big-endian | OSCTimeTag |
OSCTimeTag(1) |
.timeTag(1) |
string, null-terminated | OSCStringAltValue |
OSCStringAltValue("String") |
.stringAlt("String") |
4-byte MIDI channel voice | OSCMIDIValue |
OSCMIDIValue( ... ) |
.midi( ... ) |
impulse/infinitum/bang | OSCImpulseValue |
OSCImpulseValue() |
.impulse |
null | OSCNullValue |
OSCNullValue() |
.null |
OSCKit adds the following interpolated types:
Int // transparently encodes as Int32 core type, converting any BinaryInteger
Int8 // transparently encodes as Int32 core type
Int16 // transparently encodes as Int32 core type
UInt // transparently encodes as Int64 core type
UInt8 // transparently encodes as Int32 core type
UInt16 // transparently encodes as Int32 core type
UInt32 // transparently encodes as Int64 core type
Float16 // transparently encodes as Float32 core type
Float80 // transparently encodes as Double extended type
Substring // transparently encodes as String core type
OSCKit also adds the following opaque type-erasure types:
AnyOSCNumberValue // wraps any BinaryInteger or BinaryFloatingPoint
Will be added in future. In the meantime, refer to this README's Getting Started section, and check out the Example projects.
Coded by a bunch of ๐น hamsters in a trenchcoat that calls itself @orchetect.
Licensed under the MIT license. See LICENSE for details.
If you enjoy using OSCKit and want to contribute to open-source financially, GitHub sponsorship is much appreciated. Feedback and code contributions are also welcome.
Contributions are welcome. Feel free to post an Issue to discuss.