UNPKG

signalk-barograph

Version:

SignalK plugin to influx environment data & Barograph to visualize atmospheric pressure

320 lines (301 loc) 14.2 kB
/* Copyright © 2024 Inspired Technologies GmbH. Rights Reserved. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ const { InfluxDB, Point } = require('@influxdata/influxdb-client') const { HealthAPI } = require('@influxdata/influxdb-client-apis') const { DateTime } = require('luxon') const cache = require('./cache') let cacheBuffer = [] function login(clientOptions, log) { try { const influxDB = new InfluxDB(clientOptions) log ("Influx Login successful") return influxDB } catch (err) { log ("Error logging into influx: "+err) } } async function health (influxDB, log, callback) { log("Determining influx health") const healthAPI = new HealthAPI(influxDB) await healthAPI .getHealth() .then((result /* : HealthCheck */) => { log('Influx healthCheck: ' + (result.status === 'pass' ? 'OK' : 'NOT OK')) return callback(influxDB, result) }) .catch(error => { log("HealthCheck Error: "+error) return false }) } function config(root, interval) { if (interval<1000) interval = 1000 switch (root) { case 'environment': return [ { path: 'environment.forecast.time', period: 10*interval, policy: "fixed" }, { path: 'environment.sunlight.times.sunrise', period:900*interval, policy:"fixed", convert: 'dt|>s', config: 'sunlight.times|>forecast.time' }, { path: 'environment.sunlight.times.sunset', period:900*interval, policy:"fixed", convert: 'dt|>s', config: 'sunlight.times|>forecast.time' }, { path: 'environment.inside.temperature', period: 60*interval, policy: "instant", minPeriod: interval }, { path: 'environment.outside.temperature', period: 60*interval, policy: "instant", minPeriod: interval, trend: "temperature" }, { path: 'environment.water.temperature', period: 60*interval, policy: "instant", minPeriod: interval }, { path: 'environment.forecast.temperature', period: 900*interval, policy: "fixed" }, { path: 'environment.forecast.temperature.minimum', period: 900*interval, policy: "fixed" }, { path: 'environment.forecast.temperature.maximum', period: 900*interval, policy: "fixed" }, { path: 'environment.forecast.temperature.feelslike', period: 900*interval, policy: "fixed" }, { path: 'environment.inside.pressure', period: 60*interval, policy: "instant", minPeriod: interval }, { path: 'environment.outside.pressure', period: 60*interval, policy: "instant", minPeriod: interval, trend: "pressure" }, { path: 'environment.forecast.pressure', period: 900*interval, policy: "instant", minPeriod: interval, }, { path: 'environment.inside.humidity', period: 60*interval, policy: "instant", minPeriod: interval }, { path: 'environment.outside.humidity', period: 60*interval, policy: "instant", minPeriod: interval }, { path: 'environment.forecast.relativeHumidity', period: 900*interval, policy: "instant", minPeriod: interval, config: 'relativeHumidity|>humidity' }, { path: 'environment.forecast.description', period: 900*interval, policy:"fixed" }, { path: 'environment.forecast.wind.direction', period: 900*interval, policy:"fixed" }, { path: 'environment.forecast.wind.speed', period: 900*interval, policy:"fixed" }, { path: 'environment.forecast.wind.gust', period: 900*interval, policy:"fixed" }, { path: 'environment.forecast.weather.visibility', period: 900*interval, policy:"fixed" }, { path: 'environment.forecast.weather.clouds', period: 900*interval, policy:"fixed" }, { path: 'environment.forecast.weather.uvindex', period: 900*interval, policy:"fixed" }, { path: 'environment.forecast.weather.icon', period: 900*interval, policy:"fixed" }, { path: 'environment.forecast.weather.code', period: 900*interval, policy:"fixed" }, { path: 'environment.wind.directionTrue', period: 60*interval, policy: "fixed", trend: "winddir", config: "wind.directionTrue|>outside.wind.direction" }, { path: 'environment.outside.wind.direction', period:60*interval, policy :"instant", trend: "winddir" }, { path: 'environment.outside.wind.speed', period:60*interval, policy :"instant"}, { path: 'environment.outside.wind.gust', period:60*interval, policy :"instant"} ]; case 'navigation': return [ { path: 'navigation.gnss.antennaAltitude', period:60*interval, policy: 'fixed', trend:'altitude' }, { path: 'navigation.position', period:60*interval, policy: 'fixed', trend: 'position' } ] default: return [] } } function buffer(metrics) { cacheBuffer = metrics } function post (influxdb, metrics, config, log) { // [Required] Organization | Empty for 1.8.x // [Required] Bucket | Database/Retention Policy // Precision of timestamp. [`ns`, `us`, `ms`, `s`]. The default would be `ns` for other data const writeAPI = influxdb.getWriteApi(config.organization, config.write, 'ms') // TODO: setup default tags for all writes through this API writeAPI.useDefaultTags({id: config.id}) // write point with the appropriate (client-side) timestamp // log(JSON.stringify(metrics)) let measurements = {} for (i=0; i<metrics.length; i++) { writeAPI.writePoint(metrics[i]) measurements[metrics[i].name] = (measurements[metrics[i].name] ? measurements[metrics[i].name]+1 : 1) // log(`${i+1}: ${metrics[i].toLineProtocol(writeAPI)}`) } writeAPI .close() .then(() => { log(measurements) cacheResult = cache.load(config.cacheDir, log) if (cacheResult === false) { return } else { let cached = cache.send(cacheResult, config.cacheDir) log('Sending '+cached.length+' cached data points to be uploaded to influx') let points = [] cached.forEach(p => { let point = new Point(p.name) .tag(Object.keys(p.tags)[0], p.tags[Object.keys(p.tags)[0]]) .timestamp(p.timestamp) if (typeof p.fields[Object.keys(p.fields)[0]]==='float' || parseFloat(p.fields[Object.keys(p.fields)[0]]).toString()!=='NaN') point.floatField(Object.keys(p.fields)[0], parseFloat(p.fields[Object.keys(p.fields)[0]])) else point.stringField(Object.keys(p.fields)[0], p.fields[Object.keys(p.fields)[0]]) points.push(point) }) // log(JSON.stringify(points)) post(influxdb, points, config, log) } }) .catch(err => { // Handle errors cache.push(cacheBuffer, config.cacheDir, log) cacheBuffer = [] log(`Caching metrics because ${err.message}`); const cacheResult = cache.load(config.cacheDir, log) if (cacheResult !== false) { log(`${cacheResult.length} files cached`) } }) } function format (path, values, timestamp, skSource) { if (values === null){ values = 0 } //Set variables for metric let point = null // Get correct measurement based on path based on path config const skPath = path.split('.') switch (skPath.length) { case 6: // extended - use double tagging switch (typeof values) { case 'string': point = new Point(skPath[4]) .tag(skPath[0], skPath[1]) .tag(skPath[2], skPath[3]) .stringField(skPath[5], values) break; case 'object': point = new Point(skPath[4]) .tag(skPath[0], skPath[1]) .tag(skPath[2], skPath[3]) .stringField(skPath[5], JSON.stringify(values)) break; case 'boolean': point = new Point(skPath[4]) .tag(skPath[0], skPath[1]) .tag(skPath[2], skPath[3]) .booleanField(skPath[5], values) break; default: point = new Point(skPath[4]) .tag(skPath[0], skPath[1]) .tag(skPath[2], skPath[3]) .floatField(skPath[5], values) break; } break; case 5: // extended - use double tagging switch (typeof values) { case 'string': point = new Point(skPath[3]) .tag(skPath[0], skPath[1]) .tag(skPath[1], skPath[2]) .stringField(skPath[4], values) break; case 'object': point = new Point(skPath[3]) .tag(skPath[0], skPath[1]) .tag(skPath[1], skPath[2]) .stringField(skPath[4], JSON.stringify(values)) break; case 'boolean': point = new Point(skPath[3]) .tag(skPath[0], skPath[1]) .tag(skPath[1], skPath[2]) .booleanField(skPath[4], values) break; default: point = new Point(skPath[3]) .tag(skPath[0], skPath[1]) .tag(skPath[1], skPath[2]) .floatField(skPath[4], values) break; } break; case 4: // default switch (typeof values) { case 'string': point = new Point(skPath[2]) .tag(skPath[0], skPath[1]) .stringField(skPath[3], values) break; case 'object': point = new Point(skPath[2]) .tag(skPath[0], skPath[1]) .stringField(skPath[3], JSON.stringify(values)) break; case 'boolean': point = new Point(skPath[2]) .tag(skPath[0], skPath[1]) .booleanField(skPath[3], values) break; default: point = new Point(skPath[2]) .tag(skPath[0], skPath[1]) .floatField(skPath[3], values) break; } break; case 3: // default switch (typeof values) { case 'string': point = new Point(skPath[2]) .tag(skPath[0], skPath[1]) .stringField('value', values) break; case 'object': point = new Point(skPath[2]) .tag(skPath[0], skPath[1]) .stringField('value', JSON.stringify(values)) break; case 'boolean': point = new Point(skPath[2]) .tag(skPath[0], skPath[1]) .booleanField('value', values) break; default: point = new Point(skPath[2]) .tag(skPath[0], skPath[1]) .floatField('value', values) break; } break; case 2: // to be verified switch (typeof values) { case 'string': point = new Point(skPath[1]) .tag(skPath[0], '') .stringField('value', values) break; case 'object': point = new Point(skPath[1]) .tag(skPath[0], '') .stringField('value', JSON.stringify(values)) break; case 'boolean': point = new Point(skPath[1]) .tag(skPath[0], '') .booleanField('value', values) break; default: point = new Point(skPath[1]) .tag(skPath[0], '') .floatField('value', values) break; } break; case 1: default: // invalid fields = null break; } if (skSource && skSource!=='') point.tag('source', skSource) point.timestamp = (timestamp ? timestamp.toMillis() : DateTime.utc().toMillis()) return point } module.exports = { login, // login to InfluxDB health, // check InfluxDB health config, // create default configuration buffer, // load cache if post can be complete post, // post to InfluxDB format // format measurement before sending }