mirror of
https://github.com/l5yth/potato-mesh.git
synced 2026-05-06 13:32:47 +02:00
456 lines
16 KiB
Ruby
456 lines
16 KiB
Ruby
# 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.
|
|
|
|
# frozen_string_literal: true
|
|
|
|
module PotatoMesh
|
|
module App
|
|
module Queries
|
|
MAX_QUERY_LIMIT = 1000
|
|
|
|
# Remove nil or empty values from an API response hash to reduce payload size.
|
|
# Integer keys emitted by SQLite are ignored because the JSON representation
|
|
# only exposes symbolic keys. Strings containing only whitespace are treated
|
|
# as empty to mirror sanitisation elsewhere in the application.
|
|
#
|
|
# @param row [Hash] raw database row to compact.
|
|
# @return [Hash] cleaned hash without blank values.
|
|
def compact_api_row(row)
|
|
return {} unless row.is_a?(Hash)
|
|
|
|
row.each_with_object({}) do |(key, value), acc|
|
|
next if key.is_a?(Integer)
|
|
next if value.nil?
|
|
|
|
if value.is_a?(String)
|
|
trimmed = value.strip
|
|
next if trimmed.empty?
|
|
acc[key] = value
|
|
next
|
|
end
|
|
|
|
next if value.respond_to?(:empty?) && value.empty?
|
|
|
|
acc[key] = value
|
|
end
|
|
end
|
|
|
|
# Normalise a caller-provided limit to a sane, positive integer.
|
|
#
|
|
# @param limit [Object] value coerced to an integer.
|
|
# @param default [Integer] fallback used when coercion fails.
|
|
# @return [Integer] limit clamped between 1 and MAX_QUERY_LIMIT.
|
|
def coerce_query_limit(limit, default: 200)
|
|
coerced = begin
|
|
if limit.is_a?(Integer)
|
|
limit
|
|
else
|
|
Integer(limit, 10)
|
|
end
|
|
rescue ArgumentError, TypeError
|
|
nil
|
|
end
|
|
|
|
coerced = default if coerced.nil? || coerced <= 0
|
|
coerced = MAX_QUERY_LIMIT if coerced > MAX_QUERY_LIMIT
|
|
coerced
|
|
end
|
|
|
|
def node_reference_tokens(node_ref)
|
|
parts = canonical_node_parts(node_ref)
|
|
canonical_id, numeric_id = parts ? parts[0, 2] : [nil, nil]
|
|
|
|
string_values = []
|
|
numeric_values = []
|
|
|
|
case node_ref
|
|
when Integer
|
|
numeric_values << node_ref
|
|
string_values << node_ref.to_s
|
|
when Numeric
|
|
coerced = node_ref.to_i
|
|
numeric_values << coerced
|
|
string_values << coerced.to_s
|
|
when String
|
|
trimmed = node_ref.strip
|
|
unless trimmed.empty?
|
|
string_values << trimmed
|
|
numeric_values << trimmed.to_i if trimmed.match?(/\A-?\d+\z/)
|
|
end
|
|
when nil
|
|
# no-op
|
|
else
|
|
coerced = node_ref.to_s.strip
|
|
string_values << coerced unless coerced.empty?
|
|
end
|
|
|
|
if canonical_id
|
|
string_values << canonical_id
|
|
string_values << canonical_id.upcase
|
|
end
|
|
|
|
if numeric_id
|
|
numeric_values << numeric_id
|
|
string_values << numeric_id.to_s
|
|
end
|
|
|
|
cleaned_strings = string_values.compact.map(&:to_s).map(&:strip).reject(&:empty?).uniq
|
|
cleaned_numbers = numeric_values.compact.map do |value|
|
|
begin
|
|
Integer(value, 10)
|
|
rescue ArgumentError, TypeError
|
|
nil
|
|
end
|
|
end.compact.uniq
|
|
|
|
{
|
|
string_values: cleaned_strings,
|
|
numeric_values: cleaned_numbers,
|
|
}
|
|
end
|
|
|
|
def node_lookup_clause(node_ref, string_columns:, numeric_columns: [])
|
|
tokens = node_reference_tokens(node_ref)
|
|
string_values = tokens[:string_values]
|
|
numeric_values = tokens[:numeric_values]
|
|
|
|
clauses = []
|
|
params = []
|
|
|
|
unless string_columns.empty? || string_values.empty?
|
|
string_columns.each do |column|
|
|
placeholders = Array.new(string_values.length, "?").join(", ")
|
|
clauses << "#{column} IN (#{placeholders})"
|
|
params.concat(string_values)
|
|
end
|
|
end
|
|
|
|
unless numeric_columns.empty? || numeric_values.empty?
|
|
numeric_columns.each do |column|
|
|
placeholders = Array.new(numeric_values.length, "?").join(", ")
|
|
clauses << "#{column} IN (#{placeholders})"
|
|
params.concat(numeric_values)
|
|
end
|
|
end
|
|
|
|
return nil if clauses.empty?
|
|
|
|
["(#{clauses.join(" OR ")})", params]
|
|
end
|
|
|
|
def query_nodes(limit, node_ref: nil)
|
|
limit = coerce_query_limit(limit)
|
|
db = open_database(readonly: true)
|
|
db.results_as_hash = true
|
|
now = Time.now.to_i
|
|
min_last_heard = now - PotatoMesh::Config.week_seconds
|
|
params = []
|
|
where_clauses = []
|
|
|
|
if node_ref
|
|
clause = node_lookup_clause(node_ref, string_columns: ["node_id"], numeric_columns: ["num"])
|
|
return [] unless clause
|
|
where_clauses << clause.first
|
|
params.concat(clause.last)
|
|
else
|
|
where_clauses << "last_heard >= ?"
|
|
params << min_last_heard
|
|
end
|
|
|
|
if private_mode?
|
|
where_clauses << "(role IS NULL OR role <> 'CLIENT_HIDDEN')"
|
|
end
|
|
|
|
sql = <<~SQL
|
|
SELECT node_id, short_name, long_name, hw_model, role, snr,
|
|
battery_level, voltage, last_heard, first_heard,
|
|
uptime_seconds, channel_utilization, air_util_tx,
|
|
position_time, location_source, precision_bits,
|
|
latitude, longitude, altitude, lora_freq, modem_preset
|
|
FROM nodes
|
|
SQL
|
|
sql += " WHERE #{where_clauses.join(" AND ")}\n" if where_clauses.any?
|
|
sql += <<~SQL
|
|
ORDER BY last_heard DESC
|
|
LIMIT ?
|
|
SQL
|
|
params << limit
|
|
|
|
rows = db.execute(sql, params)
|
|
rows = rows.select do |r|
|
|
last_candidate = [r["last_heard"], r["position_time"], r["first_heard"]]
|
|
.map { |value| coerce_integer(value) }
|
|
.compact
|
|
.max
|
|
last_candidate && last_candidate >= min_last_heard
|
|
end
|
|
rows.each do |r|
|
|
r["role"] ||= "CLIENT"
|
|
lh = r["last_heard"]&.to_i
|
|
pt = r["position_time"]&.to_i
|
|
lh = now if lh && lh > now
|
|
pt = nil if pt && pt > now
|
|
r["last_heard"] = lh
|
|
r["position_time"] = pt
|
|
r["last_seen_iso"] = Time.at(lh).utc.iso8601 if lh
|
|
r["pos_time_iso"] = Time.at(pt).utc.iso8601 if pt
|
|
pb = r["precision_bits"]
|
|
r["precision_bits"] = pb.to_i if pb
|
|
end
|
|
rows.map { |row| compact_api_row(row) }
|
|
ensure
|
|
db&.close
|
|
end
|
|
|
|
def query_messages(limit, node_ref: nil)
|
|
limit = coerce_query_limit(limit)
|
|
db = open_database(readonly: true)
|
|
db.results_as_hash = true
|
|
params = []
|
|
where_clauses = ["COALESCE(TRIM(m.encrypted), '') = ''"]
|
|
now = Time.now.to_i
|
|
min_rx_time = now - PotatoMesh::Config.week_seconds
|
|
where_clauses << "m.rx_time >= ?"
|
|
params << min_rx_time
|
|
|
|
if node_ref
|
|
clause = node_lookup_clause(node_ref, string_columns: ["m.from_id", "m.to_id"])
|
|
return [] unless clause
|
|
where_clauses << clause.first
|
|
params.concat(clause.last)
|
|
end
|
|
|
|
sql = <<~SQL
|
|
SELECT m.id, m.rx_time, m.rx_iso, m.from_id, m.to_id, m.channel,
|
|
m.portnum, m.text, m.encrypted, m.rssi, m.hop_limit,
|
|
m.lora_freq, m.modem_preset, m.channel_name, m.snr
|
|
FROM messages m
|
|
SQL
|
|
sql += " WHERE #{where_clauses.join(" AND ")}\n"
|
|
sql += <<~SQL
|
|
ORDER BY m.rx_time DESC
|
|
LIMIT ?
|
|
SQL
|
|
params << limit
|
|
rows = db.execute(sql, params)
|
|
rows.each do |r|
|
|
r.delete_if { |key, _| key.is_a?(Integer) }
|
|
if PotatoMesh::Config.debug? && (r["from_id"].nil? || r["from_id"].to_s.strip.empty?)
|
|
raw = db.execute("SELECT * FROM messages WHERE id = ?", [r["id"]]).first
|
|
debug_log(
|
|
"Message query produced empty sender",
|
|
context: "queries.messages",
|
|
stage: "raw_row",
|
|
row: raw,
|
|
)
|
|
end
|
|
|
|
canonical_from_id = string_or_nil(normalize_node_id(db, r["from_id"]))
|
|
node_id = canonical_from_id || string_or_nil(r["from_id"])
|
|
|
|
if canonical_from_id
|
|
raw_from_id = string_or_nil(r["from_id"])
|
|
if raw_from_id.nil? || raw_from_id.match?(/\A[0-9]+\z/)
|
|
r["from_id"] = canonical_from_id
|
|
elsif raw_from_id.start_with?("!") && raw_from_id.casecmp(canonical_from_id) != 0
|
|
r["from_id"] = canonical_from_id
|
|
end
|
|
end
|
|
|
|
r["node_id"] = node_id if node_id
|
|
|
|
if PotatoMesh::Config.debug? && (r["from_id"].nil? || r["from_id"].to_s.strip.empty?)
|
|
debug_log(
|
|
"Message query produced empty sender",
|
|
context: "queries.messages",
|
|
stage: "after_normalization",
|
|
row: r,
|
|
)
|
|
end
|
|
end
|
|
rows
|
|
ensure
|
|
db&.close
|
|
end
|
|
|
|
def query_positions(limit, node_ref: nil)
|
|
limit = coerce_query_limit(limit)
|
|
db = open_database(readonly: true)
|
|
db.results_as_hash = true
|
|
params = []
|
|
where_clauses = []
|
|
now = Time.now.to_i
|
|
min_rx_time = now - PotatoMesh::Config.week_seconds
|
|
where_clauses << "COALESCE(rx_time, position_time, 0) >= ?"
|
|
params << min_rx_time
|
|
|
|
if node_ref
|
|
clause = node_lookup_clause(node_ref, string_columns: ["node_id"], numeric_columns: ["node_num"])
|
|
return [] unless clause
|
|
where_clauses << clause.first
|
|
params.concat(clause.last)
|
|
end
|
|
|
|
sql = <<~SQL
|
|
SELECT * FROM positions
|
|
SQL
|
|
sql += " WHERE #{where_clauses.join(" AND ")}\n" if where_clauses.any?
|
|
sql += <<~SQL
|
|
ORDER BY rx_time DESC
|
|
LIMIT ?
|
|
SQL
|
|
params << limit
|
|
rows = db.execute(sql, params)
|
|
rows.each do |r|
|
|
rx_time = coerce_integer(r["rx_time"])
|
|
r["rx_time"] = rx_time if rx_time
|
|
r["rx_iso"] = Time.at(rx_time).utc.iso8601 if rx_time && string_or_nil(r["rx_iso"]).nil?
|
|
|
|
node_num = coerce_integer(r["node_num"])
|
|
r["node_num"] = node_num if node_num
|
|
|
|
position_time = coerce_integer(r["position_time"])
|
|
position_time = nil if position_time && position_time > now
|
|
r["position_time"] = position_time
|
|
r["position_time_iso"] = Time.at(position_time).utc.iso8601 if position_time
|
|
|
|
r["precision_bits"] = coerce_integer(r["precision_bits"])
|
|
r["sats_in_view"] = coerce_integer(r["sats_in_view"])
|
|
r["pdop"] = coerce_float(r["pdop"])
|
|
r["snr"] = coerce_float(r["snr"])
|
|
end
|
|
rows.map { |row| compact_api_row(row) }
|
|
ensure
|
|
db&.close
|
|
end
|
|
|
|
def query_neighbors(limit, node_ref: nil)
|
|
limit = coerce_query_limit(limit)
|
|
db = open_database(readonly: true)
|
|
db.results_as_hash = true
|
|
params = []
|
|
where_clauses = []
|
|
now = Time.now.to_i
|
|
min_rx_time = now - PotatoMesh::Config.week_seconds
|
|
where_clauses << "COALESCE(rx_time, 0) >= ?"
|
|
params << min_rx_time
|
|
|
|
if node_ref
|
|
clause = node_lookup_clause(node_ref, string_columns: ["node_id", "neighbor_id"])
|
|
return [] unless clause
|
|
where_clauses << clause.first
|
|
params.concat(clause.last)
|
|
end
|
|
|
|
sql = <<~SQL
|
|
SELECT * FROM neighbors
|
|
SQL
|
|
sql += " WHERE #{where_clauses.join(" AND ")}\n" if where_clauses.any?
|
|
sql += <<~SQL
|
|
ORDER BY rx_time DESC
|
|
LIMIT ?
|
|
SQL
|
|
params << limit
|
|
rows = db.execute(sql, params)
|
|
rows.each do |r|
|
|
rx_time = coerce_integer(r["rx_time"])
|
|
rx_time = now if rx_time && rx_time > now
|
|
r["rx_time"] = rx_time if rx_time
|
|
r["rx_iso"] = Time.at(rx_time).utc.iso8601 if rx_time
|
|
r["snr"] = coerce_float(r["snr"])
|
|
end
|
|
rows.map { |row| compact_api_row(row) }
|
|
ensure
|
|
db&.close
|
|
end
|
|
|
|
def query_telemetry(limit, node_ref: nil)
|
|
limit = coerce_query_limit(limit)
|
|
db = open_database(readonly: true)
|
|
db.results_as_hash = true
|
|
params = []
|
|
where_clauses = []
|
|
now = Time.now.to_i
|
|
min_rx_time = now - PotatoMesh::Config.week_seconds
|
|
where_clauses << "COALESCE(rx_time, telemetry_time, 0) >= ?"
|
|
params << min_rx_time
|
|
|
|
if node_ref
|
|
clause = node_lookup_clause(node_ref, string_columns: ["node_id"], numeric_columns: ["node_num"])
|
|
return [] unless clause
|
|
where_clauses << clause.first
|
|
params.concat(clause.last)
|
|
end
|
|
|
|
sql = <<~SQL
|
|
SELECT * FROM telemetry
|
|
SQL
|
|
sql += " WHERE #{where_clauses.join(" AND ")}\n" if where_clauses.any?
|
|
sql += <<~SQL
|
|
ORDER BY rx_time DESC
|
|
LIMIT ?
|
|
SQL
|
|
params << limit
|
|
rows = db.execute(sql, params)
|
|
rows.each do |r|
|
|
rx_time = coerce_integer(r["rx_time"])
|
|
r["rx_time"] = rx_time if rx_time
|
|
r["rx_iso"] = Time.at(rx_time).utc.iso8601 if rx_time && string_or_nil(r["rx_iso"]).nil?
|
|
|
|
node_num = coerce_integer(r["node_num"])
|
|
r["node_num"] = node_num if node_num
|
|
|
|
telemetry_time = coerce_integer(r["telemetry_time"])
|
|
telemetry_time = nil if telemetry_time && telemetry_time > now
|
|
r["telemetry_time"] = telemetry_time
|
|
r["telemetry_time_iso"] = Time.at(telemetry_time).utc.iso8601 if telemetry_time
|
|
|
|
r["channel"] = coerce_integer(r["channel"])
|
|
r["hop_limit"] = coerce_integer(r["hop_limit"])
|
|
r["rssi"] = coerce_integer(r["rssi"])
|
|
r["bitfield"] = coerce_integer(r["bitfield"])
|
|
r["snr"] = coerce_float(r["snr"])
|
|
r["battery_level"] = coerce_float(r["battery_level"])
|
|
r["voltage"] = coerce_float(r["voltage"])
|
|
r["channel_utilization"] = coerce_float(r["channel_utilization"])
|
|
r["air_util_tx"] = coerce_float(r["air_util_tx"])
|
|
r["uptime_seconds"] = coerce_integer(r["uptime_seconds"])
|
|
r["temperature"] = coerce_float(r["temperature"])
|
|
r["relative_humidity"] = coerce_float(r["relative_humidity"])
|
|
r["barometric_pressure"] = coerce_float(r["barometric_pressure"])
|
|
r["gas_resistance"] = coerce_float(r["gas_resistance"])
|
|
r["current"] = coerce_float(r["current"])
|
|
r["iaq"] = coerce_integer(r["iaq"])
|
|
r["distance"] = coerce_float(r["distance"])
|
|
r["lux"] = coerce_float(r["lux"])
|
|
r["white_lux"] = coerce_float(r["white_lux"])
|
|
r["ir_lux"] = coerce_float(r["ir_lux"])
|
|
r["uv_lux"] = coerce_float(r["uv_lux"])
|
|
r["wind_direction"] = coerce_integer(r["wind_direction"])
|
|
r["wind_speed"] = coerce_float(r["wind_speed"])
|
|
r["weight"] = coerce_float(r["weight"])
|
|
r["wind_gust"] = coerce_float(r["wind_gust"])
|
|
r["wind_lull"] = coerce_float(r["wind_lull"])
|
|
r["radiation"] = coerce_float(r["radiation"])
|
|
r["rainfall_1h"] = coerce_float(r["rainfall_1h"])
|
|
r["rainfall_24h"] = coerce_float(r["rainfall_24h"])
|
|
r["soil_moisture"] = coerce_integer(r["soil_moisture"])
|
|
r["soil_temperature"] = coerce_float(r["soil_temperature"])
|
|
end
|
|
rows.map { |row| compact_api_row(row) }
|
|
ensure
|
|
db&.close
|
|
end
|
|
end
|
|
end
|
|
end
|