-
-
Save Ahmed-Mujahid/d28987a6fff41ecf5d1a8bd5d51f196d to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// | |
// UDPListener.swift | |
// | |
// Created by Michael Robert Ellis on 12/16/21. | |
// | |
import Foundation | |
import Network | |
import Combine | |
class UDPListener: ObservableObject { | |
var listener: NWListener? | |
var connection: NWConnection? | |
var queue = DispatchQueue.global(qos: .userInitiated) | |
/// New data will be place in this variable to be received by observers | |
@Published private(set) public var messageReceived: Data? | |
/// When there is an active listening NWConnection this will be `true` | |
@Published private(set) public var isReady: Bool = false | |
/// Default value `true`, this will become false if the UDPListener ceases listening for any reason | |
@Published public var listening: Bool = true | |
/// A convenience init using Int instead of NWEndpoint.Port | |
convenience init(on port: Int) { | |
self.init(on: NWEndpoint.Port(integerLiteral: NWEndpoint.Port.IntegerLiteralType(port))) | |
} | |
/// Use this init or the one that takes an Int to start the listener | |
init(on port: NWEndpoint.Port) { | |
let params = NWParameters.udp | |
params.allowFastOpen = true | |
self.listener = try? NWListener(using: params, on: port) | |
self.listener?.stateUpdateHandler = { update in | |
switch update { | |
case .ready: | |
self.isReady = true | |
print("Listener connected to port \(port)") | |
case .failed, .cancelled: | |
// Announce we are no longer able to listen | |
self.listening = false | |
self.isReady = false | |
print("Listener disconnected from port \(port)") | |
default: | |
print("Listener connecting to port \(port)...") | |
} | |
} | |
self.listener?.newConnectionHandler = { connection in | |
print("Listener receiving new message") | |
self.createConnection(connection: connection) | |
} | |
self.listener?.start(queue: self.queue) | |
} | |
func createConnection(connection: NWConnection) { | |
self.connection = connection | |
self.connection?.stateUpdateHandler = { (newState) in | |
switch (newState) { | |
case .ready: | |
print("Listener ready to receive message - \(connection)") | |
self.receive() | |
case .cancelled, .failed: | |
print("Listener failed to receive message - \(connection)") | |
// Cancel the listener, something went wrong | |
self.listener?.cancel() | |
// Announce we are no longer able to listen | |
self.listening = false | |
default: | |
print("Listener waiting to receive message - \(connection)") | |
} | |
} | |
self.connection?.start(queue: .global()) | |
} | |
func receive() { | |
self.connection?.receiveMessage { data, context, isComplete, error in | |
if let unwrappedError = error { | |
print("Error: NWError received in \(#function) - \(unwrappedError)") | |
return | |
} | |
guard isComplete, let data = data else { | |
print("Error: Received nil Data with context - \(String(describing: context))") | |
return | |
} | |
self.messageReceived = data | |
if self.listening { | |
self.receive() | |
} | |
} | |
} | |
func cancel() { | |
self.listening = false | |
self.connection?.cancel() | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment