diff --git a/hdw-a333x-common/src/systems/datalink/router/src/webinterfaces/HoppieConnector.ts b/hdw-a333x-common/src/systems/datalink/router/src/webinterfaces/HoppieConnector.ts new file mode 100644 index 000000000..5d32d1b4f --- /dev/null +++ b/hdw-a333x-common/src/systems/datalink/router/src/webinterfaces/HoppieConnector.ts @@ -0,0 +1,395 @@ +// Copyright (c) 2021 FlyByWire Simulations +// SPDX-License-Identifier: GPL-3.0 + +import { NXDataStore } from '@flybywiresim/fbw-sdk'; +import { Hoppie } from '@flybywiresim/api-client'; +import { + AtsuStatusCodes, + CpdlcMessage, + FreetextMessage, + FansMode, + CpdlcMessagesUplink, + CpdlcMessageElement, + CpdlcMessageContent, + CpdlcMessageExpectedResponseType, + AtsuMessage, + AtsuMessageNetwork, + AtsuMessageDirection, + AtsuMessageComStatus, + AtsuMessageSerializationFormat, +} from '@datalink/common'; + +/** + * Defines the connector to the hoppies network + */ +export class HoppieConnector { + private static flightNumber: string = ''; + + public static fansMode: FansMode = FansMode.FansNone; + + public static async activateHoppie() { + SimVar.SetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number', 0); + + if (NXDataStore.get('CONFIG_HOPPIE_ENABLED', 'DISABLED') === 'DISABLED') { + console.log('Hoppie deactivated in EFB'); + return; + } + + if (NXDataStore.get('CONFIG_HOPPIE_USERID', '') === '') { + console.log('No Hoppie-ID set'); + return; + } + + const metarSrc = NXDataStore.get('CONFIG_METAR_SRC', 'MSFS'); + if (metarSrc !== 'VATSIM' && metarSrc !== 'IVAO') { + console.log('Invalid METAR source'); + return; + } + + const atisSrc = NXDataStore.get('CONFIG_ATIS_SRC', 'FAA'); + if (atisSrc !== 'VATSIM' && atisSrc !== 'IVAO') { + console.log('Invalid ATIS source'); + return; + } + + const body = { + logon: NXDataStore.get('CONFIG_HOPPIE_USERID', ''), + from: 'HDWA333X', + to: 'ALL-CALLSIGNS', + type: 'ping', + packet: '', + }; + + Hoppie.sendRequest(body).then((resp) => { + if (resp.response !== 'error {illegal logon code}') { + SimVar.SetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number', 1); + console.log('Activated Hoppie ID'); + } else { + console.log('Invalid Hoppie-ID set'); + } + }); + } + + public static deactivateHoppie(): void { + SimVar.SetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number', 0); + } + + public static async connect(flightNo: string): Promise { + if (SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') !== 1) { + HoppieConnector.flightNumber = flightNo; + return AtsuStatusCodes.NoHoppieConnection; + } + + return HoppieConnector.isCallsignInUse(flightNo).then((code) => { + if (code === AtsuStatusCodes.Ok) { + HoppieConnector.flightNumber = flightNo; + return HoppieConnector.poll().then(() => code); + } + return code; + }); + } + + public static disconnect(): AtsuStatusCodes { + HoppieConnector.flightNumber = ''; + return AtsuStatusCodes.Ok; + } + + public static async isCallsignInUse(station: string): Promise { + if (SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') !== 1) { + return AtsuStatusCodes.NoHoppieConnection; + } + + const body = { + logon: NXDataStore.get('CONFIG_HOPPIE_USERID', ''), + from: station, + to: 'ALL-CALLSIGNS', + type: 'ping', + packet: station, + }; + const text = await Hoppie.sendRequest(body).then((resp) => resp.response); + + if (text === 'error {callsign already in use}') { + return AtsuStatusCodes.CallsignInUse; + } + if (text.includes('error')) { + return AtsuStatusCodes.ProxyError; + } + if (text.startsWith('ok') !== true) { + return AtsuStatusCodes.ComFailed; + } + + return AtsuStatusCodes.Ok; + } + + public static async isStationAvailable(station: string): Promise { + if (SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') !== 1 || HoppieConnector.flightNumber === '') { + return AtsuStatusCodes.NoHoppieConnection; + } + + if (station === HoppieConnector.flightNumber) { + return AtsuStatusCodes.OwnCallsign; + } + + const body = { + logon: NXDataStore.get('CONFIG_HOPPIE_USERID', ''), + from: HoppieConnector.flightNumber, + to: 'ALL-CALLSIGNS', + type: 'ping', + packet: station, + }; + const text = await Hoppie.sendRequest(body).then((resp) => resp.response); + + if (text.includes('error')) { + return AtsuStatusCodes.ProxyError; + } + if (text.startsWith('ok') !== true) { + return AtsuStatusCodes.ComFailed; + } + if (text !== `ok {${station}}`) { + return AtsuStatusCodes.NoAtc; + } + + return AtsuStatusCodes.Ok; + } + + private static async sendMessage(message: AtsuMessage, type: string): Promise { + if (SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') !== 1 || HoppieConnector.flightNumber === '') { + return AtsuStatusCodes.NoHoppieConnection; + } + + const body = { + logon: NXDataStore.get('CONFIG_HOPPIE_USERID', ''), + from: HoppieConnector.flightNumber, + to: message.Station, + type, + packet: message.serialize(AtsuMessageSerializationFormat.Network), + }; + const text = await Hoppie.sendRequest(body).then((resp) => resp.response).catch(() => 'proxy'); + + if (text === 'proxy') { + return AtsuStatusCodes.ProxyError; + } + + if (text !== 'ok') { + return AtsuStatusCodes.ComFailed; + } + + return AtsuStatusCodes.Ok; + } + + public static async sendTelexMessage(message: AtsuMessage, force: boolean): Promise { + if (HoppieConnector.flightNumber !== '' && (force || SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') === 1)) { + return HoppieConnector.sendMessage(message, 'telex'); + } + return AtsuStatusCodes.NoHoppieConnection; + } + + public static async sendCpdlcMessage(message: CpdlcMessage, force: boolean): Promise { + if (HoppieConnector.flightNumber !== '' && (force || SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') === 1)) { + return HoppieConnector.sendMessage(message, 'cpdlc'); + } + return AtsuStatusCodes.NoHoppieConnection; + } + + private static levenshteinDistance(template: string, message: string, content: CpdlcMessageContent[]): number { + let elements = message.replace(/\n/g, ' ').split(' '); + let validContent = true; + + // try to match the content + content.forEach((entry) => { + const result = entry.validateAndReplaceContent(elements); + if (!result.matched) { + validContent = false; + } else { + elements = result.remaining; + } + }); + if (!validContent) return 100000; + const correctedMessage = elements.join(' '); + + // initialize the track matrix + const track = Array(correctedMessage.length + 1).fill(null).map(() => Array(template.length + 1).fill(null)); + for (let i = 0; i <= template.length; ++i) track[0][i] = i; + for (let i = 0; i <= correctedMessage.length; ++i) track[i][0] = i; + + for (let j = 1; j <= correctedMessage.length; ++j) { + for (let i = 1; i <= template.length; ++i) { + const indicator = template[i - 1] === correctedMessage[j - 1] ? 0 : 1; + track[j][i] = Math.min( + track[j][i - 1] + 1, // delete + track[j - 1][i] + 1, // insert + track[j - 1][i - 1] + indicator, // substitude + ); + } + } + + return track[correctedMessage.length][template.length]; + } + + private static cpdlcMessageClassification(message: string): CpdlcMessageElement | undefined { + const scores: [number, string][] = []; + let minScore = 100000; + + // clear the message from marker, etc. + const clearedMessage = message.replace('@', '').replace('_', ' '); + + // test all uplink messages + for (const ident in CpdlcMessagesUplink) { + if ({}.hasOwnProperty.call(CpdlcMessagesUplink, ident)) { + const data = CpdlcMessagesUplink[ident]; + + if (HoppieConnector.fansMode === FansMode.FansNone || data[1].FansModes.includes(HoppieConnector.fansMode)) { + let minDistance = 100000; + + data[0].forEach((template) => { + const distance = HoppieConnector.levenshteinDistance(template, clearedMessage, data[1].Content); + if (minDistance > distance) minDistance = distance; + }); + + scores.push([minDistance, ident]); + if (minScore > minDistance) minScore = minDistance; + } + } + } + + // get all entries with the minimal score + let matches: string[] = []; + scores.forEach((elem) => { + if (elem[0] === minScore) matches.push(elem[1]); + }); + + console.log(`Found matches: ${matches}, score: ${minScore}`); + if (matches.length === 0) return undefined; + + // check if message without parameters are in, but the minScore not empty + if (matches.length > 1 && minScore !== 0) { + const nonEmpty = matches.filter((match) => CpdlcMessagesUplink[match][1].Content.length !== 0); + if (nonEmpty.length !== 0 && matches.length !== nonEmpty.length) { + console.log(`Ignoring ${matches.length - nonEmpty.length} messages without arguments. Remaining ${nonEmpty}`); + matches = nonEmpty; + } + } + + // check if more than the freetext-entry is valid + if (matches.length > 1) { + const nonFreetext = matches.filter((match) => match !== 'UM169' && match !== 'UM183'); + if (nonFreetext.length !== 0 && matches.length !== nonFreetext.length) { + console.log(`Ignoring ${matches.length - nonFreetext.length} freetext messages. Remaining: ${nonFreetext}`); + matches = nonFreetext; + } + } + + // check if the FANS mode is invalid + if (matches.length > 1 && this.fansMode !== FansMode.FansNone) { + const validFans = matches.filter((match) => CpdlcMessagesUplink[match][1].FansModes.findIndex((elem) => elem === this.fansMode) !== -1); + if (validFans.length !== 0 && matches.length !== validFans.length) { + console.log(`Ignoring ${matches.length - validFans.length} invalid FANS messages. Remaining: ${validFans}`); + matches = validFans; + } + } + + // TODO add some more heuristic about messages + + // create a deep-copy of the message + const retval: CpdlcMessageElement = CpdlcMessagesUplink[matches[0]][1].deepCopy(); + let elements = message.split(' '); + console.log(`Selected UM-ID: ${matches[0]}`); + + // parse the content and store it in the deep copy + retval.Content.forEach((entry) => { + const result = entry.validateAndReplaceContent(elements); + elements = result.remaining; + }); + + return retval; + } + + public static async poll(): Promise<[AtsuStatusCodes, AtsuMessage[]]> { + const retval: AtsuMessage[] = []; + + if (SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') !== 1 || HoppieConnector.flightNumber === '') { + return [AtsuStatusCodes.NoHoppieConnection, retval]; + } + + try { + const body = { + logon: NXDataStore.get('CONFIG_HOPPIE_USERID', ''), + from: HoppieConnector.flightNumber, + to: HoppieConnector.flightNumber, + type: 'poll', + }; + const text = await Hoppie.sendRequest(body).then((resp) => resp.response).catch(() => 'proxy'); + + // proxy error during request + if (text === 'proxy') { + return [AtsuStatusCodes.ProxyError, retval]; + } + + // something went wrong + if (!text.startsWith('ok')) { + return [AtsuStatusCodes.ComFailed, retval]; + } + + // split up the received data into multiple messages + let messages = text.split(/({[\s\S\n]*?})/gm); + messages = messages.filter((elem) => elem !== 'ok' && elem !== 'ok ' && elem !== '} ' && elem !== '}' && elem !== ''); + + // create the messages + messages.forEach((element) => { + // get the single entries of the message + // example: [CALLSIGN telex, {Hello world!}] + const entries = element.substring(1).split(/({[\s\S\n]*?})/gm); + + // get all relevant information + const metadata = entries[0].split(' '); + const sender = metadata[0].toUpperCase(); + const type = metadata[1].toLowerCase(); + const content = entries[1].replace(/{/, '').replace(/}/, '').toUpperCase(); + + switch (type) { + case 'telex': + const freetext = new FreetextMessage(); + freetext.Network = AtsuMessageNetwork.Hoppie; + freetext.Station = sender; + freetext.Direction = AtsuMessageDirection.Uplink; + freetext.ComStatus = AtsuMessageComStatus.Received; + freetext.Message = content; + retval.push(freetext); + break; + case 'cpdlc': + const cpdlc = new CpdlcMessage(); + cpdlc.Station = sender; + cpdlc.Direction = AtsuMessageDirection.Uplink; + cpdlc.ComStatus = AtsuMessageComStatus.Received; + + // split up the data + const elements = content.split('/'); + cpdlc.CurrentTransmissionId = parseInt(elements[2]); + if (elements[3] !== '') { + cpdlc.PreviousTransmissionId = parseInt(elements[3]); + } + cpdlc.Message = elements[5].replace(/@/g, '').replace(/_/g, '\n'); + cpdlc.Content.push(HoppieConnector.cpdlcMessageClassification(cpdlc.Message)); + if ((elements[4] as CpdlcMessageExpectedResponseType) !== cpdlc.Content[0]?.ExpectedResponse) { + cpdlc.Content[0].ExpectedResponse = (elements[4] as CpdlcMessageExpectedResponseType); + } + + retval.push(cpdlc); + break; + default: + break; + } + }); + + return [AtsuStatusCodes.Ok, retval]; + } catch (_err) { + return [AtsuStatusCodes.NoHoppieConnection, []]; + } + } + + public static pollInterval(): number { + // To comply with Hoppie rate limits, we choose a random number between 45 and 75, as recommend by Hoppie. Ref to: https://www.hoppie.nl/acars/system/tech.html + const pollTimeSeconds = Math.floor(Math.random() * (75 - 45 + 1)) + 45; + return pollTimeSeconds * 1000; + } +} \ No newline at end of file diff --git a/hdw-a339x-common/src/systems/datalink/router/src/webinterfaces/HoppieConnector.ts b/hdw-a339x-common/src/systems/datalink/router/src/webinterfaces/HoppieConnector.ts new file mode 100644 index 000000000..d78efbd74 --- /dev/null +++ b/hdw-a339x-common/src/systems/datalink/router/src/webinterfaces/HoppieConnector.ts @@ -0,0 +1,395 @@ +// Copyright (c) 2021 FlyByWire Simulations +// SPDX-License-Identifier: GPL-3.0 + +import { NXDataStore } from '@flybywiresim/fbw-sdk'; +import { Hoppie } from '@flybywiresim/api-client'; +import { + AtsuStatusCodes, + CpdlcMessage, + FreetextMessage, + FansMode, + CpdlcMessagesUplink, + CpdlcMessageElement, + CpdlcMessageContent, + CpdlcMessageExpectedResponseType, + AtsuMessage, + AtsuMessageNetwork, + AtsuMessageDirection, + AtsuMessageComStatus, + AtsuMessageSerializationFormat, +} from '@datalink/common'; + +/** + * Defines the connector to the hoppies network + */ +export class HoppieConnector { + private static flightNumber: string = ''; + + public static fansMode: FansMode = FansMode.FansNone; + + public static async activateHoppie() { + SimVar.SetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number', 0); + + if (NXDataStore.get('CONFIG_HOPPIE_ENABLED', 'DISABLED') === 'DISABLED') { + console.log('Hoppie deactivated in EFB'); + return; + } + + if (NXDataStore.get('CONFIG_HOPPIE_USERID', '') === '') { + console.log('No Hoppie-ID set'); + return; + } + + const metarSrc = NXDataStore.get('CONFIG_METAR_SRC', 'MSFS'); + if (metarSrc !== 'VATSIM' && metarSrc !== 'IVAO') { + console.log('Invalid METAR source'); + return; + } + + const atisSrc = NXDataStore.get('CONFIG_ATIS_SRC', 'FAA'); + if (atisSrc !== 'VATSIM' && atisSrc !== 'IVAO') { + console.log('Invalid ATIS source'); + return; + } + + const body = { + logon: NXDataStore.get('CONFIG_HOPPIE_USERID', ''), + from: 'HDWA339X', + to: 'ALL-CALLSIGNS', + type: 'ping', + packet: '', + }; + + Hoppie.sendRequest(body).then((resp) => { + if (resp.response !== 'error {illegal logon code}') { + SimVar.SetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number', 1); + console.log('Activated Hoppie ID'); + } else { + console.log('Invalid Hoppie-ID set'); + } + }); + } + + public static deactivateHoppie(): void { + SimVar.SetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number', 0); + } + + public static async connect(flightNo: string): Promise { + if (SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') !== 1) { + HoppieConnector.flightNumber = flightNo; + return AtsuStatusCodes.NoHoppieConnection; + } + + return HoppieConnector.isCallsignInUse(flightNo).then((code) => { + if (code === AtsuStatusCodes.Ok) { + HoppieConnector.flightNumber = flightNo; + return HoppieConnector.poll().then(() => code); + } + return code; + }); + } + + public static disconnect(): AtsuStatusCodes { + HoppieConnector.flightNumber = ''; + return AtsuStatusCodes.Ok; + } + + public static async isCallsignInUse(station: string): Promise { + if (SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') !== 1) { + return AtsuStatusCodes.NoHoppieConnection; + } + + const body = { + logon: NXDataStore.get('CONFIG_HOPPIE_USERID', ''), + from: station, + to: 'ALL-CALLSIGNS', + type: 'ping', + packet: station, + }; + const text = await Hoppie.sendRequest(body).then((resp) => resp.response); + + if (text === 'error {callsign already in use}') { + return AtsuStatusCodes.CallsignInUse; + } + if (text.includes('error')) { + return AtsuStatusCodes.ProxyError; + } + if (text.startsWith('ok') !== true) { + return AtsuStatusCodes.ComFailed; + } + + return AtsuStatusCodes.Ok; + } + + public static async isStationAvailable(station: string): Promise { + if (SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') !== 1 || HoppieConnector.flightNumber === '') { + return AtsuStatusCodes.NoHoppieConnection; + } + + if (station === HoppieConnector.flightNumber) { + return AtsuStatusCodes.OwnCallsign; + } + + const body = { + logon: NXDataStore.get('CONFIG_HOPPIE_USERID', ''), + from: HoppieConnector.flightNumber, + to: 'ALL-CALLSIGNS', + type: 'ping', + packet: station, + }; + const text = await Hoppie.sendRequest(body).then((resp) => resp.response); + + if (text.includes('error')) { + return AtsuStatusCodes.ProxyError; + } + if (text.startsWith('ok') !== true) { + return AtsuStatusCodes.ComFailed; + } + if (text !== `ok {${station}}`) { + return AtsuStatusCodes.NoAtc; + } + + return AtsuStatusCodes.Ok; + } + + private static async sendMessage(message: AtsuMessage, type: string): Promise { + if (SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') !== 1 || HoppieConnector.flightNumber === '') { + return AtsuStatusCodes.NoHoppieConnection; + } + + const body = { + logon: NXDataStore.get('CONFIG_HOPPIE_USERID', ''), + from: HoppieConnector.flightNumber, + to: message.Station, + type, + packet: message.serialize(AtsuMessageSerializationFormat.Network), + }; + const text = await Hoppie.sendRequest(body).then((resp) => resp.response).catch(() => 'proxy'); + + if (text === 'proxy') { + return AtsuStatusCodes.ProxyError; + } + + if (text !== 'ok') { + return AtsuStatusCodes.ComFailed; + } + + return AtsuStatusCodes.Ok; + } + + public static async sendTelexMessage(message: AtsuMessage, force: boolean): Promise { + if (HoppieConnector.flightNumber !== '' && (force || SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') === 1)) { + return HoppieConnector.sendMessage(message, 'telex'); + } + return AtsuStatusCodes.NoHoppieConnection; + } + + public static async sendCpdlcMessage(message: CpdlcMessage, force: boolean): Promise { + if (HoppieConnector.flightNumber !== '' && (force || SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') === 1)) { + return HoppieConnector.sendMessage(message, 'cpdlc'); + } + return AtsuStatusCodes.NoHoppieConnection; + } + + private static levenshteinDistance(template: string, message: string, content: CpdlcMessageContent[]): number { + let elements = message.replace(/\n/g, ' ').split(' '); + let validContent = true; + + // try to match the content + content.forEach((entry) => { + const result = entry.validateAndReplaceContent(elements); + if (!result.matched) { + validContent = false; + } else { + elements = result.remaining; + } + }); + if (!validContent) return 100000; + const correctedMessage = elements.join(' '); + + // initialize the track matrix + const track = Array(correctedMessage.length + 1).fill(null).map(() => Array(template.length + 1).fill(null)); + for (let i = 0; i <= template.length; ++i) track[0][i] = i; + for (let i = 0; i <= correctedMessage.length; ++i) track[i][0] = i; + + for (let j = 1; j <= correctedMessage.length; ++j) { + for (let i = 1; i <= template.length; ++i) { + const indicator = template[i - 1] === correctedMessage[j - 1] ? 0 : 1; + track[j][i] = Math.min( + track[j][i - 1] + 1, // delete + track[j - 1][i] + 1, // insert + track[j - 1][i - 1] + indicator, // substitude + ); + } + } + + return track[correctedMessage.length][template.length]; + } + + private static cpdlcMessageClassification(message: string): CpdlcMessageElement | undefined { + const scores: [number, string][] = []; + let minScore = 100000; + + // clear the message from marker, etc. + const clearedMessage = message.replace('@', '').replace('_', ' '); + + // test all uplink messages + for (const ident in CpdlcMessagesUplink) { + if ({}.hasOwnProperty.call(CpdlcMessagesUplink, ident)) { + const data = CpdlcMessagesUplink[ident]; + + if (HoppieConnector.fansMode === FansMode.FansNone || data[1].FansModes.includes(HoppieConnector.fansMode)) { + let minDistance = 100000; + + data[0].forEach((template) => { + const distance = HoppieConnector.levenshteinDistance(template, clearedMessage, data[1].Content); + if (minDistance > distance) minDistance = distance; + }); + + scores.push([minDistance, ident]); + if (minScore > minDistance) minScore = minDistance; + } + } + } + + // get all entries with the minimal score + let matches: string[] = []; + scores.forEach((elem) => { + if (elem[0] === minScore) matches.push(elem[1]); + }); + + console.log(`Found matches: ${matches}, score: ${minScore}`); + if (matches.length === 0) return undefined; + + // check if message without parameters are in, but the minScore not empty + if (matches.length > 1 && minScore !== 0) { + const nonEmpty = matches.filter((match) => CpdlcMessagesUplink[match][1].Content.length !== 0); + if (nonEmpty.length !== 0 && matches.length !== nonEmpty.length) { + console.log(`Ignoring ${matches.length - nonEmpty.length} messages without arguments. Remaining ${nonEmpty}`); + matches = nonEmpty; + } + } + + // check if more than the freetext-entry is valid + if (matches.length > 1) { + const nonFreetext = matches.filter((match) => match !== 'UM169' && match !== 'UM183'); + if (nonFreetext.length !== 0 && matches.length !== nonFreetext.length) { + console.log(`Ignoring ${matches.length - nonFreetext.length} freetext messages. Remaining: ${nonFreetext}`); + matches = nonFreetext; + } + } + + // check if the FANS mode is invalid + if (matches.length > 1 && this.fansMode !== FansMode.FansNone) { + const validFans = matches.filter((match) => CpdlcMessagesUplink[match][1].FansModes.findIndex((elem) => elem === this.fansMode) !== -1); + if (validFans.length !== 0 && matches.length !== validFans.length) { + console.log(`Ignoring ${matches.length - validFans.length} invalid FANS messages. Remaining: ${validFans}`); + matches = validFans; + } + } + + // TODO add some more heuristic about messages + + // create a deep-copy of the message + const retval: CpdlcMessageElement = CpdlcMessagesUplink[matches[0]][1].deepCopy(); + let elements = message.split(' '); + console.log(`Selected UM-ID: ${matches[0]}`); + + // parse the content and store it in the deep copy + retval.Content.forEach((entry) => { + const result = entry.validateAndReplaceContent(elements); + elements = result.remaining; + }); + + return retval; + } + + public static async poll(): Promise<[AtsuStatusCodes, AtsuMessage[]]> { + const retval: AtsuMessage[] = []; + + if (SimVar.GetSimVarValue('L:A32NX_HOPPIE_ACTIVE', 'number') !== 1 || HoppieConnector.flightNumber === '') { + return [AtsuStatusCodes.NoHoppieConnection, retval]; + } + + try { + const body = { + logon: NXDataStore.get('CONFIG_HOPPIE_USERID', ''), + from: HoppieConnector.flightNumber, + to: HoppieConnector.flightNumber, + type: 'poll', + }; + const text = await Hoppie.sendRequest(body).then((resp) => resp.response).catch(() => 'proxy'); + + // proxy error during request + if (text === 'proxy') { + return [AtsuStatusCodes.ProxyError, retval]; + } + + // something went wrong + if (!text.startsWith('ok')) { + return [AtsuStatusCodes.ComFailed, retval]; + } + + // split up the received data into multiple messages + let messages = text.split(/({[\s\S\n]*?})/gm); + messages = messages.filter((elem) => elem !== 'ok' && elem !== 'ok ' && elem !== '} ' && elem !== '}' && elem !== ''); + + // create the messages + messages.forEach((element) => { + // get the single entries of the message + // example: [CALLSIGN telex, {Hello world!}] + const entries = element.substring(1).split(/({[\s\S\n]*?})/gm); + + // get all relevant information + const metadata = entries[0].split(' '); + const sender = metadata[0].toUpperCase(); + const type = metadata[1].toLowerCase(); + const content = entries[1].replace(/{/, '').replace(/}/, '').toUpperCase(); + + switch (type) { + case 'telex': + const freetext = new FreetextMessage(); + freetext.Network = AtsuMessageNetwork.Hoppie; + freetext.Station = sender; + freetext.Direction = AtsuMessageDirection.Uplink; + freetext.ComStatus = AtsuMessageComStatus.Received; + freetext.Message = content; + retval.push(freetext); + break; + case 'cpdlc': + const cpdlc = new CpdlcMessage(); + cpdlc.Station = sender; + cpdlc.Direction = AtsuMessageDirection.Uplink; + cpdlc.ComStatus = AtsuMessageComStatus.Received; + + // split up the data + const elements = content.split('/'); + cpdlc.CurrentTransmissionId = parseInt(elements[2]); + if (elements[3] !== '') { + cpdlc.PreviousTransmissionId = parseInt(elements[3]); + } + cpdlc.Message = elements[5].replace(/@/g, '').replace(/_/g, '\n'); + cpdlc.Content.push(HoppieConnector.cpdlcMessageClassification(cpdlc.Message)); + if ((elements[4] as CpdlcMessageExpectedResponseType) !== cpdlc.Content[0]?.ExpectedResponse) { + cpdlc.Content[0].ExpectedResponse = (elements[4] as CpdlcMessageExpectedResponseType); + } + + retval.push(cpdlc); + break; + default: + break; + } + }); + + return [AtsuStatusCodes.Ok, retval]; + } catch (_err) { + return [AtsuStatusCodes.NoHoppieConnection, []]; + } + } + + public static pollInterval(): number { + // To comply with Hoppie rate limits, we choose a random number between 45 and 75, as recommend by Hoppie. Ref to: https://www.hoppie.nl/acars/system/tech.html + const pollTimeSeconds = Math.floor(Math.random() * (75 - 45 + 1)) + 45; + return pollTimeSeconds * 1000; + } +} \ No newline at end of file diff --git a/hdw-a339x/src/systems/instruments/src/EFB/Settings/Pages/AtsuAocPage.tsx b/hdw-a339x/src/systems/instruments/src/EFB/Settings/Pages/AtsuAocPage.tsx index 3e836b58e..aa3dd7a09 100644 --- a/hdw-a339x/src/systems/instruments/src/EFB/Settings/Pages/AtsuAocPage.tsx +++ b/hdw-a339x/src/systems/instruments/src/EFB/Settings/Pages/AtsuAocPage.tsx @@ -38,7 +38,7 @@ export const AtsuAocPage = () => { const body = { logon: value, - from: 'FBWA32NX', + from: 'HDWA339X', to: 'ALL-CALLSIGNS', type: 'ping', packet: '',