Meshtastic-Apple/Meshtastic/Views/Settings/Config/Module/MQTTConfig.swift

317 lines
11 KiB
Swift
Raw Normal View History

//
// MQTT.swift
// Meshtastic
//
// Copyright (c) Garth Vander Houwen 9/4/22.
//
import SwiftUI
struct MQTTConfig: View {
2023-03-06 10:33:18 -08:00
@Environment(\.managedObjectContext) var context
@EnvironmentObject var bleManager: BLEManager
@Environment(\.dismiss) private var goBack
var node: NodeInfoEntity?
@State private var isPresentingSaveConfirm: Bool = false
@State var hasChanges: Bool = false
@State var enabled = false
2023-08-06 17:41:46 -07:00
@State var proxyToClientEnabled = false
@State var address = ""
@State var username = ""
@State var password = ""
@State var encryptionEnabled = false
@State var jsonEnabled = false
2023-05-04 21:49:35 -07:00
@State var tlsEnabled = true
@State var root = "msh"
2023-03-06 10:33:18 -08:00
var body: some View {
2023-03-06 10:33:18 -08:00
2022-12-30 11:08:59 -08:00
Form {
if node != nil && node?.metadata == nil && node?.num ?? 0 != bleManager.connectedPeripheral?.num ?? 0 {
Text("There has been no response to a request for device metadata over the admin channel for this node.")
.font(.callout)
.foregroundColor(.orange)
2023-03-14 12:44:10 -07:00
} else if node != nil && node?.num ?? 0 != bleManager.connectedPeripheral?.num ?? 0 {
// Let users know what is going on if they are using remote admin and don't have the config yet
2023-03-14 12:44:10 -07:00
if node?.mqttConfig == nil {
Text("MQTT config data was requested over the admin channel but no response has been returned from the remote node. You can check the status of admin message requests in the admin message log.")
.font(.callout)
.foregroundColor(.orange)
} else {
Text("Remote administration for: \(node?.user?.longName ?? "Unknown")")
.font(.title3)
.onAppear {
setMqttValues()
}
}
2023-03-14 12:44:10 -07:00
} else if node != nil && node?.num ?? 0 == bleManager.connectedPeripheral?.num ?? 0 {
Text("Configuration for: \(node?.user?.longName ?? "Unknown")")
.font(.title3)
} else {
Text("Please connect to a radio to configure settings.")
.font(.callout)
.foregroundColor(.orange)
}
2022-12-30 11:08:59 -08:00
Section(header: Text("options")) {
Toggle(isOn: $enabled) {
2022-12-30 11:08:59 -08:00
Label("enabled", systemImage: "dot.radiowaves.right")
}
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
2023-08-06 17:41:46 -07:00
Toggle(isOn: $proxyToClientEnabled) {
2023-08-10 10:27:54 -07:00
Label("mqtt.clientproxy", systemImage: "iphone.radiowaves.left.and.right")
2023-08-06 17:41:46 -07:00
}
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
2023-03-06 10:33:18 -08:00
2022-12-30 11:08:59 -08:00
Toggle(isOn: $encryptionEnabled) {
2022-12-30 11:08:59 -08:00
Label("Encryption Enabled", systemImage: "lock.icloud")
}
2022-12-30 11:08:59 -08:00
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
2023-05-04 21:49:35 -07:00
Toggle(isOn: $tlsEnabled) {
Label("TLS Enabled", systemImage: "checkmark.shield.fill")
}
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
2022-12-30 11:08:59 -08:00
Toggle(isOn: $jsonEnabled) {
2022-12-30 11:08:59 -08:00
Label("JSON Enabled", systemImage: "ellipsis.curlybraces")
}
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
2023-05-04 21:49:35 -07:00
Text("JSON mode is not reccomended it is incomplete and unstable.")
.font(.caption2)
2022-12-30 11:08:59 -08:00
}
Section(header: Text("Custom Server")) {
HStack {
Label("Address", systemImage: "server.rack")
TextField("Server Address", text: $address)
.foregroundColor(.gray)
.autocapitalization(.none)
.disableAutocorrection(true)
2023-03-06 10:33:18 -08:00
.onChange(of: address, perform: { _ in
2022-12-30 11:08:59 -08:00
let totalBytes = address.utf8.count
// Only mess with the value if it is too big
if totalBytes > 62 {
let firstNBytes = Data(username.utf8.prefix(62))
2022-12-30 11:08:59 -08:00
if let maxBytesString = String(data: firstNBytes, encoding: String.Encoding.utf8) {
// Set the shortName back to the last place where it was the right size
address = maxBytesString
}
}
hasChanges = true
})
.foregroundColor(.gray)
.keyboardType(.default)
}
.autocorrectionDisabled()
2023-03-06 10:33:18 -08:00
2022-12-30 11:08:59 -08:00
HStack {
Label("mqtt.username", systemImage: "person.text.rectangle")
TextField("mqtt.username", text: $username)
.foregroundColor(.gray)
.autocapitalization(.none)
.disableAutocorrection(true)
2023-03-06 10:33:18 -08:00
.onChange(of: username, perform: { _ in
2022-12-30 11:08:59 -08:00
let totalBytes = username.utf8.count
2023-03-06 10:33:18 -08:00
2022-12-30 11:08:59 -08:00
// Only mess with the value if it is too big
if totalBytes > 62 {
2022-12-30 11:08:59 -08:00
let firstNBytes = Data(username.utf8.prefix(62))
2023-03-06 10:33:18 -08:00
2022-12-30 11:08:59 -08:00
if let maxBytesString = String(data: firstNBytes, encoding: String.Encoding.utf8) {
2023-03-06 10:33:18 -08:00
2022-12-30 11:08:59 -08:00
// Set the shortName back to the last place where it was the right size
username = maxBytesString
}
}
hasChanges = true
})
.foregroundColor(.gray)
}
.keyboardType(.default)
.scrollDismissesKeyboard(.interactively)
HStack {
Label("password", systemImage: "wallet.pass")
TextField("password", text: $password)
.foregroundColor(.gray)
.autocapitalization(.none)
.disableAutocorrection(true)
2023-03-06 10:33:18 -08:00
.onChange(of: password, perform: { _ in
2022-12-30 11:08:59 -08:00
let totalBytes = password.utf8.count
2023-03-06 10:33:18 -08:00
2022-12-30 11:08:59 -08:00
// Only mess with the value if it is too big
if totalBytes > 62 {
let firstNBytes = Data(password.utf8.prefix(62))
2023-03-06 10:33:18 -08:00
2022-12-30 11:08:59 -08:00
if let maxBytesString = String(data: firstNBytes, encoding: String.Encoding.utf8) {
2023-03-06 10:33:18 -08:00
2022-12-30 11:08:59 -08:00
// Set the shortName back to the last place where it was the right size
password = maxBytesString
}
2022-12-30 11:08:59 -08:00
}
hasChanges = true
})
.foregroundColor(.gray)
}
2022-12-30 11:08:59 -08:00
.keyboardType(.default)
.scrollDismissesKeyboard(.interactively)
2023-05-04 21:49:35 -07:00
HStack {
Label("Root Topic", systemImage: "tree")
TextField("Root Topic", text: $root)
.foregroundColor(.gray)
.onChange(of: root, perform: { _ in
let totalBytes = root.utf8.count
// Only mess with the value if it is too big
if totalBytes > 14 {
let firstNBytes = Data(root.utf8.prefix(14))
if let maxBytesString = String(data: firstNBytes, encoding: String.Encoding.utf8) {
// Set the shortName back to the last place where it was the right size
root = maxBytesString
}
}
})
.foregroundColor(.gray)
}
.keyboardType(.asciiCapable)
.scrollDismissesKeyboard(.interactively)
.disableAutocorrection(true)
Text("The root topic to use for MQTT messages. Default is \"msh\". This is useful if you want to use a single MQTT server for multiple meshtastic networks and separate them via ACLs")
.font(.caption2)
}
2022-12-30 11:08:59 -08:00
Text("WiFi or Ethernet must also be enabled for MQTT to work. You can set uplink and downlink for each channel.")
.font(.callout)
}
.scrollDismissesKeyboard(.interactively)
2023-02-01 09:19:45 -08:00
.disabled(self.bleManager.connectedPeripheral == nil || node?.mqttConfig == nil)
2023-03-06 10:33:18 -08:00
2022-12-30 11:08:59 -08:00
Button {
isPresentingSaveConfirm = true
} label: {
Label("save", systemImage: "square.and.arrow.down")
}
.disabled(bleManager.connectedPeripheral == nil || !hasChanges)
.buttonStyle(.bordered)
.buttonBorderShape(.capsule)
.controlSize(.large)
.padding()
.confirmationDialog(
"are.you.sure",
isPresented: $isPresentingSaveConfirm,
titleVisibility: .visible
) {
2023-02-22 20:31:08 -08:00
let connectedNode = getNodeInfo(id: bleManager.connectedPeripheral?.num ?? -1, context: context)
if connectedNode != nil {
let nodeName = node?.user?.longName ?? "unknown".localized
let buttonText = String.localizedStringWithFormat("save.config %@".localized, nodeName)
Button(buttonText) {
var mqtt = ModuleConfig.MQTTConfig()
mqtt.enabled = self.enabled
2023-08-06 17:41:46 -07:00
mqtt.proxyToClientEnabled = self.proxyToClientEnabled
mqtt.address = self.address
mqtt.username = self.username
mqtt.password = self.password
2023-05-04 21:49:35 -07:00
mqtt.root = self.root
mqtt.encryptionEnabled = self.encryptionEnabled
mqtt.jsonEnabled = self.jsonEnabled
2023-05-04 21:49:35 -07:00
mqtt.tlsEnabled = self.tlsEnabled
let adminMessageId = bleManager.saveMQTTConfig(config: mqtt, fromUser: connectedNode!.user!, toUser: node!.user!, adminIndex: connectedNode?.myInfo?.adminIndex ?? 0)
if adminMessageId > 0 {
// Should show a saved successfully alert once I know that to be true
// for now just disable the button after a successful save
hasChanges = false
goBack()
}
}
}
}
2022-12-30 11:08:59 -08:00
message: {
Text("config.save.confirm")
}
2022-12-13 07:49:46 -08:00
.navigationTitle("mqtt.config")
.navigationBarItems(trailing:
ZStack {
ConnectedDevice(bluetoothOn: bleManager.isSwitchedOn, deviceConnected: bleManager.connectedPeripheral != nil, name: (bleManager.connectedPeripheral != nil) ? bleManager.connectedPeripheral.shortName : "????")
})
.onAppear {
2022-10-15 01:00:13 -07:00
self.bleManager.context = context
setMqttValues()
2023-03-06 10:33:18 -08:00
2023-02-01 09:19:45 -08:00
// Need to request a TelemetryModuleConfig from the remote node before allowing changes
if bleManager.connectedPeripheral != nil && node?.mqttConfig == nil {
2023-02-01 09:19:45 -08:00
print("empty mqtt module config")
let connectedNode = getNodeInfo(id: bleManager.connectedPeripheral.num, context: context)
2023-03-05 14:40:07 -08:00
if node != nil && connectedNode != nil {
_ = bleManager.requestMqttModuleConfig(fromUser: connectedNode!.user!, toUser: node!.user!, adminIndex: connectedNode?.myInfo?.adminIndex ?? 0)
2023-02-01 09:19:45 -08:00
}
}
}
2023-05-04 21:49:35 -07:00
.onChange(of: address) { newAddress in
if node != nil && node?.mqttConfig != nil {
if newAddress != node!.mqttConfig!.address { hasChanges = true }
}
}
.onChange(of: username) { newUsername in
if node != nil && node?.mqttConfig != nil {
if newUsername != node!.mqttConfig!.username { hasChanges = true }
}
}
.onChange(of: password) { newPassword in
if node != nil && node?.mqttConfig != nil {
if newPassword != node!.mqttConfig!.password { hasChanges = true }
}
}
.onChange(of: root) { newRoot in
if node != nil && node?.mqttConfig != nil {
if newRoot != node!.mqttConfig!.root { hasChanges = true }
}
}
.onChange(of: enabled) { newEnabled in
2023-01-02 20:11:58 -08:00
if node != nil && node?.mqttConfig != nil {
2022-09-10 17:14:03 -07:00
if newEnabled != node!.mqttConfig!.enabled { hasChanges = true }
}
}
2023-08-06 17:41:46 -07:00
.onChange(of: proxyToClientEnabled) { newProxyToClientEnabled in
if node != nil && node?.mqttConfig != nil {
if newProxyToClientEnabled != node!.mqttConfig!.proxyToClientEnabled { hasChanges = true }
}
}
.onChange(of: encryptionEnabled) { newEncryptionEnabled in
2023-01-02 20:11:58 -08:00
if node != nil && node?.mqttConfig != nil {
if newEncryptionEnabled != node!.mqttConfig!.encryptionEnabled { hasChanges = true }
}
}
.onChange(of: jsonEnabled) { newJsonEnabled in
2023-01-02 20:11:58 -08:00
if node != nil && node?.mqttConfig != nil {
if newJsonEnabled != node!.mqttConfig!.jsonEnabled { hasChanges = true }
}
}
2023-05-04 21:49:35 -07:00
.onChange(of: tlsEnabled) { newTlsEnabled in
if node != nil && node?.mqttConfig != nil {
if newTlsEnabled != node!.mqttConfig!.tlsEnabled { hasChanges = true }
}
}
}
func setMqttValues() {
self.enabled = (node?.mqttConfig?.enabled ?? false)
2023-08-06 17:41:46 -07:00
self.proxyToClientEnabled = (node?.mqttConfig?.proxyToClientEnabled ?? false)
self.address = node?.mqttConfig?.address ?? ""
self.username = node?.mqttConfig?.username ?? ""
self.password = node?.mqttConfig?.password ?? ""
2023-05-04 21:49:35 -07:00
self.root = node?.mqttConfig?.root ?? "msh"
self.encryptionEnabled = (node?.mqttConfig?.encryptionEnabled ?? false)
self.jsonEnabled = (node?.mqttConfig?.jsonEnabled ?? false)
2023-05-04 21:49:35 -07:00
self.tlsEnabled = (node?.mqttConfig?.tlsEnabled ?? false)
self.hasChanges = false
}
}