A Swift framework for connecting to Phoenix Channels in your iOS app.
There are already a few options available for working with Phoenix Channels, such as ObjCPhoenixClient and SwiftPhoenixClient. Both of these are great, and in fact, Stone took a lot of inspiration from the design of ObjCPhoenixClient. But I felt both of these could be made to take advantage of the additional type safety that Swift provides. Because of this, Stone attempts to provide overloads for using custom enums in place of Strings for things like Channel topics and event handlers. It also makes use of Result<T>
enums for things like callbacks for events. Additionally, Stone has built in support for Presence tracking (added in Phoenix 1.2).
Well, as we all know, Swift and Phoenix are both named after birds. You know, two birds, one Stone?...
You should use Stone if you've looking to connect to Channels on your Phoenix server, and are okay with using a fairly opinionated framework that itself has a few dependencies. Stone depends on SwiftWebSocket for Web Socket communication, as well as Wrap and Unbox for JSON serialization/deserialization. I built this framework for a project I'm working on myself, but if you find it helpful then by all means use it!
To create a Socket, simply initialize an instance of the provided Socket class.
guard let url = NSURL(string: "ws://localhost:4000/socket"),
socket = Socket(url: url, heartbeatInterval: 15.0) else {
return
}
Sockets were build to require minimal customization, but you can change the following settings if desired.
socket.shouldReconnectOnError = false // Default is true
socket.shouldAutoJoinChannels = false // Default is true
socket.reconnectInterval = 10.0 // Default is 5.0 (units are seconds)
To receive callbacks for any Socket level events, you can utilize the following hooks.
socket.onOpen = {
print("socket open")
}
socket.onMessage = { (result: Result<Message>) in
// Will print every single message event that comes over the socket.
// print("received message: \(result)")
}
socket.onError = { (error: NSError) in
print("error received: \(error)")
}
socket.onClose = { (code: Int, reason: String, clean: Bool) in
print("socket closed - code: \(code), reason: \(reason), clean: \(clean)")
}
After your Socket is set up, you can optionally provide its connect method with parameters to be included in the URL's query when a connection is made. The below example uses Array<NSURLQueryItem>
, but there is another overload available that takes Dictionary<QueryStringConvertible, QueryStringConvertible>
to force all parameters to provide an implementation of QueryStringConvertible to escape themselves for a query string.
Since stone provides a default implementation of this protocol for String
, you can make use of the toQueryItems() instance method attached to all Dictionary<String, String>
s to convert them into Array<NSURLQueryItem>
.
let params = [NSURLQueryItem(name: "user_id", value: "iPhone")]
socket.connect(params)
When you're done with a socket, you can call socket.disconnect()
, to disconnect from the server.
If you want to reconnect, using the parameters supplied during the first connection, simply call socket.reconnect()
.
Channels are defined on a Socket by Socket basis, and are considered to be unique by their topic. To create a Channel, all you have to do is initialize one, passing the topic as input. This topic can either be a String, or an Enum whose RawType is String.
let channel = Channel(topic: MyTopics.Lobby)
Once you've created a channel, you can use its onEvent
method to get callbacks when different events occur on the given Channel.
channel.onEvent(Event.Custom("new:msg")) { (result: Result<Message>) in
do {
let message: Message = try result.value()
} catch {
print(error)
}
}
To stop receiving callbacks for a given Event, use offEvent()
.
channel.offEvent(Event.Custom("new:msg"))
If desired, Stone is capable of tracking Presence information in Channels. By default, this is disabled, but can be enabled as easily as setting the shouldTrackPresence
instance variable.
channel.shouldTrackPresence = true
Tracking Presence changes can be done by setting the Presence related callbacks on your Channel.
channel.onPresenceDiff { (result: Result<PresenceDiff> in
do {
let diff: PresenceDiff = try result.value()
let leaves: [PresenceChange] = diff.leaves
let joins: [PresenceChange] = diff.joins
} catch {
print(error)
}
}
channel.onPresenceState { (result: Result<Array<PresenceChange>>) in
do {
let connections: [PresenceChange] = try result.value()
} catch {
print(error)
}
}
When you're done configuring your Channel, just add it to your Socket. If you've left socket.shouldAutoJoinChannels
enabled, then you've done. If you've disabled it, you'll need to explicitly call channel.join()
when you've ready to join the Channel's topic.
socket.addChannel(channel)
PresenceChange objects are a simple implementation detail of Stone. They're nothing more than structs containing the name
of the change, and its associated metas
Dictionary. When a Presence state change occurs, you'll be given an Array of these, and when a diff occurs, you'll get a PresenceDiff object, which contains two Arrays of PresenceChanges representing the joins
, and leaves
that happened during this diff.
Once your Socket is connected, and you've joined a Channel, creating and sending Messages is quite simple. Here's an example.
let payload = [
"body": someTextField.text ?? "",
"user": UIDevice.currentDevice().name
]
let message = Message(
topic: MyTopics.Lobby,
event: Event.Custom("new:msg"),
payload: payload
)
channel.sendMessage(message) { (result: Result<Message>) in
do {
let callbackMessage = try result.value()
} catch {
print(error)
}
}
In an attempt to keep all event handling as type safe as possible, Stone provides an Event enum to try to wrap both default and custom events. Some examples of the difference Event types that can be created are as follows.
let phoenixEvent = Event.Phoenix(.Join)
let customEvent = Event.Custom("new:msg")
let presenceEvent = Event.Presence(.Diff)
Add the following line to your Cartfile:
github "0x7fs/Stone" ~> 0.2.2
Check Carthage's README for up to date installation instructions.
As I stated earlier, I made this framework to support my own use, but would love for it to support yours as well. In the spirit of this, if I borked something, or you can think of a cool feature that Stone is missing, please raise an issue and I'll try to incorporate it. If you'd like to help out, pull requests are more than welcome. All I ask is that you try to keep your code style the same as is used in the rest of Stone.
The MIT License (MIT). See LICENSE.md for more information. SwiftWebSocket, Wrap and Unbox are all MIT licensed as well.