173 lines
5.8 KiB
JavaScript
Raw Normal View History

2021-09-27 21:52:27 +02:00
let net = require('net')
let xml2js = require('xml2js');
// @see https://www.vmix.com/help20/index.htm?TCPAPI.html
class VmixConnector {
constructor(ip, port, communicator) {
this.ip = ip
this.port = port
this.communicator = communicator
this.client
this.wasHelloReceived = false
this.wasSubcribeOkReceived = false
this.intervalHandle
this.xmlQueryInterval = 5000
}
connect() {
const client = new net.Socket()
this.client = client
const connectClient = () => {
this.wasHelloReceived = false
this.wasSubcribeOkReceived = false
console.log(this.port, this.ip)
client.connect(this.port, this.ip)
}
const queryXml = () => {
if(!client.connecting && !client.pending && !client.destroyed) {
client.write("XML\r\n")
}
}
connectClient()
client.on("connect", () => {
console.log('Connected to vMix')
this.communicator.notifyMixerIsConnected()
})
client.on("ready", () => {
client.write("SUBSCRIBE TALLY\r\n")
// @TODO: we need to poll for new channels or renames. Is there a way to subscribe to those?
this.intervalHandle = setInterval(queryXml, this.xmlQueryInterval)
queryXml()
})
client.on("timeout", () => {
console.error("Connection to vMix timed out")
})
client.on("error", error => {
console.error(`${error.name}: ${error.message}`)
})
client.on('data', this.onData.bind(this))
client.on('close', (hadError) => {
this.communicator.notifyMixerIsDisconnected()
console.log("Connection to vMix closed")
if(this.intervalHandle) {
clearInterval(this.intervalHandle);
this.intervalHandle = undefined;
}
if (hadError) {
console.debug("Connection to vMix is reconnected after an error")
setTimeout(() => {
client.connect(this.port, this.ip)
}, 200)
}
})
}
onData(data) {
data.toString().replace(/[\r\n]*$/, "").split("\r\n").forEach(command => {
console.debug(`> ${command}`)
if (command.startsWith("VERSION OK")) {
this.wasHelloReceived = true
} else if (command.startsWith("SUBSCRIBE OK TALLY")) {
this.wasSubcribeOkReceived = true
} else if (command.startsWith("TALLY OK")) {
this.handleTallyCommand(command)
} else if (command.startsWith("XML ")) {
// @TODO: it would be better to detect the "XML" response itself, not the payload
} else if (command.startsWith("<vmix>")) {
this.handleXmlCommand(command)
} else {
console.debug("Ignoring unkown command from vmix")
}
}, this)
}
handleTallyCommand(command) {
const result = command.match(/^TALLY OK (\d*)$/)
if (result === null) {
console.error("Tally OK command was ill formed")
} else {
const [_, state] = result
let programs = []
let previews = []
// vMix encodes tally states as numbers:
// @see https://www.vmix.com/help20/index.htm?TCPAPI.html
// 0 = off
// 1 = program
// 2 = preview
state.split('').forEach((val, idx) => {
if (val === "1") {
programs.push(idx + 1)
} else if (val === "2") {
previews.push(idx + 1)
}
})
this.communicator.notifyProgramChanged(programs, previews)
}
}
handleXmlCommand(command) {
xml2js.parseString(command, (error, result) => {
if (error) {
console.error(`Error parsing XML response from vMix: ${error}`)
} else {
const inputs = (result.vmix || {}).inputs
if(inputs === undefined) {
console.log("XML from vMix looks faulty. Could not find inputs.")
} else {
const count = inputs[0].input.length
const names = inputs[0].input.reduce((map, input, idx) => {
map[idx+1] = input.$.shortTitle
return map
}, {})
this.communicator.notifyChannels(count, names)
}
}
})
}
disconnect() {
this.wasHelloReceived = false
this.wasSubcribeOkReceived = false
const promise = new Promise(resolve => {
if(this.intervalHandle) {
clearInterval(this.intervalHandle);
this.intervalHandle = undefined;
}
if (this.client && ! this.client.destroyed) {
if (this.client.isConnected) {
// if we are connected: try to be nice
this.client.end(() => {
console.log("Disconnected from vMix")
resolve()
})
} else {
// if not: be rude
this.client.destroy()
resolve()
}
} else {
resolve()
}
})
this.client = null
return promise
}
isConnected() {
return this.client && !this.client.destroyed && this.wasHelloReceived && this.wasSubcribeOkReceived
}
}
VmixConnector.ID = "vmix"
VmixConnector.defaultIp = "127.0.0.1"
VmixConnector.defaultPort = 8099
module.exports = VmixConnector;