mirror of
https://github.com/jeremysrand/ListenerApp.git
synced 2024-12-31 03:31:34 +00:00
366 lines
11 KiB
Swift
366 lines
11 KiB
Swift
//
|
|
// GSConnection.swift
|
|
// ListenerGS
|
|
//
|
|
// Created by Jeremy Rand on 2022-03-14.
|
|
//
|
|
|
|
import Foundation
|
|
import os
|
|
|
|
struct GSConnectionErrorMessage: Identifiable {
|
|
var id: String { message }
|
|
let title: String
|
|
let message: String
|
|
}
|
|
|
|
enum GSConnectionState {
|
|
case disconnected
|
|
case connecting
|
|
case connected
|
|
case listening
|
|
case stoplistening
|
|
case deleting
|
|
}
|
|
|
|
extension GSConnectionState: CustomStringConvertible
|
|
{
|
|
var description: String {
|
|
switch self {
|
|
case .disconnected:
|
|
return "disconnected"
|
|
case .connecting:
|
|
return "connecting"
|
|
case .connected:
|
|
return "connected"
|
|
case .listening:
|
|
return "listening"
|
|
case .stoplistening:
|
|
return "stop listening"
|
|
case .deleting:
|
|
return "deleting"
|
|
}
|
|
}
|
|
}
|
|
|
|
protocol SpeechForwarderProtocol
|
|
{
|
|
func startListening(connection: GSConnection) -> Bool
|
|
func stopListening()
|
|
}
|
|
|
|
class GSConnection : ObservableObject {
|
|
@Published var state = GSConnectionState.disconnected
|
|
@Published var textHeard = ""
|
|
@Published var errorMessage : GSConnectionErrorMessage?
|
|
|
|
var speechForwarder : SpeechForwarderProtocol?
|
|
|
|
static let LISTEN_STATE_MSG = 1
|
|
static let LISTEN_TEXT_MSG = 2
|
|
static let LISTEN_SEND_MORE = 3
|
|
|
|
static let port = 19026
|
|
|
|
private var destination = ""
|
|
private var client: TCPClient?
|
|
|
|
private let logger = Logger()
|
|
|
|
private let readQueue = OperationQueue()
|
|
private let writeQueue = OperationQueue()
|
|
private var mainQueue = OperationQueue.main
|
|
|
|
private var canSend = true
|
|
private var lastSent = ""
|
|
|
|
private func changeState(newState : GSConnectionState)
|
|
{
|
|
let oldState = state
|
|
if (oldState == newState) {
|
|
return;
|
|
}
|
|
|
|
var legalTransition = false
|
|
switch (newState)
|
|
{
|
|
case .disconnected:
|
|
legalTransition = ((oldState == .connected) || (oldState == .connecting) || (oldState == .stoplistening))
|
|
|
|
case .connecting:
|
|
legalTransition = (oldState == .disconnected)
|
|
|
|
case .connected:
|
|
legalTransition = ((oldState == .connecting) || (oldState == .listening) || (oldState == .stoplistening))
|
|
|
|
case .listening:
|
|
legalTransition = (oldState == .connected)
|
|
|
|
case .stoplistening:
|
|
legalTransition = ((oldState == .connected) || (oldState == .listening))
|
|
|
|
case .deleting:
|
|
legalTransition = true
|
|
}
|
|
|
|
if (!legalTransition) {
|
|
logger.error("Illegal requested state transition from \(oldState) to \(newState)")
|
|
errorOccurred(title: "Bad State Change", message: "Illegal state transition from \(oldState) to \(newState)")
|
|
} else {
|
|
state = newState
|
|
}
|
|
}
|
|
|
|
func errorOccurred(title: String, message : String)
|
|
{
|
|
mainQueue.addOperation {
|
|
self.errorMessage = GSConnectionErrorMessage(title: title, message: message)
|
|
}
|
|
}
|
|
|
|
private func connectionFailed() {
|
|
errorOccurred(title: "Connect Error", message: "Failed to connect to \(destination)")
|
|
changeState(newState:.disconnected)
|
|
}
|
|
|
|
private func connectionSuccessful()
|
|
{
|
|
changeState(newState:.connected)
|
|
logger.debug("Connected to \(self.destination)")
|
|
}
|
|
|
|
func connect(destination : String) {
|
|
self.destination = destination
|
|
changeState(newState: .connecting)
|
|
readQueue.addOperation { [weak self, destination] in
|
|
self?.logger.debug("Attempting to connect to \(destination)")
|
|
let client = TCPClient(address: destination, port: Int32(GSConnection.port))
|
|
switch client.connect(timeout: 10) {
|
|
case .success:
|
|
self?.mainQueue.addOperation {
|
|
self?.client = client
|
|
self?.connectionSuccessful()
|
|
}
|
|
case .failure(let error):
|
|
client.close()
|
|
self?.logger.error("Failed to connect to \(destination): \(String(describing: error))")
|
|
self?.mainQueue.addOperation {
|
|
self?.connectionFailed()
|
|
}
|
|
return
|
|
}
|
|
|
|
while (true) {
|
|
guard let byteArray = client.read(2) else {
|
|
break
|
|
}
|
|
|
|
if (byteArray.count != 2) {
|
|
break
|
|
}
|
|
|
|
guard let self = self else {
|
|
break
|
|
}
|
|
let data = Data(byteArray)
|
|
do {
|
|
let unpacked = try unpack("<h", data)
|
|
if (unpacked[0] as? Int == GSConnection.LISTEN_SEND_MORE) {
|
|
self.mainQueue.addOperation {
|
|
self.canSend = true
|
|
self.trySend()
|
|
}
|
|
} else {
|
|
self.logger.error("Unexpected message on socket from \(destination)")
|
|
self.errorOccurred(title: "Protocol Error", message: "Unexpected message from the GS")
|
|
break
|
|
}
|
|
}
|
|
catch {
|
|
self.logger.error("Unable to unpack message on socket from \(destination)")
|
|
self.errorOccurred(title: "Protocol Error", message: "Unexpected message from the GS")
|
|
break
|
|
}
|
|
}
|
|
|
|
self?.mainQueue.addOperation { self?.disconnect() }
|
|
}
|
|
}
|
|
|
|
deinit {
|
|
changeState(newState:.deleting)
|
|
disconnect()
|
|
}
|
|
|
|
func disconnect() {
|
|
if (state == .listening) {
|
|
stopListening()
|
|
}
|
|
|
|
if (client != nil) {
|
|
client!.close()
|
|
self.client = nil
|
|
}
|
|
|
|
waitForWriteQueue()
|
|
waitForReadQueue()
|
|
|
|
if (state != .deleting) {
|
|
changeState(newState:.disconnected)
|
|
}
|
|
}
|
|
|
|
func stopListening() {
|
|
logger.debug("Stopped listening")
|
|
if let speechForwarder = speechForwarder {
|
|
speechForwarder.stopListening()
|
|
self.speechForwarder = nil
|
|
}
|
|
|
|
if (state == .listening) {
|
|
changeState(newState: .stoplistening)
|
|
trySend()
|
|
}
|
|
}
|
|
|
|
private func sendListenMsg(isListening: Bool) -> Bool {
|
|
guard let client = client else { return false }
|
|
|
|
switch (client.send(data: pack("<hh", [GSConnection.LISTEN_STATE_MSG, isListening ? 1 : 0]))) {
|
|
case .success:
|
|
break
|
|
case .failure(let error):
|
|
self.logger.error("Unable to send header: \(String(describing: error))")
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func listen(speechForwarder: SpeechForwarderProtocol) {
|
|
textHeard = ""
|
|
lastSent = ""
|
|
writeQueue.addOperation { [weak self] in
|
|
guard let self = self else { return }
|
|
if (!self.sendListenMsg(isListening: true)) {
|
|
self.errorOccurred(title: "Write Error", message: "Unable to send data to the GS")
|
|
return
|
|
}
|
|
|
|
self.mainQueue.addOperation {
|
|
self.changeState(newState: .listening)
|
|
if (!speechForwarder.startListening(connection: self)) {
|
|
self.logger.error("Unable to start listening")
|
|
self.errorOccurred(title: "Speech Error", message: "Unable to start listening for speech")
|
|
self.stopListening()
|
|
return
|
|
}
|
|
self.speechForwarder = speechForwarder
|
|
}
|
|
}
|
|
}
|
|
|
|
private func trySend() {
|
|
if (textHeard == lastSent) {
|
|
if (state == .stoplistening) {
|
|
writeQueue.addOperation {
|
|
_ = self.sendListenMsg(isListening: false)
|
|
}
|
|
changeState(newState: .connected)
|
|
}
|
|
return
|
|
}
|
|
|
|
if (!canSend) {
|
|
return
|
|
}
|
|
|
|
canSend = false
|
|
let stringToSend = textHeard
|
|
writeQueue.addOperation {
|
|
if self.send(latestText: stringToSend, lastSent: self.lastSent) {
|
|
self.mainQueue.addOperation {
|
|
self.lastSent = stringToSend
|
|
self.trySend()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func set(text:String) {
|
|
textHeard = text
|
|
trySend()
|
|
}
|
|
|
|
private func send(latestText : String, lastSent: String) -> Bool {
|
|
guard let client = client else { return false }
|
|
var commonChars = lastSent.count
|
|
while (commonChars > 0) {
|
|
if (latestText.prefix(commonChars) == lastSent.prefix(commonChars)) {
|
|
break
|
|
}
|
|
commonChars -= 1
|
|
}
|
|
var stringToSend = ""
|
|
if (commonChars < lastSent.count) {
|
|
stringToSend = String(repeating: "\u{7f}", count: lastSent.count - commonChars)
|
|
}
|
|
stringToSend.append(contentsOf: latestText.suffix(latestText.count - commonChars).replacingOccurrences(of: "\n", with: "\r"))
|
|
|
|
if (stringToSend.count == 0) {
|
|
return false
|
|
}
|
|
|
|
// JSR_TODO - Handle strings to send that are longer than 64K (doubt that would happen though)
|
|
let nsEnc = CFStringConvertEncodingToNSStringEncoding(CFStringEncoding(CFStringBuiltInEncodings.macRoman.rawValue))
|
|
let encoding = String.Encoding(rawValue: nsEnc) // String.Encoding
|
|
if let bytes = stringToSend.data(using: encoding) {
|
|
switch (client.send(data: pack("<hh", [GSConnection.LISTEN_TEXT_MSG, bytes.count]))) {
|
|
case .success:
|
|
switch (client.send(data: bytes)) {
|
|
case .success:
|
|
logger.debug("Sent text \"\(stringToSend)\"")
|
|
break
|
|
case .failure(let error):
|
|
mainQueue.addOperation {
|
|
self.errorOccurred(title: "Write Error", message: "Unable to send text to the GS")
|
|
self.disconnect()
|
|
}
|
|
logger.error("Failed to send text: \(String(describing: error))")
|
|
return false
|
|
}
|
|
case .failure(let error):
|
|
mainQueue.addOperation {
|
|
self.errorOccurred(title: "Write Error", message: "Unable to send text to the GS")
|
|
self.disconnect()
|
|
}
|
|
logger.error("Failed to send text: \(String(describing: error))")
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func setMainQueueForTest() {
|
|
mainQueue = OperationQueue()
|
|
}
|
|
|
|
func waitForMain() {
|
|
mainQueue.waitUntilAllOperationsAreFinished()
|
|
}
|
|
|
|
func waitForReadQueue() {
|
|
readQueue.waitUntilAllOperationsAreFinished()
|
|
}
|
|
|
|
func waitForWriteQueue() {
|
|
writeQueue.waitUntilAllOperationsAreFinished()
|
|
}
|
|
|
|
func waitForAllQueues() {
|
|
waitForWriteQueue()
|
|
waitForReadQueue()
|
|
waitForMain()
|
|
}
|
|
}
|
|
|