mirror of
https://github.com/l5yth/potato-mesh.git
synced 2026-07-04 17:01:25 +02:00
522213c040
* Restore modular app functionality * Fix federation thread settings and add coverage * Use Sinatra set for federation threads * Restore 41447 as default web port
2544 lines
88 KiB
Ruby
2544 lines
88 KiB
Ruby
# Copyright (C) 2025 l5yth
|
|
#
|
|
# 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
|
|
|
|
require "spec_helper"
|
|
require "sqlite3"
|
|
require "json"
|
|
require "time"
|
|
require "base64"
|
|
require "uri"
|
|
|
|
RSpec.describe "Potato Mesh Sinatra app" do
|
|
let(:app) { Sinatra::Application }
|
|
let(:application_class) { PotatoMesh::Application }
|
|
|
|
describe "configuration" do
|
|
it "sets the default HTTP port to 41_447" do
|
|
expect(app.settings.port).to eq(41_447)
|
|
end
|
|
end
|
|
|
|
describe ".resolve_port" do
|
|
around do |example|
|
|
original_present = ENV.key?("PORT")
|
|
original_value = ENV["PORT"] if original_present
|
|
ENV.delete("PORT")
|
|
example.run
|
|
ensure
|
|
if original_present
|
|
ENV["PORT"] = original_value
|
|
else
|
|
ENV.delete("PORT")
|
|
end
|
|
end
|
|
|
|
it "returns the default port when the environment is unset" do
|
|
expect(application_class.resolve_port).to eq(41_447)
|
|
end
|
|
|
|
it "parses the environment override when provided" do
|
|
ENV["PORT"] = "51515"
|
|
expect(application_class.resolve_port).to eq(51_515)
|
|
end
|
|
|
|
it "falls back to the default port when parsing fails" do
|
|
ENV["PORT"] = "potato"
|
|
expect(application_class.resolve_port).to eq(41_447)
|
|
end
|
|
end
|
|
|
|
# Return the absolute filesystem path to the requested fixture.
|
|
#
|
|
# @param name [String] fixture filename relative to the tests directory.
|
|
# @return [String] absolute path to the fixture file.
|
|
def fixture_path(name)
|
|
File.expand_path("../../tests/#{name}", __dir__)
|
|
end
|
|
|
|
# Execute the provided block with a configured SQLite connection.
|
|
#
|
|
# @param readonly [Boolean] whether to open the database in read-only mode.
|
|
# @yieldparam db [SQLite3::Database] open database handle.
|
|
# @return [void]
|
|
def with_db(readonly: false)
|
|
db = SQLite3::Database.new(PotatoMesh::Config.db_path, readonly: readonly)
|
|
db.busy_timeout = PotatoMesh::Config.db_busy_timeout_ms
|
|
db.execute("PRAGMA foreign_keys = ON")
|
|
yield db
|
|
ensure
|
|
db&.close
|
|
end
|
|
|
|
# Remove all rows from the tables used by the application under test.
|
|
#
|
|
# @return [void]
|
|
def clear_database
|
|
with_db do |db|
|
|
db.execute("DELETE FROM instances")
|
|
db.execute("DELETE FROM neighbors")
|
|
db.execute("DELETE FROM messages")
|
|
db.execute("DELETE FROM nodes")
|
|
db.execute("DELETE FROM positions")
|
|
db.execute("DELETE FROM telemetry")
|
|
end
|
|
ensure_self_instance_record!
|
|
end
|
|
|
|
# Build a hash excluding entries whose values are nil.
|
|
#
|
|
# @param hash [Hash] collection filtered for nil values.
|
|
# @return [Hash] hash containing only keys with non-nil values.
|
|
def reject_nil_values(hash)
|
|
hash.reject { |_, value| value.nil? }
|
|
end
|
|
|
|
# Construct a request payload mirroring the structure produced by the daemon.
|
|
#
|
|
# @param node [Hash] node attributes from the fixture dataset.
|
|
# @return [Hash] payload formatted for the API.
|
|
def build_node_payload(node)
|
|
payload = {
|
|
"user" => reject_nil_values(
|
|
"shortName" => node["short_name"],
|
|
"longName" => node["long_name"],
|
|
"hwModel" => node["hw_model"],
|
|
"role" => node["role"],
|
|
),
|
|
"hwModel" => node["hw_model"],
|
|
"lastHeard" => node["last_heard"],
|
|
"snr" => node["snr"],
|
|
}
|
|
|
|
metrics = reject_nil_values(
|
|
"batteryLevel" => node["battery_level"],
|
|
"voltage" => node["voltage"],
|
|
"channelUtilization" => node["channel_utilization"],
|
|
"airUtilTx" => node["air_util_tx"],
|
|
"uptimeSeconds" => node["uptime_seconds"],
|
|
)
|
|
payload["deviceMetrics"] = metrics unless metrics.empty?
|
|
|
|
position = reject_nil_values(
|
|
"time" => node["position_time"],
|
|
"latitude" => node["latitude"],
|
|
"longitude" => node["longitude"],
|
|
"altitude" => node["altitude"],
|
|
"locationSource" => node["location_source"],
|
|
"precisionBits" => node["precision_bits"],
|
|
)
|
|
payload["position"] = position unless position.empty?
|
|
|
|
payload
|
|
end
|
|
|
|
# Determine the expected last heard timestamp for a node fixture.
|
|
#
|
|
# @param node [Hash] node attributes from the fixture dataset.
|
|
# @return [Integer, nil] canonical last heard timestamp.
|
|
def expected_last_heard(node)
|
|
[node["last_heard"], node["position_time"]].compact.max
|
|
end
|
|
|
|
# Assemble the expected row persisted in the nodes table.
|
|
#
|
|
# @param node [Hash] node attributes from the fixture dataset.
|
|
# @return [Hash] expected database row for assertions.
|
|
def expected_node_row(node)
|
|
final_last = expected_last_heard(node)
|
|
{
|
|
"node_id" => node["node_id"],
|
|
"short_name" => node["short_name"],
|
|
"long_name" => node["long_name"],
|
|
"hw_model" => node["hw_model"],
|
|
"role" => node["role"] || "CLIENT",
|
|
"snr" => node["snr"],
|
|
"battery_level" => node["battery_level"],
|
|
"voltage" => node["voltage"],
|
|
"last_heard" => final_last,
|
|
"first_heard" => final_last,
|
|
"uptime_seconds" => node["uptime_seconds"],
|
|
"channel_utilization" => node["channel_utilization"],
|
|
"air_util_tx" => node["air_util_tx"],
|
|
"position_time" => node["position_time"],
|
|
"location_source" => node["location_source"],
|
|
"precision_bits" => node["precision_bits"],
|
|
"latitude" => node["latitude"],
|
|
"longitude" => node["longitude"],
|
|
"altitude" => node["altitude"],
|
|
}
|
|
end
|
|
|
|
# Assert equality while supporting tolerance for floating point comparisons.
|
|
#
|
|
# @param actual [Object] observed value.
|
|
# @param expected [Object] expected value.
|
|
# @param tolerance [Float] acceptable delta for floating point values.
|
|
# @return [void]
|
|
def expect_same_value(actual, expected, tolerance: 1e-6)
|
|
if expected.nil?
|
|
expect(actual).to be_nil
|
|
elsif expected.is_a?(Float)
|
|
expect(actual).to be_within(tolerance).of(expected)
|
|
else
|
|
expect(actual).to eq(expected)
|
|
end
|
|
end
|
|
|
|
# Import all nodes defined in the fixture file via the HTTP API.
|
|
#
|
|
# @return [void]
|
|
def import_nodes_fixture
|
|
nodes_fixture.each do |node|
|
|
payload = { node["node_id"] => build_node_payload(node) }
|
|
post "/api/nodes", payload.to_json, auth_headers
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
end
|
|
end
|
|
|
|
# Import all messages defined in the fixture file via the HTTP API.
|
|
#
|
|
# @return [void]
|
|
def import_messages_fixture
|
|
messages_fixture.each do |message|
|
|
payload = message.reject { |key, _| key == "node" }
|
|
post "/api/messages", payload.to_json, auth_headers
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
end
|
|
end
|
|
|
|
let(:api_token) { "spec-token" }
|
|
let(:auth_headers) do
|
|
{
|
|
"CONTENT_TYPE" => "application/json",
|
|
"HTTP_AUTHORIZATION" => "Bearer #{api_token}",
|
|
}
|
|
end
|
|
let(:nodes_fixture) { JSON.parse(File.read(fixture_path("nodes.json"))) }
|
|
let(:messages_fixture) { JSON.parse(File.read(fixture_path("messages.json"))) }
|
|
let(:telemetry_fixture) { JSON.parse(File.read(fixture_path("telemetry.json"))) }
|
|
let(:reference_time) do
|
|
latest = nodes_fixture.map { |node| node["last_heard"] }.compact.max
|
|
Time.at((latest || Time.now.to_i) + 1000)
|
|
end
|
|
|
|
describe "federation announcers" do
|
|
class DummyThread
|
|
attr_accessor :name, :report_on_exception, :block
|
|
|
|
def alive?
|
|
false
|
|
end
|
|
end
|
|
|
|
let(:dummy_thread) { DummyThread.new }
|
|
|
|
before do
|
|
app.set(:initial_federation_thread, nil)
|
|
app.set(:federation_thread, nil)
|
|
end
|
|
|
|
it "stores and clears the initial federation thread" do
|
|
allow(app).to receive(:announce_instance_to_all_domains)
|
|
allow(Thread).to receive(:new) do |&block|
|
|
dummy_thread.block = block
|
|
dummy_thread
|
|
end
|
|
|
|
result = app.start_initial_federation_announcement!
|
|
|
|
expect(result).to be(dummy_thread)
|
|
expect(app.settings.initial_federation_thread).to be(dummy_thread)
|
|
expect(dummy_thread.block).not_to be_nil
|
|
|
|
expect { dummy_thread.block.call }.to change {
|
|
app.settings.initial_federation_thread
|
|
}.from(dummy_thread).to(nil)
|
|
end
|
|
|
|
it "stores the recurring federation announcer thread" do
|
|
allow(Thread).to receive(:new) do |&block|
|
|
dummy_thread.block = block
|
|
dummy_thread
|
|
end
|
|
|
|
result = app.start_federation_announcer!
|
|
|
|
expect(result).to be(dummy_thread)
|
|
expect(app.settings.federation_thread).to be(dummy_thread)
|
|
end
|
|
end
|
|
|
|
before do
|
|
@original_token = ENV["API_TOKEN"]
|
|
@original_private = ENV["PRIVATE"]
|
|
ENV["API_TOKEN"] = api_token
|
|
ENV.delete("PRIVATE")
|
|
allow(Time).to receive(:now).and_return(reference_time)
|
|
clear_database
|
|
end
|
|
|
|
after do
|
|
ENV["API_TOKEN"] = @original_token
|
|
if @original_private.nil?
|
|
ENV.delete("PRIVATE")
|
|
else
|
|
ENV["PRIVATE"] = @original_private
|
|
end
|
|
end
|
|
|
|
describe "helper utilities" do
|
|
describe "#fetch_config_string" do
|
|
around do |example|
|
|
key = "SPEC_FETCH"
|
|
original = ENV[key]
|
|
begin
|
|
ENV.delete(key)
|
|
example.run
|
|
ensure
|
|
if original.nil?
|
|
ENV.delete(key)
|
|
else
|
|
ENV[key] = original
|
|
end
|
|
end
|
|
end
|
|
|
|
it "returns the default when the environment variable is missing" do
|
|
expect(fetch_config_string("SPEC_FETCH", "fallback")).to eq("fallback")
|
|
end
|
|
|
|
it "strips whitespace and rejects blank overrides" do
|
|
ENV["SPEC_FETCH"] = " \t "
|
|
expect(fetch_config_string("SPEC_FETCH", "fallback")).to eq("fallback")
|
|
|
|
ENV["SPEC_FETCH"] = " override "
|
|
expect(fetch_config_string("SPEC_FETCH", "fallback")).to eq("override")
|
|
end
|
|
end
|
|
|
|
describe "#determine_instance_domain" do
|
|
around do |example|
|
|
original = ENV["INSTANCE_DOMAIN"]
|
|
begin
|
|
ENV.delete("INSTANCE_DOMAIN")
|
|
example.run
|
|
ensure
|
|
if original.nil?
|
|
ENV.delete("INSTANCE_DOMAIN")
|
|
else
|
|
ENV["INSTANCE_DOMAIN"] = original
|
|
end
|
|
end
|
|
end
|
|
|
|
it "uses the environment override when provided" do
|
|
ENV["INSTANCE_DOMAIN"] = " example.org "
|
|
|
|
domain, source = determine_instance_domain
|
|
|
|
expect(domain).to eq("example.org")
|
|
expect(source).to eq(:environment)
|
|
end
|
|
|
|
it "normalises scheme-based environment overrides" do
|
|
ENV["INSTANCE_DOMAIN"] = " https://Example.Org "
|
|
|
|
domain, source = determine_instance_domain
|
|
|
|
expect(domain).to eq("example.org")
|
|
expect(source).to eq(:environment)
|
|
end
|
|
|
|
it "allows IP addresses configured via the environment" do
|
|
ENV["INSTANCE_DOMAIN"] = "http://127.0.0.1"
|
|
|
|
domain, source = determine_instance_domain
|
|
|
|
expect(domain).to eq("127.0.0.1")
|
|
expect(source).to eq(:environment)
|
|
end
|
|
|
|
it "rejects instance domains containing path components" do
|
|
ENV["INSTANCE_DOMAIN"] = "https://example.org/app"
|
|
|
|
expect { determine_instance_domain }.to raise_error(
|
|
RuntimeError,
|
|
/must not include a path component/,
|
|
)
|
|
end
|
|
|
|
it "falls back to reverse DNS when available" do
|
|
address = Addrinfo.ip("203.0.113.10")
|
|
allow(Socket).to receive(:ip_address_list).and_return([address])
|
|
allow(Resolv).to receive(:getname).with("203.0.113.10").and_return("chara.htznr.fault.dev")
|
|
|
|
domain, source = determine_instance_domain
|
|
|
|
expect(domain).to eq("chara.htznr.fault.dev")
|
|
expect(source).to eq(:reverse_dns)
|
|
end
|
|
|
|
it "falls back to a public IP address when reverse DNS is unavailable" do
|
|
public_address = Addrinfo.ip("203.0.113.20")
|
|
allow(Socket).to receive(:ip_address_list).and_return([public_address])
|
|
allow(Resolv).to receive(:getname).and_raise(Resolv::ResolvError)
|
|
|
|
domain, source = determine_instance_domain
|
|
|
|
expect(domain).to eq("203.0.113.20")
|
|
expect(source).to eq(:public_ip)
|
|
end
|
|
|
|
it "falls back to a protected IP address when only private networks exist" do
|
|
private_address = Addrinfo.ip("10.0.0.5")
|
|
allow(Socket).to receive(:ip_address_list).and_return([private_address])
|
|
allow(Resolv).to receive(:getname).and_raise(Resolv::ResolvError)
|
|
|
|
domain, source = determine_instance_domain
|
|
|
|
expect(domain).to eq("10.0.0.5")
|
|
expect(source).to eq(:protected_ip)
|
|
end
|
|
|
|
it "falls back to a local IP address when no other sources are available" do
|
|
loopback_address = Addrinfo.ip("127.0.0.1")
|
|
allow(Socket).to receive(:ip_address_list).and_return([loopback_address])
|
|
allow(Resolv).to receive(:getname).and_raise(Resolv::ResolvError)
|
|
|
|
domain, source = determine_instance_domain
|
|
|
|
expect(domain).to eq("127.0.0.1")
|
|
expect(source).to eq(:local_ip)
|
|
end
|
|
end
|
|
|
|
describe "#determine_app_version" do
|
|
let(:repo_root) { File.expand_path("..", __dir__) }
|
|
let(:git_dir) { File.join(repo_root, ".git") }
|
|
|
|
before do
|
|
allow(File).to receive(:directory?).and_call_original
|
|
end
|
|
|
|
it "returns the fallback when the git directory is missing" do
|
|
allow(File).to receive(:directory?).with(git_dir).and_return(false)
|
|
|
|
expect(determine_app_version).to eq(PotatoMesh::Config.version_fallback)
|
|
end
|
|
|
|
it "returns the fallback when git describe fails" do
|
|
allow(File).to receive(:directory?).with(git_dir).and_return(true)
|
|
status = instance_double(Process::Status, success?: false)
|
|
allow(Open3).to receive(:capture2).and_return(["ignored", status])
|
|
|
|
expect(determine_app_version).to eq(PotatoMesh::Config.version_fallback)
|
|
end
|
|
|
|
it "returns the fallback when git describe output is empty" do
|
|
allow(File).to receive(:directory?).with(git_dir).and_return(true)
|
|
status = instance_double(Process::Status, success?: true)
|
|
allow(Open3).to receive(:capture2).and_return(["\n", status])
|
|
|
|
expect(determine_app_version).to eq(PotatoMesh::Config.version_fallback)
|
|
end
|
|
|
|
it "returns the original describe output when the format is unexpected" do
|
|
allow(File).to receive(:directory?).with(git_dir).and_return(true)
|
|
status = instance_double(Process::Status, success?: true)
|
|
allow(Open3).to receive(:capture2).and_return(["weird-output", status])
|
|
|
|
expect(determine_app_version).to eq("weird-output")
|
|
end
|
|
|
|
it "normalises the version when no commits are ahead of the tag" do
|
|
allow(File).to receive(:directory?).with(git_dir).and_return(true)
|
|
status = instance_double(Process::Status, success?: true)
|
|
allow(Open3).to receive(:capture2).and_return(["v1.2.3-0-gabcdef1", status])
|
|
|
|
expect(determine_app_version).to eq("v1.2.3")
|
|
end
|
|
|
|
it "includes commit metadata when ahead of the tag" do
|
|
allow(File).to receive(:directory?).with(git_dir).and_return(true)
|
|
status = instance_double(Process::Status, success?: true)
|
|
allow(Open3).to receive(:capture2).and_return(["v1.2.3-5-gabcdef1", status])
|
|
|
|
expect(determine_app_version).to eq("v1.2.3+5-abcdef1")
|
|
end
|
|
|
|
it "returns the fallback when git describe raises an error" do
|
|
allow(File).to receive(:directory?).with(git_dir).and_return(true)
|
|
allow(Open3).to receive(:capture2).and_raise(StandardError, "boom")
|
|
|
|
expect(determine_app_version).to eq(PotatoMesh::Config.version_fallback)
|
|
end
|
|
end
|
|
|
|
describe "string coercion helpers" do
|
|
it "normalises strings and nil values" do
|
|
expect(sanitized_string(" spaced ")).to eq("spaced")
|
|
expect(sanitized_string(nil)).to eq("")
|
|
end
|
|
|
|
it "returns nil for blank matrix rooms" do
|
|
allow(PotatoMesh::Config).to receive(:matrix_room).and_return(" \t ")
|
|
expect(sanitized_matrix_room).to be_nil
|
|
end
|
|
|
|
it "coerces string_or_nil inputs" do
|
|
expect(string_or_nil(" hello \n")).to eq("hello")
|
|
expect(string_or_nil(" ")).to be_nil
|
|
expect(string_or_nil(123)).to eq("123")
|
|
end
|
|
end
|
|
|
|
describe "#coerce_integer" do
|
|
it "coerces integers and floats" do
|
|
expect(coerce_integer(5)).to eq(5)
|
|
expect(coerce_integer(7.9)).to eq(7)
|
|
end
|
|
|
|
it "coerces numeric strings" do
|
|
expect(coerce_integer(" 42 ")).to eq(42)
|
|
expect(coerce_integer("0x1a")).to eq(26)
|
|
expect(coerce_integer("12.8")).to eq(12)
|
|
end
|
|
|
|
it "returns nil for invalid values" do
|
|
expect(coerce_integer("not-a-number")).to be_nil
|
|
expect(coerce_integer(Float::INFINITY)).to be_nil
|
|
end
|
|
end
|
|
|
|
describe "#coerce_float" do
|
|
it "coerces numeric types" do
|
|
expect(coerce_float(5)).to eq(5.0)
|
|
expect(coerce_float(3.2)).to eq(3.2)
|
|
end
|
|
|
|
it "coerces numeric strings" do
|
|
expect(coerce_float(" 8.5 ")).to eq(8.5)
|
|
end
|
|
|
|
it "returns nil for invalid inputs" do
|
|
expect(coerce_float("bad")).to be_nil
|
|
expect(coerce_float(Float::INFINITY)).to be_nil
|
|
end
|
|
end
|
|
|
|
describe "JSON normalisation helpers" do
|
|
it "normalises nested hashes" do
|
|
input = { foo: { bar: 1, baz: [1, { qux: 2 }] } }
|
|
result = normalize_json_value(input)
|
|
expect(result).to eq("foo" => { "bar" => 1, "baz" => [1, { "qux" => 2 }] })
|
|
end
|
|
|
|
it "parses JSON strings into hashes" do
|
|
json = '{"foo": {"bar": 1}}'
|
|
expect(normalize_json_object(json)).to eq("foo" => { "bar" => 1 })
|
|
end
|
|
|
|
it "returns nil for invalid JSON objects" do
|
|
expect(normalize_json_object("not json")).to be_nil
|
|
expect(normalize_json_object(123)).to be_nil
|
|
end
|
|
end
|
|
|
|
describe "distance helpers" do
|
|
it "formats integers without trailing decimals" do
|
|
expect(formatted_distance_km(120.0)).to eq("120")
|
|
expect(formatted_distance_km(12.34)).to eq("12.3")
|
|
end
|
|
|
|
it "returns nil when the maximum distance is invalid" do
|
|
allow(PotatoMesh::Config).to receive(:max_node_distance_km).and_return(-5)
|
|
expect(sanitized_max_distance_km).to be_nil
|
|
|
|
allow(PotatoMesh::Config).to receive(:max_node_distance_km).and_return("string")
|
|
expect(sanitized_max_distance_km).to be_nil
|
|
|
|
allow(PotatoMesh::Config).to receive(:max_node_distance_km).and_return(15.5)
|
|
expect(sanitized_max_distance_km).to eq(15.5)
|
|
end
|
|
end
|
|
|
|
describe "#secure_token_match?" do
|
|
it "performs constant-time comparison for matching strings" do
|
|
expect(secure_token_match?("abc", "abc")).to be(true)
|
|
end
|
|
|
|
it "returns false when inputs differ" do
|
|
expect(secure_token_match?("abc", "xyz")).to be(false)
|
|
expect(secure_token_match?("abc", nil)).to be(false)
|
|
end
|
|
|
|
it "handles secure compare errors" do
|
|
stub_const("Rack::Utils::SecurityError", Class.new(StandardError))
|
|
allow(Rack::Utils).to receive(:secure_compare).and_raise(Rack::Utils::SecurityError.new("boom"))
|
|
expect(secure_token_match?("abc", "abc")).to be(false)
|
|
end
|
|
end
|
|
|
|
describe "#with_busy_retry" do
|
|
it "raises once the retry budget is exhausted" do
|
|
attempts = 0
|
|
|
|
expect do
|
|
with_busy_retry(max_retries: 2, base_delay: 0.0) do
|
|
attempts += 1
|
|
raise SQLite3::BusyException if attempts <= 3
|
|
end
|
|
end.to raise_error(SQLite3::BusyException)
|
|
|
|
expect(attempts).to eq(3)
|
|
end
|
|
end
|
|
|
|
describe "#resolve_node_num" do
|
|
it "reads numeric aliases from payloads" do
|
|
expect(resolve_node_num(nil, "num" => 42)).to eq(42)
|
|
expect(resolve_node_num(nil, "num" => 7.2)).to eq(7)
|
|
expect(resolve_node_num(nil, "num" => " 123 ")).to eq(123)
|
|
expect(resolve_node_num("!feedcafe", "num" => "feedcafe")).to eq(0xfeedcafe)
|
|
end
|
|
|
|
it "infers the numeric alias from the canonical identifier" do
|
|
expect(resolve_node_num("!00ff00aa", {})).to eq(0x00ff00aa)
|
|
end
|
|
|
|
it "returns nil for invalid identifiers" do
|
|
expect(resolve_node_num("!nothex", {})).to be_nil
|
|
expect(resolve_node_num(nil, "num" => "")).to be_nil
|
|
expect(resolve_node_num("", {})).to be_nil
|
|
end
|
|
end
|
|
|
|
describe "#canonical_node_parts" do
|
|
it "parses integers, strings, and fallbacks" do
|
|
parts = canonical_node_parts(123, nil)
|
|
expect(parts).to eq(["!0000007b", 123, "007B"])
|
|
|
|
parts = canonical_node_parts("!feedcafe", nil)
|
|
expect(parts).to eq(["!feedcafe", 0xfeedcafe, "CAFE"])
|
|
|
|
parts = canonical_node_parts("0x10", nil)
|
|
expect(parts).to eq(["!00000010", 16, "0010"])
|
|
|
|
parts = canonical_node_parts(nil, 31)
|
|
expect(parts).to eq(["!0000001f", 31, "001F"])
|
|
end
|
|
|
|
it "rejects invalid references" do
|
|
expect(canonical_node_parts("", nil)).to be_nil
|
|
expect(canonical_node_parts("not-valid", nil)).to be_nil
|
|
expect(canonical_node_parts(-5, nil)).to be_nil
|
|
expect(canonical_node_parts(Object.new, nil)).to be_nil
|
|
end
|
|
end
|
|
|
|
describe "#ensure_unknown_node" do
|
|
it "does not create duplicate placeholder nodes" do
|
|
node_id = "!dupe0001"
|
|
with_db do |db|
|
|
db.execute("INSERT INTO nodes(node_id) VALUES (?)", [node_id])
|
|
expect(ensure_unknown_node(db, node_id, nil, heard_time: reference_time.to_i)).to be_falsey
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#touch_node_last_seen" do
|
|
it "updates nodes using fallback numeric identifiers" do
|
|
node_id = "!12345678"
|
|
node_num = 0x1234_5678
|
|
rx_time = reference_time.to_i - 30
|
|
|
|
with_db do |db|
|
|
db.execute(
|
|
"INSERT INTO nodes(node_id, num, last_heard, first_heard) VALUES (?,?,?,?)",
|
|
[node_id, node_num, rx_time - 120, rx_time - 180],
|
|
)
|
|
|
|
updated = touch_node_last_seen(db, nil, node_num, rx_time: rx_time, source: :spec)
|
|
expect(updated).to be_truthy
|
|
end
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT last_heard, first_heard FROM nodes WHERE node_id = ?",
|
|
[node_id],
|
|
)
|
|
expect(row["last_heard"]).to eq(rx_time)
|
|
expect(row["first_heard"]).to eq(rx_time - 180)
|
|
end
|
|
end
|
|
|
|
it "returns nil when the timestamp cannot be coerced" do
|
|
with_db do |db|
|
|
expect(touch_node_last_seen(db, "!unknown", nil, rx_time: " ")).to be_nil
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#normalize_node_id" do
|
|
it "resolves numeric aliases to canonical identifiers" do
|
|
node_id = "!alias000"
|
|
with_db do |db|
|
|
db.execute(
|
|
"INSERT INTO nodes(node_id, num) VALUES (?, ?)",
|
|
[node_id, 321],
|
|
)
|
|
end
|
|
|
|
with_db(readonly: true) do |db|
|
|
expect(normalize_node_id(db, "321")).to eq(node_id)
|
|
expect(normalize_node_id(db, "!missing")).to be_nil
|
|
expect(normalize_node_id(db, nil)).to be_nil
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "logging configuration" do
|
|
before do
|
|
Sinatra::Application.apply_logger_level!
|
|
end
|
|
|
|
after do
|
|
Sinatra::Application.apply_logger_level!
|
|
end
|
|
|
|
it "defaults to WARN when debug logging is disabled" do
|
|
expect(Sinatra::Application.settings.logger.level).to eq(Logger::WARN)
|
|
end
|
|
|
|
it "switches to DEBUG when debug logging is enabled" do
|
|
allow(PotatoMesh::Config).to receive(:debug?).and_return(true)
|
|
Sinatra::Application.apply_logger_level!
|
|
|
|
expect(Sinatra::Application.settings.logger.level).to eq(Logger::DEBUG)
|
|
end
|
|
end
|
|
|
|
describe "GET /favicon.ico" do
|
|
it "serves the bundled favicon when available" do
|
|
get "/favicon.ico"
|
|
expect(last_response).to be_ok
|
|
expect(last_response.headers["Content-Type"]).to eq("image/vnd.microsoft.icon")
|
|
end
|
|
|
|
it "falls back to the SVG logo when the favicon is missing" do
|
|
ico_path = File.join(Sinatra::Application.settings.public_folder, "favicon.ico")
|
|
allow(File).to receive(:file?).and_call_original
|
|
allow(File).to receive(:file?).with(ico_path).and_return(false)
|
|
|
|
get "/favicon.ico"
|
|
|
|
expect(last_response).to be_ok
|
|
expect(last_response.headers["Content-Type"]).to eq("image/svg+xml")
|
|
end
|
|
end
|
|
|
|
describe "GET /potatomesh-logo.svg" do
|
|
it "serves the cached SVG asset when present" do
|
|
get "/potatomesh-logo.svg"
|
|
expect(last_response).to be_ok
|
|
expect(last_response.headers["Content-Type"]).to eq("image/svg+xml")
|
|
end
|
|
|
|
it "returns 404 when the asset is missing" do
|
|
svg_path = File.expand_path("potatomesh-logo.svg", Sinatra::Application.settings.public_folder)
|
|
allow(File).to receive(:exist?).and_return(false)
|
|
allow(File).to receive(:readable?).and_return(false)
|
|
|
|
get "/potatomesh-logo.svg"
|
|
|
|
expect(last_response.status).to eq(404)
|
|
end
|
|
end
|
|
|
|
describe "GET /" do
|
|
it "responds successfully" do
|
|
get "/"
|
|
expect(last_response).to be_ok
|
|
end
|
|
|
|
it "includes the application version in the footer" do
|
|
get "/"
|
|
expect(last_response.body).to include("#{APP_VERSION}")
|
|
end
|
|
|
|
it "includes SEO metadata from configuration" do
|
|
allow(PotatoMesh::Config).to receive(:site_name).and_return("Spec Mesh Title")
|
|
allow(PotatoMesh::Config).to receive(:default_channel).and_return("#SpecChannel")
|
|
allow(PotatoMesh::Config).to receive(:default_frequency).and_return("915MHz")
|
|
allow(PotatoMesh::Config).to receive(:max_node_distance_km).and_return(120.5)
|
|
allow(PotatoMesh::Config).to receive(:matrix_room).and_return(" #spec-room:example.org ")
|
|
|
|
expected_description = "Live Meshtastic mesh map for Spec Mesh Title on #SpecChannel (915MHz). Track nodes, messages, and coverage in real time. Shows nodes within roughly 120.5 km of the map center. Join the community in #spec-room:example.org on Matrix."
|
|
|
|
get "/"
|
|
|
|
expect(last_response.body).to include(%(meta name="description" content="#{expected_description}" />))
|
|
expect(last_response.body).to include('<meta property="og:title" content="Spec Mesh Title" />')
|
|
expect(last_response.body).to include('<meta property="og:site_name" content="Spec Mesh Title" />')
|
|
expect(last_response.body).to include('<meta name="twitter:image" content="http://example.org/potatomesh-logo.svg" />')
|
|
end
|
|
end
|
|
|
|
describe "database initialization" do
|
|
it "creates the schema when booting" do
|
|
expect(File).to exist(PotatoMesh::Config.db_path)
|
|
|
|
db = SQLite3::Database.new(PotatoMesh::Config.db_path, readonly: true)
|
|
tables = db.execute("SELECT name FROM sqlite_master WHERE type='table' AND name IN ('nodes','messages')").flatten
|
|
|
|
expect(tables).to include("nodes")
|
|
expect(tables).to include("messages")
|
|
ensure
|
|
db&.close
|
|
end
|
|
end
|
|
|
|
describe "authentication" do
|
|
it "rejects requests without a matching bearer token" do
|
|
post "/api/nodes", {}.to_json, { "CONTENT_TYPE" => "application/json" }
|
|
|
|
expect(last_response.status).to eq(403)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "Forbidden")
|
|
end
|
|
|
|
it "rejects requests when the API token is not configured" do
|
|
ENV["API_TOKEN"] = nil
|
|
|
|
post "/api/messages", {}.to_json, { "CONTENT_TYPE" => "application/json" }
|
|
|
|
expect(last_response.status).to eq(403)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "Forbidden")
|
|
ensure
|
|
ENV["API_TOKEN"] = api_token
|
|
end
|
|
|
|
it "rejects requests with the wrong bearer token" do
|
|
headers = auth_headers.merge("HTTP_AUTHORIZATION" => "Bearer wrong-token")
|
|
|
|
post "/api/messages", {}.to_json, headers
|
|
|
|
expect(last_response.status).to eq(403)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "Forbidden")
|
|
end
|
|
|
|
it "does not accept alternate authorization schemes" do
|
|
basic = Base64.strict_encode64("attacker:password")
|
|
headers = auth_headers.merge("HTTP_AUTHORIZATION" => "Basic #{basic}")
|
|
|
|
post "/api/nodes", {}.to_json, headers
|
|
|
|
expect(last_response.status).to eq(403)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "Forbidden")
|
|
end
|
|
|
|
it "rejects tokens with unexpected trailing characters" do
|
|
headers = auth_headers.merge("HTTP_AUTHORIZATION" => "Bearer #{api_token} ")
|
|
|
|
post "/api/messages", {}.to_json, headers
|
|
|
|
expect(last_response.status).to eq(403)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "Forbidden")
|
|
end
|
|
end
|
|
|
|
describe "POST /api/instances" do
|
|
let(:instance_key) { OpenSSL::PKey::RSA.new(2048) }
|
|
let(:domain) { "mesh.example" }
|
|
let(:pubkey) { instance_key.public_key.export }
|
|
let(:last_update_time) { Time.now.to_i }
|
|
let(:instance_attributes) do
|
|
{
|
|
id: "mesh-instance-1",
|
|
domain: domain,
|
|
pubkey: pubkey,
|
|
name: "Example Mesh",
|
|
version: "1.2.3",
|
|
channel: "#MeshNet",
|
|
frequency: "915MHz",
|
|
latitude: 52.5,
|
|
longitude: 13.4,
|
|
last_update_time: last_update_time,
|
|
is_private: false,
|
|
}
|
|
end
|
|
let(:instance_signature_payload) do
|
|
canonical_instance_payload(instance_attributes)
|
|
end
|
|
let(:instance_signature) do
|
|
Base64.strict_encode64(
|
|
instance_key.sign(OpenSSL::Digest::SHA256.new, instance_signature_payload),
|
|
)
|
|
end
|
|
let(:instance_payload) do
|
|
{
|
|
"id" => instance_attributes[:id],
|
|
"domain" => domain,
|
|
"pubkey" => pubkey,
|
|
"name" => instance_attributes[:name],
|
|
"version" => instance_attributes[:version],
|
|
"channel" => instance_attributes[:channel],
|
|
"frequency" => instance_attributes[:frequency],
|
|
"latitude" => instance_attributes[:latitude],
|
|
"longitude" => instance_attributes[:longitude],
|
|
"lastUpdateTime" => instance_attributes[:last_update_time],
|
|
"isPrivate" => instance_attributes[:is_private],
|
|
"signature" => instance_signature,
|
|
}
|
|
end
|
|
let(:remote_signed_payload) do
|
|
JSON.generate(
|
|
{
|
|
"publicKey" => pubkey,
|
|
"name" => instance_attributes[:name],
|
|
"version" => instance_attributes[:version],
|
|
"domain" => domain,
|
|
"lastUpdate" => last_update_time,
|
|
},
|
|
sort_keys: true,
|
|
)
|
|
end
|
|
let(:well_known_document) do
|
|
{
|
|
"publicKey" => pubkey,
|
|
"domain" => domain,
|
|
"name" => instance_attributes[:name],
|
|
"version" => instance_attributes[:version],
|
|
"lastUpdate" => last_update_time,
|
|
"signatureAlgorithm" => "rsa-sha256",
|
|
"signedPayload" => Base64.strict_encode64(remote_signed_payload),
|
|
"signature" => Base64.strict_encode64(
|
|
instance_key.sign(OpenSSL::Digest::SHA256.new, remote_signed_payload),
|
|
),
|
|
}
|
|
end
|
|
let(:remote_nodes) do
|
|
now = Time.now.to_i
|
|
Array.new(PotatoMesh::Config.remote_instance_min_node_count) do |index|
|
|
{
|
|
"node_id" => "remote-node-#{index}",
|
|
"last_heard" => now - index,
|
|
}
|
|
end
|
|
end
|
|
|
|
before do
|
|
allow_any_instance_of(Sinatra::Application).to receive(:fetch_instance_json) do |_instance, host, path|
|
|
case path
|
|
when "/.well-known/potato-mesh"
|
|
[well_known_document, URI("https://#{host}#{path}")]
|
|
when "/api/nodes"
|
|
[remote_nodes, URI("https://#{host}#{path}")]
|
|
else
|
|
[nil, []]
|
|
end
|
|
end
|
|
end
|
|
|
|
it "stores a federated instance when validation succeeds" do
|
|
post "/api/instances", instance_payload.to_json, { "CONTENT_TYPE" => "application/json" }
|
|
|
|
expect(last_response.status).to eq(201)
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "registered")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT * FROM instances WHERE id = ?",
|
|
[instance_attributes[:id]],
|
|
)
|
|
|
|
expect(row).not_to be_nil
|
|
expect(row["domain"]).to eq(domain)
|
|
expect(row["pubkey"]).to eq(pubkey)
|
|
expect(row["signature"]).to eq(instance_signature)
|
|
expect(row["is_private"]).to eq(0)
|
|
end
|
|
end
|
|
|
|
it "rejects registrations with invalid signatures" do
|
|
invalid_payload = instance_payload.merge("signature" => Base64.strict_encode64("invalid"))
|
|
|
|
expect_any_instance_of(Object).to receive(:warn).with(/invalid signature/).at_least(:once)
|
|
|
|
post "/api/instances", invalid_payload.to_json, { "CONTENT_TYPE" => "application/json" }
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "invalid signature")
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM instances")
|
|
expect(count).to eq(1)
|
|
end
|
|
end
|
|
|
|
it "rejects registrations targeting restricted literal IPs when a port is supplied" do
|
|
restricted_domain = "127.0.0.1:8080"
|
|
restricted_attributes = instance_attributes.merge(domain: restricted_domain)
|
|
restricted_signature_payload = canonical_instance_payload(restricted_attributes)
|
|
restricted_signature = Base64.strict_encode64(
|
|
instance_key.sign(OpenSSL::Digest::SHA256.new, restricted_signature_payload),
|
|
)
|
|
restricted_payload = instance_payload.merge(
|
|
"domain" => restricted_domain,
|
|
"signature" => restricted_signature,
|
|
)
|
|
|
|
expect_any_instance_of(Object).to receive(:warn).with(/restricted IP address/).at_least(:once)
|
|
|
|
post "/api/instances", restricted_payload.to_json, { "CONTENT_TYPE" => "application/json" }
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "restricted domain")
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM instances")
|
|
expect(count).to eq(1)
|
|
end
|
|
end
|
|
|
|
it "accepts signatures when the optional isPrivate field is omitted" do
|
|
unsigned_attributes = instance_attributes.merge(is_private: nil)
|
|
unsigned_payload_json = canonical_instance_payload(unsigned_attributes)
|
|
unsigned_signature = Base64.strict_encode64(
|
|
instance_key.sign(OpenSSL::Digest::SHA256.new, unsigned_payload_json),
|
|
)
|
|
payload_without_private = instance_payload.reject { |key, _| key == "isPrivate" }
|
|
payload_without_private["signature"] = unsigned_signature
|
|
|
|
post "/api/instances", payload_without_private.to_json, { "CONTENT_TYPE" => "application/json" }
|
|
|
|
expect(last_response.status).to eq(201)
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "registered")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT * FROM instances WHERE id = ?",
|
|
[instance_attributes[:id]],
|
|
)
|
|
|
|
expect(row).not_to be_nil
|
|
expect(row["is_private"]).to eq(0)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "GET /api/instances" do
|
|
let(:remote_key) { OpenSSL::PKey::RSA.new(2048) }
|
|
|
|
it "returns the self instance record" do
|
|
get "/api/instances"
|
|
|
|
expect(last_response).to be_ok
|
|
payload = JSON.parse(last_response.body)
|
|
self_entry = payload.find { |entry| entry["id"] == SELF_INSTANCE_ID }
|
|
|
|
expect(self_entry).not_to be_nil
|
|
expect(self_entry["domain"]).not_to be_nil
|
|
expect(self_entry["isPrivate"]).to eq(false)
|
|
expect(self_entry["signature"]).not_to be_nil
|
|
end
|
|
|
|
it "includes previously stored remote registrations" do
|
|
remote_attributes = {
|
|
id: "remote-instance-1",
|
|
domain: "remote.example",
|
|
pubkey: remote_key.public_key.export,
|
|
name: "Remote Mesh",
|
|
version: "9.8.7",
|
|
channel: "#Remote",
|
|
frequency: "915MHz",
|
|
latitude: 51.5,
|
|
longitude: -0.1,
|
|
last_update_time: Time.now.to_i,
|
|
is_private: false,
|
|
}
|
|
remote_signature_payload = canonical_instance_payload(remote_attributes)
|
|
remote_signature = Base64.strict_encode64(
|
|
remote_key.sign(OpenSSL::Digest::SHA256.new, remote_signature_payload),
|
|
)
|
|
|
|
with_db do |db|
|
|
upsert_instance_record(db, remote_attributes, remote_signature)
|
|
end
|
|
|
|
get "/api/instances"
|
|
|
|
expect(last_response).to be_ok
|
|
payload = JSON.parse(last_response.body)
|
|
remote_entry = payload.find { |entry| entry["id"] == remote_attributes[:id] }
|
|
|
|
expect(remote_entry).not_to be_nil
|
|
expect(remote_entry["domain"]).to eq("remote.example")
|
|
expect(remote_entry["isPrivate"]).to eq(false)
|
|
expect(remote_entry["signature"]).to eq(remote_signature)
|
|
end
|
|
end
|
|
|
|
describe "POST /api/nodes" do
|
|
it "imports nodes from fixture data into the database" do
|
|
import_nodes_fixture
|
|
|
|
expected_nodes = nodes_fixture.map do |node|
|
|
[node["node_id"], expected_node_row(node)]
|
|
end.to_h
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
rows = db.execute(<<~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, latitude, longitude, altitude
|
|
FROM nodes
|
|
ORDER BY node_id
|
|
SQL
|
|
|
|
expect(rows.size).to eq(expected_nodes.size)
|
|
|
|
rows.each do |row|
|
|
expected = expected_nodes.fetch(row["node_id"])
|
|
expect(row["short_name"]).to eq(expected["short_name"])
|
|
expect(row["long_name"]).to eq(expected["long_name"])
|
|
expect(row["hw_model"]).to eq(expected["hw_model"])
|
|
expect(row["role"]).to eq(expected["role"])
|
|
expect_same_value(row["snr"], expected["snr"])
|
|
expect_same_value(row["battery_level"], expected["battery_level"])
|
|
expect_same_value(row["voltage"], expected["voltage"])
|
|
expect(row["last_heard"]).to eq(expected["last_heard"])
|
|
expect(row["first_heard"]).to eq(expected["first_heard"])
|
|
expect_same_value(row["uptime_seconds"], expected["uptime_seconds"])
|
|
expect_same_value(row["channel_utilization"], expected["channel_utilization"])
|
|
expect_same_value(row["air_util_tx"], expected["air_util_tx"])
|
|
expect_same_value(row["position_time"], expected["position_time"])
|
|
expect_same_value(row["latitude"], expected["latitude"])
|
|
expect_same_value(row["longitude"], expected["longitude"])
|
|
expect_same_value(row["altitude"], expected["altitude"])
|
|
end
|
|
end
|
|
end
|
|
|
|
it "returns 400 when the payload is not valid JSON" do
|
|
post "/api/nodes", "{", auth_headers
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "invalid JSON")
|
|
end
|
|
|
|
it "updates timestamps when the payload omits lastHeard" do
|
|
node_id = "!spectime01"
|
|
payload = {
|
|
node_id => {
|
|
"user" => { "shortName" => "Spec Time" },
|
|
},
|
|
}
|
|
|
|
post "/api/nodes", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT last_heard, first_heard FROM nodes WHERE node_id = ?",
|
|
[node_id],
|
|
)
|
|
|
|
expect(row["last_heard"]).to eq(reference_time.to_i)
|
|
expect(row["first_heard"]).to eq(reference_time.to_i)
|
|
end
|
|
end
|
|
|
|
it "preserves the original first_heard when updating nodes" do
|
|
node_id = "!spectime02"
|
|
initial_first = reference_time.to_i - 600
|
|
initial_last = reference_time.to_i - 300
|
|
|
|
with_db do |db|
|
|
db.execute(
|
|
"INSERT INTO nodes(node_id, last_heard, first_heard) VALUES (?,?,?)",
|
|
[node_id, initial_last, initial_first],
|
|
)
|
|
end
|
|
|
|
payload = {
|
|
node_id => {
|
|
"user" => { "shortName" => "Spec Update" },
|
|
"lastHeard" => reference_time.to_i,
|
|
},
|
|
}
|
|
|
|
post "/api/nodes", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT last_heard, first_heard FROM nodes WHERE node_id = ?",
|
|
[node_id],
|
|
)
|
|
|
|
expect(row["last_heard"]).to eq(reference_time.to_i)
|
|
expect(row["first_heard"]).to eq(initial_first)
|
|
end
|
|
end
|
|
|
|
it "returns 400 when more than 1000 nodes are provided" do
|
|
payload = (0..1000).each_with_object({}) do |i, acc|
|
|
acc["node-#{i}"] = {}
|
|
end
|
|
|
|
post "/api/nodes", payload.to_json, auth_headers
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "too many nodes")
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM nodes")
|
|
expect(count).to eq(0)
|
|
end
|
|
end
|
|
|
|
it "returns 413 when the request body exceeds the configured byte limit" do
|
|
limit = 64
|
|
allow(PotatoMesh::Config).to receive(:max_json_body_bytes).and_return(limit)
|
|
payload = { "huge-node" => { "user" => { "shortName" => "A" * (limit + 50) } } }.to_json
|
|
expect(payload.bytesize).to be > limit
|
|
|
|
post "/api/nodes", payload, auth_headers
|
|
|
|
expect(last_response.status).to eq(413)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "payload too large")
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM nodes")
|
|
expect(count).to eq(0)
|
|
end
|
|
end
|
|
|
|
it "treats SQL-looking node identifiers as plain data" do
|
|
malicious_id = "spec-node'); DROP TABLE nodes;--"
|
|
payload = {
|
|
malicious_id => {
|
|
"user" => { "shortName" => "Spec Attack" },
|
|
"lastHeard" => reference_time.to_i,
|
|
},
|
|
}
|
|
|
|
post "/api/nodes", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT node_id, short_name FROM nodes WHERE node_id = ?",
|
|
[malicious_id],
|
|
)
|
|
|
|
expect(row["node_id"]).to eq(malicious_id)
|
|
expect(row["short_name"]).to eq("Spec Attack")
|
|
|
|
tables = db.get_first_value(
|
|
"SELECT COUNT(*) FROM sqlite_master WHERE type='table' AND name='nodes'",
|
|
)
|
|
expect(tables).to eq(1)
|
|
end
|
|
end
|
|
|
|
it "retries node upserts when the database reports it is locked" do
|
|
node = nodes_fixture.first
|
|
payload = { node["node_id"] => build_node_payload(node) }
|
|
|
|
call_count = 0
|
|
allow_any_instance_of(SQLite3::Database).to receive(:execute).and_wrap_original do |method, sql, *args|
|
|
if sql.include?("INSERT INTO nodes")
|
|
call_count += 1
|
|
raise SQLite3::BusyException, "database is locked" if call_count == 1
|
|
end
|
|
method.call(sql, *args)
|
|
end
|
|
|
|
post "/api/nodes", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
expect(call_count).to be >= 2
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM nodes WHERE node_id = ?", [node["node_id"]])
|
|
expect(count).to eq(1)
|
|
|
|
last_heard = db.get_first_value("SELECT last_heard FROM nodes WHERE node_id = ?", [node["node_id"]])
|
|
expect(last_heard).to eq(expected_last_heard(node))
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#ensure_unknown_node" do
|
|
it "creates a hidden placeholder with timestamps for chat notifications" do
|
|
with_db do |db|
|
|
created = ensure_unknown_node(db, "!1234abcd", nil, heard_time: reference_time.to_i)
|
|
expect(created).to be_truthy
|
|
end
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
<<~SQL,
|
|
SELECT short_name, long_name, role, last_heard, first_heard
|
|
FROM nodes
|
|
WHERE node_id = ?
|
|
SQL
|
|
["!1234abcd"],
|
|
)
|
|
|
|
expect(row["short_name"]).to eq("ABCD")
|
|
expect(row["long_name"]).to eq("Meshtastic ABCD")
|
|
expect(row["role"]).to eq("CLIENT_HIDDEN")
|
|
expect(row["last_heard"]).to eq(reference_time.to_i)
|
|
expect(row["first_heard"]).to eq(reference_time.to_i)
|
|
end
|
|
end
|
|
|
|
it "leaves timestamps nil when no receive time is provided" do
|
|
with_db do |db|
|
|
created = ensure_unknown_node(db, "!1111beef", nil)
|
|
expect(created).to be_truthy
|
|
end
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
<<~SQL,
|
|
SELECT last_heard, first_heard
|
|
FROM nodes
|
|
WHERE node_id = ?
|
|
SQL
|
|
["!1111beef"],
|
|
)
|
|
|
|
expect(row["last_heard"]).to be_nil
|
|
expect(row["first_heard"]).to be_nil
|
|
end
|
|
end
|
|
|
|
it "returns false when the node already exists" do
|
|
with_db do |db|
|
|
expect(ensure_unknown_node(db, "!0000c0de", nil)).to be_truthy
|
|
expect(ensure_unknown_node(db, "!0000c0de", nil)).to be_falsey
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "POST /api/messages" do
|
|
it "persists messages from fixture data" do
|
|
import_nodes_fixture
|
|
import_messages_fixture
|
|
|
|
expected_messages = messages_fixture.map do |message|
|
|
[message["id"], message.reject { |key, _| key == "node" }]
|
|
end.to_h
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
rows = db.execute(<<~SQL)
|
|
SELECT id, rx_time, rx_iso, from_id, to_id, channel,
|
|
portnum, text, snr, rssi, hop_limit
|
|
FROM messages
|
|
ORDER BY id
|
|
SQL
|
|
|
|
expect(rows.size).to eq(expected_messages.size)
|
|
|
|
rows.each do |row|
|
|
expected = expected_messages.fetch(row["id"])
|
|
expect(row["rx_time"]).to eq(expected["rx_time"])
|
|
expect(row["rx_iso"]).to eq(expected["rx_iso"])
|
|
expect(row["from_id"]).to eq(expected["from_id"])
|
|
expect(row["to_id"]).to eq(expected["to_id"])
|
|
expect(row["channel"]).to eq(expected["channel"])
|
|
expect(row["portnum"]).to eq(expected["portnum"])
|
|
expect(row["text"]).to eq(expected["text"])
|
|
expect_same_value(row["snr"], expected["snr"])
|
|
expect(row["rssi"]).to eq(expected["rssi"])
|
|
expect(row["hop_limit"]).to eq(expected["hop_limit"])
|
|
end
|
|
end
|
|
end
|
|
|
|
it "creates hidden nodes for unknown message senders" do
|
|
payload = {
|
|
"id" => 9_999,
|
|
"rx_time" => reference_time.to_i,
|
|
"rx_iso" => reference_time.iso8601,
|
|
"from_id" => "!feedf00d",
|
|
"to_id" => "^all",
|
|
"channel" => 0,
|
|
"portnum" => "TEXT_MESSAGE_APP",
|
|
"text" => "Spec placeholder message",
|
|
}
|
|
|
|
post "/api/messages", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT node_id, num, short_name, long_name, role, last_heard, first_heard FROM nodes WHERE node_id = ?",
|
|
["!feedf00d"],
|
|
)
|
|
|
|
expect(row).not_to be_nil
|
|
expect(row["node_id"]).to eq("!feedf00d")
|
|
expect(row["num"]).to eq(0xfeedf00d)
|
|
expect(row["short_name"]).to eq("F00D")
|
|
expect(row["long_name"]).to eq("Meshtastic F00D")
|
|
expect(row["role"]).to eq("CLIENT_HIDDEN")
|
|
expect(row["last_heard"]).to eq(payload["rx_time"])
|
|
expect(row["first_heard"]).to eq(payload["rx_time"])
|
|
end
|
|
end
|
|
|
|
it "returns 400 when the payload is not valid JSON" do
|
|
post "/api/messages", "{", auth_headers
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "invalid JSON")
|
|
end
|
|
|
|
it "rejects message payloads that are larger than the configured byte limit" do
|
|
limit = 64
|
|
allow(PotatoMesh::Config).to receive(:max_json_body_bytes).and_return(limit)
|
|
payload = [{ "id" => "m1", "text" => "A" * (limit + 50) }].to_json
|
|
expect(payload.bytesize).to be > limit
|
|
|
|
post "/api/messages", payload, auth_headers
|
|
|
|
expect(last_response.status).to eq(413)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "payload too large")
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM messages")
|
|
expect(count).to eq(0)
|
|
end
|
|
end
|
|
|
|
describe "POST /api/positions" do
|
|
it "stores position packets and updates node metadata" do
|
|
node_id = "!specpos01"
|
|
node_num = 0x1234_5678
|
|
initial_last_heard = reference_time.to_i - 600
|
|
node_payload = {
|
|
node_id => {
|
|
"num" => node_num,
|
|
"user" => { "shortName" => "SpecPos" },
|
|
"lastHeard" => initial_last_heard,
|
|
"position" => {
|
|
"time" => initial_last_heard - 60,
|
|
"latitude" => 52.0,
|
|
"longitude" => 13.0,
|
|
},
|
|
},
|
|
}
|
|
|
|
post "/api/nodes", node_payload.to_json, auth_headers
|
|
expect(last_response).to be_ok
|
|
|
|
rx_time = reference_time.to_i - 120
|
|
position_time = rx_time - 30
|
|
raw_payload = { "time" => position_time, "latitude_i" => (52.5 * 1e7).to_i }
|
|
position_payload = {
|
|
"id" => 9_001,
|
|
"node_id" => node_id,
|
|
"node_num" => node_num,
|
|
"rx_time" => rx_time,
|
|
"rx_iso" => Time.at(rx_time).utc.iso8601,
|
|
"to_id" => "^all",
|
|
"latitude" => 52.5,
|
|
"longitude" => 13.4,
|
|
"altitude" => 42.0,
|
|
"position_time" => position_time,
|
|
"location_source" => "LOC_INTERNAL",
|
|
"precision_bits" => 15,
|
|
"sats_in_view" => 6,
|
|
"pdop" => 2.5,
|
|
"ground_speed" => 3.2,
|
|
"ground_track" => 180.0,
|
|
"snr" => -8.5,
|
|
"rssi" => -90,
|
|
"hop_limit" => 3,
|
|
"bitfield" => 1,
|
|
"payload_b64" => "AQI=",
|
|
"raw" => raw_payload,
|
|
}
|
|
|
|
post "/api/positions", position_payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row("SELECT * FROM positions WHERE id = ?", [9_001])
|
|
expect(row["node_id"]).to eq(node_id)
|
|
expect(row["node_num"]).to eq(node_num)
|
|
expect(row["rx_time"]).to eq(rx_time)
|
|
expect(row["rx_iso"]).to eq(Time.at(rx_time).utc.iso8601)
|
|
expect(row["position_time"]).to eq(position_time)
|
|
expect_same_value(row["latitude"], 52.5)
|
|
expect_same_value(row["longitude"], 13.4)
|
|
expect_same_value(row["altitude"], 42.0)
|
|
expect(row["location_source"]).to eq("LOC_INTERNAL")
|
|
expect(row["precision_bits"]).to eq(15)
|
|
expect(row["sats_in_view"]).to eq(6)
|
|
expect_same_value(row["pdop"], 2.5)
|
|
expect_same_value(row["ground_speed"], 3.2)
|
|
expect_same_value(row["ground_track"], 180.0)
|
|
expect_same_value(row["snr"], -8.5)
|
|
expect(row["rssi"]).to eq(-90)
|
|
expect(row["hop_limit"]).to eq(3)
|
|
expect(row["bitfield"]).to eq(1)
|
|
expect(row["payload_b64"]).to eq("AQI=")
|
|
end
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
node_row = db.get_first_row(
|
|
"SELECT last_heard, position_time, latitude, longitude, altitude, location_source, precision_bits, snr FROM nodes WHERE node_id = ?",
|
|
[node_id],
|
|
)
|
|
expect(node_row["last_heard"]).to eq(rx_time)
|
|
expect(node_row["position_time"]).to eq(position_time)
|
|
expect_same_value(node_row["latitude"], 52.5)
|
|
expect_same_value(node_row["longitude"], 13.4)
|
|
expect_same_value(node_row["altitude"], 42.0)
|
|
expect(node_row["location_source"]).to eq("LOC_INTERNAL")
|
|
expect(node_row["precision_bits"]).to eq(15)
|
|
expect_same_value(node_row["snr"], -8.5)
|
|
end
|
|
end
|
|
|
|
it "creates node records when none exist" do
|
|
node_id = "!specnew01"
|
|
node_num = 0xfeed_cafe
|
|
rx_time = reference_time.to_i - 60
|
|
position_time = rx_time - 10
|
|
payload = {
|
|
"id" => 9_002,
|
|
"node_id" => node_id,
|
|
"node_num" => node_num,
|
|
"rx_time" => rx_time,
|
|
"rx_iso" => Time.at(rx_time).utc.iso8601,
|
|
"latitude" => 52.1,
|
|
"longitude" => 13.1,
|
|
"altitude" => 33.0,
|
|
"position_time" => position_time,
|
|
"location_source" => "LOC_EXTERNAL",
|
|
}
|
|
|
|
post "/api/positions", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
node_row = db.get_first_row("SELECT * FROM nodes WHERE node_id = ?", [node_id])
|
|
expect(node_row).not_to be_nil
|
|
expect(node_row["num"]).to eq(node_num)
|
|
expect(node_row["last_heard"]).to eq(rx_time)
|
|
expect(node_row["first_heard"]).to eq(rx_time)
|
|
expect(node_row["position_time"]).to eq(position_time)
|
|
expect_same_value(node_row["latitude"], 52.1)
|
|
expect_same_value(node_row["longitude"], 13.1)
|
|
expect_same_value(node_row["altitude"], 33.0)
|
|
expect(node_row["location_source"]).to eq("LOC_EXTERNAL")
|
|
end
|
|
end
|
|
|
|
it "creates hidden nodes for unknown position senders" do
|
|
payload = {
|
|
"id" => 42,
|
|
"node_id" => "!0badc0de",
|
|
"rx_time" => reference_time.to_i,
|
|
"rx_iso" => reference_time.iso8601,
|
|
"latitude" => 52.1,
|
|
"longitude" => 13.1,
|
|
}
|
|
|
|
post "/api/positions", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT node_id, num, short_name, long_name, role FROM nodes WHERE node_id = ?",
|
|
["!0badc0de"],
|
|
)
|
|
|
|
expect(row).not_to be_nil
|
|
expect(row["node_id"]).to eq("!0badc0de")
|
|
expect(row["num"]).to eq(0x0badc0de)
|
|
expect(row["short_name"]).to eq("C0DE")
|
|
expect(row["long_name"]).to eq("Meshtastic C0DE")
|
|
expect(row["role"]).to eq("CLIENT_HIDDEN")
|
|
end
|
|
end
|
|
|
|
it "fills first_heard when updating an existing node without one" do
|
|
node_id = "!specposfh"
|
|
rx_time = reference_time.to_i - 90
|
|
|
|
with_db do |db|
|
|
db.execute(
|
|
"INSERT INTO nodes(node_id, last_heard, first_heard) VALUES (?,?,?)",
|
|
[node_id, nil, nil],
|
|
)
|
|
end
|
|
|
|
payload = {
|
|
"id" => 51,
|
|
"node_id" => node_id,
|
|
"rx_time" => rx_time,
|
|
"latitude" => 51.5,
|
|
"longitude" => -0.12,
|
|
}
|
|
|
|
post "/api/positions", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT last_heard, first_heard FROM nodes WHERE node_id = ?",
|
|
[node_id],
|
|
)
|
|
|
|
expect(row["last_heard"]).to eq(rx_time)
|
|
expect(row["first_heard"]).to eq(rx_time)
|
|
end
|
|
end
|
|
|
|
it "returns 400 when the payload is not valid JSON" do
|
|
post "/api/positions", "{", auth_headers
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "invalid JSON")
|
|
end
|
|
|
|
it "returns 400 when more than 1000 positions are provided" do
|
|
payload = Array.new(1001) { |i| { "id" => i + 1, "rx_time" => reference_time.to_i - i } }
|
|
|
|
post "/api/positions", payload.to_json, auth_headers
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "too many positions")
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM positions")
|
|
expect(count).to eq(0)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "POST /api/neighbors" do
|
|
it "stores neighbor tuples and updates node metadata" do
|
|
rx_time = reference_time.to_i - 120
|
|
neighbor_rx_time = rx_time - 30
|
|
payload = {
|
|
"node_id" => "!abc123ef",
|
|
"node_num" => 0xabc123ef,
|
|
"rx_time" => rx_time,
|
|
"neighbors" => [
|
|
{ "node_id" => "!00ff0011", "snr" => -7.5 },
|
|
{ "node_id" => 0x11223344, "snr" => 3.25, "rx_time" => neighbor_rx_time },
|
|
],
|
|
}
|
|
|
|
post "/api/neighbors", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
rows = db.execute(
|
|
"SELECT node_id, neighbor_id, snr, rx_time FROM neighbors ORDER BY neighbor_id",
|
|
)
|
|
|
|
expect(rows.size).to eq(2)
|
|
expect(rows[0]["node_id"]).to eq("!abc123ef")
|
|
expect(rows[0]["neighbor_id"]).to eq("!00ff0011")
|
|
expect_same_value(rows[0]["snr"], -7.5)
|
|
expect(rows[0]["rx_time"]).to eq(rx_time)
|
|
|
|
expect(rows[1]["node_id"]).to eq("!abc123ef")
|
|
expect(rows[1]["neighbor_id"]).to eq("!11223344")
|
|
expect_same_value(rows[1]["snr"], 3.25)
|
|
expect(rows[1]["rx_time"]).to eq(neighbor_rx_time)
|
|
end
|
|
|
|
get "/api/neighbors"
|
|
|
|
expect(last_response).to be_ok
|
|
neighbors = JSON.parse(last_response.body)
|
|
expect(neighbors.map { |row| row["neighbor_id"] }).to contain_exactly("!00ff0011", "!11223344")
|
|
expect(neighbors.first).to include("node_id" => "!abc123ef")
|
|
expect(neighbors.first["rx_iso"]).to be_a(String)
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
node_rows = db.execute(
|
|
"SELECT node_id, last_heard FROM nodes ORDER BY node_id",
|
|
)
|
|
|
|
expect(node_rows.size).to eq(3)
|
|
origin = node_rows.find { |row| row["node_id"] == "!abc123ef" }
|
|
expect(origin["last_heard"]).to eq(rx_time)
|
|
neighbor_one = node_rows.find { |row| row["node_id"] == "!00ff0011" }
|
|
expect(neighbor_one["last_heard"]).to eq(rx_time)
|
|
neighbor_two = node_rows.find { |row| row["node_id"] == "!11223344" }
|
|
expect(neighbor_two["last_heard"]).to eq(neighbor_rx_time)
|
|
end
|
|
end
|
|
|
|
it "handles broadcasts with no neighbors" do
|
|
rx_time = reference_time.to_i - 60
|
|
payload = {
|
|
"node_id" => "!cafebabe",
|
|
"rx_time" => rx_time,
|
|
"neighbors" => [],
|
|
}
|
|
|
|
post "/api/neighbors", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM neighbors")
|
|
expect(count).to eq(0)
|
|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT node_id, last_heard FROM nodes WHERE node_id = ?",
|
|
["!cafebabe"],
|
|
)
|
|
expect(row).not_to be_nil
|
|
expect(row["last_heard"]).to eq(rx_time)
|
|
end
|
|
|
|
get "/api/neighbors"
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to be_empty
|
|
end
|
|
|
|
it "returns 400 when more than 1000 neighbor packets are provided" do
|
|
payload = Array.new(1001) do |i|
|
|
{ "node_id" => format("!%08x", i), "rx_time" => reference_time.to_i - i }
|
|
end
|
|
|
|
post "/api/neighbors", payload.to_json, auth_headers
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "too many neighbor packets")
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM neighbors")
|
|
expect(count).to eq(0)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "POST /api/telemetry" do
|
|
it "stores telemetry packets and updates node metrics" do
|
|
payload = telemetry_fixture
|
|
|
|
post "/api/telemetry", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
rows = db.execute(
|
|
"SELECT * FROM telemetry ORDER BY id",
|
|
)
|
|
|
|
expect(rows.size).to eq(payload.size)
|
|
|
|
first = rows.find { |row| row["id"] == payload[0]["id"] }
|
|
expect(first).not_to be_nil
|
|
expect(first["node_id"]).to eq(payload[0]["node_id"])
|
|
expect(first["rx_time"]).to eq(payload[0]["rx_time"])
|
|
expect_same_value(first["battery_level"], payload[0]["battery_level"])
|
|
expect_same_value(first["voltage"], payload[0].dig("device_metrics", "voltage"))
|
|
expect_same_value(first["channel_utilization"], payload[0].dig("device_metrics", "channelUtilization"))
|
|
expect_same_value(first["air_util_tx"], payload[0].dig("device_metrics", "airUtilTx"))
|
|
expect(first["uptime_seconds"]).to eq(payload[0].dig("device_metrics", "uptimeSeconds"))
|
|
|
|
environment_row = rows.find { |row| row["id"] == payload[1]["id"] }
|
|
expect(environment_row["temperature"]).to be_within(1e-6).of(payload[1].dig("environment_metrics", "temperature"))
|
|
expect(environment_row["relative_humidity"]).to be_within(1e-6).of(payload[1].dig("environment_metrics", "relativeHumidity"))
|
|
expect(environment_row["barometric_pressure"]).to be_within(1e-6).of(payload[1].dig("environment_metrics", "barometricPressure"))
|
|
end
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
|
|
metrics_node = db.get_first_row(
|
|
"SELECT battery_level, voltage, channel_utilization, air_util_tx, uptime_seconds, last_heard, first_heard FROM nodes WHERE node_id = ?",
|
|
[payload[0]["node_id"]],
|
|
)
|
|
expect_same_value(metrics_node["battery_level"], payload[0]["battery_level"])
|
|
expect_same_value(metrics_node["voltage"], payload[0]["device_metrics"]["voltage"])
|
|
expect_same_value(metrics_node["channel_utilization"], payload[0]["device_metrics"]["channelUtilization"])
|
|
expect_same_value(metrics_node["air_util_tx"], payload[0]["device_metrics"]["airUtilTx"])
|
|
expect(metrics_node["uptime_seconds"]).to eq(payload[0]["device_metrics"]["uptimeSeconds"])
|
|
expect(metrics_node["last_heard"]).to eq(payload[0]["rx_time"])
|
|
expect(metrics_node["first_heard"]).to eq(payload[0]["rx_time"])
|
|
|
|
env_node = db.get_first_row(
|
|
"SELECT last_heard, battery_level, voltage FROM nodes WHERE node_id = ?",
|
|
[payload[1]["node_id"]],
|
|
)
|
|
expect(env_node["last_heard"]).to eq(payload[1]["rx_time"])
|
|
expect(env_node["battery_level"]).to be_nil
|
|
expect(env_node["voltage"]).to be_nil
|
|
|
|
local_node = db.get_first_row(
|
|
"SELECT battery_level, uptime_seconds, last_heard FROM nodes WHERE node_id = ?",
|
|
[payload[2]["node_id"]],
|
|
)
|
|
expect_same_value(local_node["battery_level"], payload[2]["device_metrics"]["battery_level"])
|
|
expect(local_node["uptime_seconds"]).to eq(payload[2]["device_metrics"]["uptime_seconds"])
|
|
expect(local_node["last_heard"]).to eq(payload[2]["rx_time"])
|
|
end
|
|
end
|
|
|
|
it "returns 400 when the payload is not valid JSON" do
|
|
post "/api/telemetry", "{", auth_headers
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "invalid JSON")
|
|
end
|
|
|
|
it "returns 400 when more than 1000 telemetry packets are provided" do
|
|
payload = Array.new(1001) { |i| { "id" => i + 1, "rx_time" => reference_time.to_i - i } }
|
|
|
|
post "/api/telemetry", payload.to_json, auth_headers
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "too many telemetry packets")
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM telemetry")
|
|
expect(count).to eq(0)
|
|
end
|
|
end
|
|
end
|
|
|
|
it "returns 400 when more than 1000 messages are provided" do
|
|
payload = Array.new(1001) { |i| { "packet_id" => i + 1 } }
|
|
|
|
post "/api/messages", payload.to_json, auth_headers
|
|
|
|
expect(last_response.status).to eq(400)
|
|
expect(JSON.parse(last_response.body)).to eq("error" => "too many messages")
|
|
|
|
with_db(readonly: true) do |db|
|
|
count = db.get_first_value("SELECT COUNT(*) FROM messages")
|
|
expect(count).to eq(0)
|
|
end
|
|
end
|
|
|
|
it "accepts array payloads, normalizes node references, and skips messages without an id" do
|
|
node_id = "!spec-normalized"
|
|
node_payload = {
|
|
node_id => {
|
|
"num" => 123,
|
|
"user" => { "shortName" => "Spec" },
|
|
"lastHeard" => reference_time.to_i - 60,
|
|
"position" => { "time" => reference_time.to_i - 120 },
|
|
},
|
|
}
|
|
|
|
post "/api/nodes", node_payload.to_json, auth_headers
|
|
expect(last_response).to be_ok
|
|
|
|
messages_payload = [
|
|
{
|
|
"packet_id" => 101,
|
|
"from_id" => "123",
|
|
"text" => "normalized",
|
|
},
|
|
{
|
|
"packet_id" => 102,
|
|
"from_id" => " ",
|
|
"text" => "blank",
|
|
},
|
|
{
|
|
"text" => "missing id",
|
|
},
|
|
]
|
|
|
|
post "/api/messages", messages_payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
rows = db.execute(
|
|
"SELECT id, from_id, to_id, rx_time, rx_iso, text, encrypted FROM messages ORDER BY id",
|
|
)
|
|
|
|
expect(rows.size).to eq(2)
|
|
|
|
first, second = rows
|
|
|
|
expect(first["id"]).to eq(101)
|
|
expect(first["from_id"]).to eq(node_id)
|
|
expect(first).not_to have_key("from_node_id")
|
|
expect(first).not_to have_key("from_node_num")
|
|
expect(first["rx_time"]).to eq(reference_time.to_i)
|
|
expect(first["rx_iso"]).to eq(reference_time.utc.iso8601)
|
|
expect(first["text"]).to eq("normalized")
|
|
expect(first).not_to have_key("to_node_id")
|
|
expect(first).not_to have_key("to_node_num")
|
|
expect(first["encrypted"]).to be_nil
|
|
|
|
expect(second["id"]).to eq(102)
|
|
expect(second["from_id"]).to be_nil
|
|
expect(second).not_to have_key("from_node_id")
|
|
expect(second).not_to have_key("from_node_num")
|
|
expect(second["rx_time"]).to eq(reference_time.to_i)
|
|
expect(second["rx_iso"]).to eq(reference_time.utc.iso8601)
|
|
expect(second["text"]).to eq("blank")
|
|
expect(second).not_to have_key("to_node_id")
|
|
expect(second).not_to have_key("to_node_num")
|
|
expect(second["encrypted"]).to be_nil
|
|
end
|
|
end
|
|
|
|
it "stores encrypted messages and resolves node references" do
|
|
sender_id = "!feedc0de"
|
|
sender_num = 0xfeedc0de
|
|
receiver_id = "!c0ffee99"
|
|
receiver_num = 0xc0ffee99
|
|
|
|
sender_node = {
|
|
"node_id" => sender_id,
|
|
"short_name" => "EncS",
|
|
"long_name" => "Encrypted Sender",
|
|
"hw_model" => "TEST",
|
|
"role" => "CLIENT",
|
|
"snr" => 5.5,
|
|
"battery_level" => 80.0,
|
|
"voltage" => 3.9,
|
|
"last_heard" => reference_time.to_i - 30,
|
|
"position_time" => reference_time.to_i - 60,
|
|
"latitude" => 52.1,
|
|
"longitude" => 13.1,
|
|
"altitude" => 42.0,
|
|
}
|
|
sender_payload = build_node_payload(sender_node)
|
|
sender_payload["num"] = sender_num
|
|
|
|
receiver_node = {
|
|
"node_id" => receiver_id,
|
|
"short_name" => "EncR",
|
|
"long_name" => "Encrypted Receiver",
|
|
"hw_model" => "TEST",
|
|
"role" => "CLIENT",
|
|
"snr" => 4.25,
|
|
"battery_level" => 75.0,
|
|
"voltage" => 3.8,
|
|
"last_heard" => reference_time.to_i - 40,
|
|
"position_time" => reference_time.to_i - 70,
|
|
"latitude" => 52.2,
|
|
"longitude" => 13.2,
|
|
"altitude" => 35.0,
|
|
}
|
|
receiver_payload = build_node_payload(receiver_node)
|
|
receiver_payload["num"] = receiver_num
|
|
|
|
post "/api/nodes", { sender_id => sender_payload }.to_json, auth_headers
|
|
expect(last_response).to be_ok
|
|
post "/api/nodes", { receiver_id => receiver_payload }.to_json, auth_headers
|
|
expect(last_response).to be_ok
|
|
|
|
encrypted_b64 = Base64.strict_encode64("secret message")
|
|
payload = {
|
|
"packet_id" => 777_001,
|
|
"rx_time" => reference_time.to_i,
|
|
"rx_iso" => reference_time.utc.iso8601,
|
|
"from_id" => sender_num.to_s,
|
|
"to_id" => receiver_id,
|
|
"channel" => 8,
|
|
"portnum" => "TEXT_MESSAGE_APP",
|
|
"encrypted" => encrypted_b64,
|
|
"snr" => -12.5,
|
|
"rssi" => -109,
|
|
"hop_limit" => 3,
|
|
}
|
|
|
|
post "/api/messages", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT from_id, to_id, text, encrypted FROM messages WHERE id = ?",
|
|
[777_001],
|
|
)
|
|
|
|
expect(row["from_id"]).to eq(sender_id)
|
|
expect(row["to_id"]).to eq(receiver_id)
|
|
expect(row["text"]).to be_nil
|
|
expect(row["encrypted"]).to eq(encrypted_b64)
|
|
|
|
node_row = db.get_first_row(
|
|
"SELECT last_heard FROM nodes WHERE node_id = ?",
|
|
[sender_id],
|
|
)
|
|
|
|
expect(node_row["last_heard"]).to eq(payload["rx_time"])
|
|
end
|
|
|
|
get "/api/messages"
|
|
expect(last_response).to be_ok
|
|
|
|
messages = JSON.parse(last_response.body)
|
|
expect(messages).to be_an(Array)
|
|
expect(messages).to be_empty
|
|
end
|
|
|
|
it "updates node last_heard for plaintext messages" do
|
|
node_id = "!plainmsg01"
|
|
initial_first = reference_time.to_i - 600
|
|
initial_last = reference_time.to_i - 300
|
|
|
|
with_db do |db|
|
|
db.execute(
|
|
"INSERT INTO nodes(node_id, last_heard, first_heard) VALUES (?,?,?)",
|
|
[node_id, initial_last, initial_first],
|
|
)
|
|
end
|
|
|
|
rx_time = reference_time.to_i - 120
|
|
payload = {
|
|
"packet_id" => 888_001,
|
|
"rx_time" => rx_time,
|
|
"rx_iso" => Time.at(rx_time).utc.iso8601,
|
|
"from_id" => node_id,
|
|
"text" => "plaintext update",
|
|
}
|
|
|
|
post "/api/messages", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT last_heard, first_heard FROM nodes WHERE node_id = ?",
|
|
[node_id],
|
|
)
|
|
|
|
expect(row["last_heard"]).to eq(rx_time)
|
|
expect(row["first_heard"]).to eq(initial_first)
|
|
end
|
|
end
|
|
|
|
it "stores messages containing SQL control characters without executing them" do
|
|
payload = {
|
|
"packet_id" => 404,
|
|
"from_id" => "attacker",
|
|
"text" => "'); DROP TABLE nodes;--",
|
|
}
|
|
|
|
post "/api/messages", payload.to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row(
|
|
"SELECT id, text FROM messages WHERE id = ?",
|
|
[404],
|
|
)
|
|
|
|
expect(row["id"]).to eq(404)
|
|
expect(row["text"]).to eq("'); DROP TABLE nodes;--")
|
|
|
|
tables = db.get_first_value(
|
|
"SELECT COUNT(*) FROM sqlite_master WHERE type='table' AND name='nodes'",
|
|
)
|
|
expect(tables).to eq(1)
|
|
end
|
|
end
|
|
|
|
it "updates existing messages only when sender information is provided" do
|
|
message_id = 9001
|
|
initial_time = reference_time.to_i - 120
|
|
initial_iso = Time.at(initial_time).utc.iso8601
|
|
base_payload = {
|
|
"packet_id" => message_id,
|
|
"rx_time" => initial_time,
|
|
"rx_iso" => initial_iso,
|
|
"to_id" => "^all",
|
|
"channel" => 1,
|
|
"portnum" => "TEXT_MESSAGE_APP",
|
|
"text" => "initial payload",
|
|
"snr" => 7.25,
|
|
"rssi" => -58,
|
|
"hop_limit" => 2,
|
|
}
|
|
|
|
post "/api/messages", base_payload.merge("from_id" => nil).to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row("SELECT id, from_id, rx_time, rx_iso, text FROM messages WHERE id = ?", [message_id])
|
|
|
|
expect(row["from_id"]).to be_nil
|
|
expect(row["rx_time"]).to eq(initial_time)
|
|
expect(row["rx_iso"]).to eq(initial_iso)
|
|
expect(row["text"]).to eq("initial payload")
|
|
end
|
|
|
|
updated_time = initial_time + 60
|
|
updated_iso = Time.at(updated_time).utc.iso8601
|
|
post "/api/messages", base_payload.merge(
|
|
"rx_time" => updated_time,
|
|
"rx_iso" => updated_iso,
|
|
"text" => "overwritten without sender",
|
|
"from_id" => " ",
|
|
).to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row("SELECT id, from_id, rx_time, rx_iso, text FROM messages WHERE id = ?", [message_id])
|
|
|
|
expect(row["from_id"]).to be_nil
|
|
expect(row["rx_time"]).to eq(initial_time)
|
|
expect(row["rx_iso"]).to eq(initial_iso)
|
|
expect(row["text"]).to eq("initial payload")
|
|
end
|
|
|
|
final_time = updated_time + 30
|
|
final_iso = Time.at(final_time).utc.iso8601
|
|
post "/api/messages", base_payload.merge(
|
|
"rx_time" => final_time,
|
|
"rx_iso" => final_iso,
|
|
"from" => "!spec-sender",
|
|
).to_json, auth_headers
|
|
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
with_db(readonly: true) do |db|
|
|
db.results_as_hash = true
|
|
row = db.get_first_row("SELECT id, from_id, rx_time, rx_iso, text FROM messages WHERE id = ?", [message_id])
|
|
|
|
expect(row["from_id"]).to eq("!spec-sender")
|
|
expect(row["rx_time"]).to eq(initial_time)
|
|
expect(row["rx_iso"]).to eq(initial_iso)
|
|
expect(row["text"]).to eq("initial payload")
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "GET /api/nodes" do
|
|
it "returns the stored nodes with derived timestamps" do
|
|
import_nodes_fixture
|
|
|
|
get "/api/nodes"
|
|
expect(last_response).to be_ok
|
|
|
|
actual = JSON.parse(last_response.body)
|
|
expect(actual.size).to eq(nodes_fixture.size)
|
|
|
|
actual_by_id = actual.each_with_object({}) do |row, acc|
|
|
acc[row["node_id"]] = row
|
|
end
|
|
|
|
nodes_fixture.each do |node|
|
|
expected = expected_node_row(node)
|
|
actual_row = actual_by_id.fetch(node["node_id"])
|
|
|
|
expect(actual_row["short_name"]).to eq(expected["short_name"])
|
|
expect(actual_row["long_name"]).to eq(expected["long_name"])
|
|
expect(actual_row["hw_model"]).to eq(expected["hw_model"])
|
|
expect(actual_row["role"]).to eq(expected["role"])
|
|
expect_same_value(actual_row["snr"], expected["snr"])
|
|
expect_same_value(actual_row["battery_level"], expected["battery_level"])
|
|
expect_same_value(actual_row["voltage"], expected["voltage"])
|
|
expect(actual_row["last_heard"]).to eq(expected["last_heard"])
|
|
expect(actual_row["first_heard"]).to eq(expected["first_heard"])
|
|
expect_same_value(actual_row["uptime_seconds"], expected["uptime_seconds"])
|
|
expect_same_value(actual_row["channel_utilization"], expected["channel_utilization"])
|
|
expect_same_value(actual_row["air_util_tx"], expected["air_util_tx"])
|
|
expect_same_value(actual_row["position_time"], expected["position_time"])
|
|
expect(actual_row["location_source"]).to eq(expected["location_source"])
|
|
expect_same_value(actual_row["precision_bits"], expected["precision_bits"])
|
|
expect_same_value(actual_row["latitude"], expected["latitude"])
|
|
expect_same_value(actual_row["longitude"], expected["longitude"])
|
|
expect_same_value(actual_row["altitude"], expected["altitude"])
|
|
|
|
if expected["last_heard"]
|
|
expected_last_seen_iso = Time.at(expected["last_heard"]).utc.iso8601
|
|
expect(actual_row["last_seen_iso"]).to eq(expected_last_seen_iso)
|
|
else
|
|
expect(actual_row["last_seen_iso"]).to be_nil
|
|
end
|
|
|
|
if node["position_time"]
|
|
expected_pos_iso = Time.at(node["position_time"]).utc.iso8601
|
|
expect(actual_row["pos_time_iso"]).to eq(expected_pos_iso)
|
|
else
|
|
expect(actual_row).not_to have_key("pos_time_iso")
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "GET /api/messages" do
|
|
it "returns the stored messages along with joined node data" do
|
|
import_nodes_fixture
|
|
import_messages_fixture
|
|
|
|
get "/api/messages"
|
|
expect(last_response).to be_ok
|
|
|
|
actual = JSON.parse(last_response.body)
|
|
expect(actual.size).to eq(messages_fixture.size)
|
|
|
|
actual_by_id = actual.each_with_object({}) do |row, acc|
|
|
acc[row["id"]] = row
|
|
end
|
|
|
|
nodes_by_id = {}
|
|
node_aliases = {}
|
|
|
|
nodes_fixture.each do |node|
|
|
node_id = node["node_id"]
|
|
expected_row = expected_node_row(node)
|
|
nodes_by_id[node_id] = expected_row
|
|
|
|
if (num = node["num"])
|
|
node_aliases[num.to_s] = node_id
|
|
end
|
|
end
|
|
|
|
messages_fixture.each do |message|
|
|
node = message["node"]
|
|
next unless node.is_a?(Hash)
|
|
|
|
canonical = node["node_id"]
|
|
num = node["num"]
|
|
next unless canonical && num
|
|
|
|
node_aliases[num.to_s] ||= canonical
|
|
end
|
|
|
|
latest_rx_by_node = {}
|
|
messages_fixture.each do |message|
|
|
rx_time = message["rx_time"]
|
|
next unless rx_time
|
|
|
|
canonical = nil
|
|
from_id = message["from_id"]
|
|
|
|
if from_id.is_a?(String)
|
|
trimmed = from_id.strip
|
|
unless trimmed.empty?
|
|
if trimmed.match?(/\A[0-9]+\z/)
|
|
canonical = node_aliases[trimmed] || trimmed
|
|
else
|
|
canonical = trimmed
|
|
end
|
|
end
|
|
end
|
|
|
|
canonical ||= message.dig("node", "node_id")
|
|
|
|
if canonical.nil?
|
|
num = message.dig("node", "num")
|
|
canonical = node_aliases[num.to_s] if num
|
|
end
|
|
|
|
next unless canonical
|
|
|
|
existing = latest_rx_by_node[canonical]
|
|
latest_rx_by_node[canonical] = [existing, rx_time].compact.max
|
|
end
|
|
|
|
messages_fixture.each do |message|
|
|
expected = message.reject { |key, _| key == "node" }
|
|
actual_row = actual_by_id.fetch(message["id"])
|
|
|
|
expect(actual_row["rx_time"]).to eq(expected["rx_time"])
|
|
expect(actual_row["rx_iso"]).to eq(expected["rx_iso"])
|
|
|
|
expected_from_id = expected["from_id"]
|
|
if expected_from_id.is_a?(String) && expected_from_id.match?(/\A[0-9]+\z/)
|
|
expected_from_id = node_aliases[expected_from_id] || expected_from_id
|
|
elsif expected_from_id.nil?
|
|
expected_from_id = message.dig("node", "node_id")
|
|
end
|
|
expect(actual_row["from_id"]).to eq(expected_from_id)
|
|
expect(actual_row).not_to have_key("from_node_id")
|
|
expect(actual_row).not_to have_key("from_node_num")
|
|
|
|
expected_to_id = expected["to_id"]
|
|
if expected_to_id.is_a?(String) && expected_to_id.match?(/\A[0-9]+\z/)
|
|
expected_to_id = node_aliases[expected_to_id] || expected_to_id
|
|
end
|
|
expect(actual_row["to_id"]).to eq(expected_to_id)
|
|
expect(actual_row).not_to have_key("to_node_id")
|
|
expect(actual_row).not_to have_key("to_node_num")
|
|
expect(actual_row["channel"]).to eq(expected["channel"])
|
|
expect(actual_row["portnum"]).to eq(expected["portnum"])
|
|
expect(actual_row["text"]).to eq(expected["text"])
|
|
expect(actual_row["encrypted"]).to eq(expected["encrypted"])
|
|
expect_same_value(actual_row["snr"], expected["snr"])
|
|
expect(actual_row["rssi"]).to eq(expected["rssi"])
|
|
expect(actual_row["hop_limit"]).to eq(expected["hop_limit"])
|
|
|
|
if expected["from_id"]
|
|
lookup_id = expected["from_id"]
|
|
node_expected = nodes_by_id[lookup_id]
|
|
|
|
unless node_expected
|
|
canonical_id = node_aliases[lookup_id.to_s]
|
|
expect(canonical_id).not_to be_nil,
|
|
"node fixture missing for from_id #{lookup_id.inspect}"
|
|
node_expected = nodes_by_id.fetch(canonical_id)
|
|
end
|
|
|
|
node_actual = actual_row.fetch("node")
|
|
|
|
expect(node_actual["node_id"]).to eq(node_expected["node_id"])
|
|
expect(node_actual["short_name"]).to eq(node_expected["short_name"])
|
|
expect(node_actual["long_name"]).to eq(node_expected["long_name"])
|
|
expect(node_actual["role"]).to eq(node_expected["role"])
|
|
expect_same_value(node_actual["snr"], node_expected["snr"])
|
|
expect_same_value(node_actual["battery_level"], node_expected["battery_level"])
|
|
expect_same_value(node_actual["voltage"], node_expected["voltage"])
|
|
expected_last_heard = node_expected["last_heard"]
|
|
latest_rx = latest_rx_by_node[node_expected["node_id"]]
|
|
if latest_rx
|
|
expected_last_heard = [expected_last_heard, latest_rx].compact.max
|
|
end
|
|
expect(node_actual["last_heard"]).to eq(expected_last_heard)
|
|
expect(node_actual["first_heard"]).to eq(node_expected["first_heard"])
|
|
expect_same_value(node_actual["latitude"], node_expected["latitude"])
|
|
expect_same_value(node_actual["longitude"], node_expected["longitude"])
|
|
expect_same_value(node_actual["altitude"], node_expected["altitude"])
|
|
else
|
|
expect(actual_row["node"]).to be_a(Hash)
|
|
expect(actual_row["node"]["node_id"]).to be_nil
|
|
end
|
|
end
|
|
end
|
|
|
|
context "when DEBUG logging is enabled" do
|
|
it "logs diagnostics for messages missing a sender" do
|
|
allow(PotatoMesh::Config).to receive(:debug?).and_return(true)
|
|
allow(Kernel).to receive(:warn)
|
|
|
|
message_id = 987_654
|
|
payload = {
|
|
"packet_id" => message_id,
|
|
"from_id" => " ",
|
|
"text" => "debug logging",
|
|
}
|
|
|
|
post "/api/messages", payload.to_json, auth_headers
|
|
expect(last_response).to be_ok
|
|
expect(JSON.parse(last_response.body)).to eq("status" => "ok")
|
|
|
|
get "/api/messages"
|
|
expect(last_response).to be_ok
|
|
|
|
expect(Kernel).to have_received(:warn).with(
|
|
a_string_matching(/\[debug\] messages row before join: .*"id"\s*=>\s*#{message_id}/),
|
|
)
|
|
expect(Kernel).to have_received(:warn).with(
|
|
a_string_matching(/\[debug\] row after join: .*"id"\s*=>\s*#{message_id}/),
|
|
)
|
|
expect(Kernel).to have_received(:warn).with(
|
|
a_string_matching(/\[debug\] row after processing: .*"id"\s*=>\s*#{message_id}/),
|
|
)
|
|
|
|
messages = JSON.parse(last_response.body)
|
|
expect(messages.size).to eq(1)
|
|
expect(messages.first["from_id"]).to be_nil
|
|
end
|
|
end
|
|
end
|
|
|
|
context "when private mode is enabled" do
|
|
before do
|
|
ENV["PRIVATE"] = "1"
|
|
end
|
|
|
|
it "returns 404 for GET /api/messages" do
|
|
get "/api/messages"
|
|
expect(last_response.status).to eq(404)
|
|
end
|
|
|
|
it "returns 404 for POST /api/messages" do
|
|
post "/api/messages", {}.to_json, auth_headers
|
|
expect(last_response.status).to eq(404)
|
|
end
|
|
|
|
it "excludes hidden clients from the nodes API" do
|
|
now = reference_time.to_i
|
|
with_db do |db|
|
|
db.execute(
|
|
"INSERT INTO nodes(node_id, short_name, long_name, hw_model, role, snr, last_heard, first_heard) VALUES(?,?,?,?,?,?,?,?)",
|
|
["!hidden", "hidn", "Hidden", "TBEAM", "CLIENT_HIDDEN", 0.0, now, now],
|
|
)
|
|
db.execute(
|
|
"INSERT INTO nodes(node_id, short_name, long_name, hw_model, role, snr, last_heard, first_heard) VALUES(?,?,?,?,?,?,?,?)",
|
|
["!visible", "vis", "Visible", "TBEAM", "CLIENT", 1.0, now, now],
|
|
)
|
|
end
|
|
|
|
get "/api/nodes?limit=10"
|
|
|
|
expect(last_response).to be_ok
|
|
nodes = JSON.parse(last_response.body)
|
|
ids = nodes.map { |node| node["node_id"] }
|
|
expect(ids).to include("!visible")
|
|
expect(ids).not_to include("!hidden")
|
|
end
|
|
|
|
it "removes the chat interface from the homepage" do
|
|
get "/"
|
|
|
|
expect(last_response).to be_ok
|
|
body = last_response.body
|
|
expect(body).not_to include('<div id="chat"')
|
|
expect(body).to include("const CHAT_ENABLED = false;")
|
|
expect(body).not_to include("Track nodes, messages, and coverage in real time.")
|
|
expect(body).to include("Track nodes and coverage in real time.")
|
|
end
|
|
end
|
|
|
|
describe "GET /api/positions" do
|
|
it "returns stored positions ordered by receive time" do
|
|
node_id = "!specfetch"
|
|
rx_times = [reference_time.to_i - 50, reference_time.to_i - 10]
|
|
rx_times.each_with_index do |rx_time, idx|
|
|
payload = {
|
|
"id" => 20_000 + idx,
|
|
"node_id" => node_id,
|
|
"rx_time" => rx_time,
|
|
"rx_iso" => Time.at(rx_time).utc.iso8601,
|
|
"position_time" => rx_time - 5,
|
|
"latitude" => 52.0 + idx,
|
|
"longitude" => 13.0 + idx,
|
|
"location_source" => "LOC_TEST",
|
|
"precision_bits" => 7 + idx,
|
|
"payload_b64" => "AQI=",
|
|
}
|
|
post "/api/positions", payload.to_json, auth_headers
|
|
expect(last_response).to be_ok
|
|
end
|
|
|
|
get "/api/positions?limit=1"
|
|
|
|
expect(last_response).to be_ok
|
|
data = JSON.parse(last_response.body)
|
|
expect(data.length).to eq(1)
|
|
entry = data.first
|
|
expect(entry["id"]).to eq(20_001)
|
|
expect(entry["node_id"]).to eq(node_id)
|
|
expect(entry["rx_time"]).to eq(rx_times.last)
|
|
expect(entry["rx_iso"]).to eq(Time.at(rx_times.last).utc.iso8601)
|
|
expect(entry["position_time"]).to eq(rx_times.last - 5)
|
|
expect(entry["position_time_iso"]).to eq(Time.at(rx_times.last - 5).utc.iso8601)
|
|
expect(entry["latitude"]).to eq(53.0)
|
|
expect(entry["longitude"]).to eq(14.0)
|
|
expect(entry["location_source"]).to eq("LOC_TEST")
|
|
expect(entry["precision_bits"]).to eq(8)
|
|
expect(entry["payload_b64"]).to eq("AQI=")
|
|
end
|
|
end
|
|
|
|
describe "GET /api/telemetry" do
|
|
it "returns stored telemetry ordered by receive time" do
|
|
post "/api/telemetry", telemetry_fixture.to_json, auth_headers
|
|
expect(last_response).to be_ok
|
|
|
|
get "/api/telemetry?limit=2"
|
|
|
|
expect(last_response).to be_ok
|
|
data = JSON.parse(last_response.body)
|
|
expect(data.length).to eq(2)
|
|
|
|
latest = telemetry_fixture.max_by { |entry| entry["rx_time"] }
|
|
second_latest = telemetry_fixture.sort_by { |entry| entry["rx_time"] }[-2]
|
|
|
|
first_entry = data.first
|
|
expect(first_entry["id"]).to eq(latest["id"])
|
|
expect(first_entry["node_id"]).to eq(latest["node_id"])
|
|
expect(first_entry["rx_time"]).to eq(latest["rx_time"])
|
|
expect(first_entry["telemetry_time"]).to eq(latest["telemetry_time"])
|
|
expect(first_entry["telemetry_time_iso"]).to eq(Time.at(latest["telemetry_time"]).utc.iso8601)
|
|
expect(first_entry).not_to have_key("device_metrics")
|
|
expect_same_value(first_entry["battery_level"], latest.dig("device_metrics", "battery_level") || latest.dig("device_metrics", "batteryLevel"))
|
|
|
|
second_entry = data.last
|
|
expect(second_entry["id"]).to eq(second_latest["id"])
|
|
expect(second_entry).not_to have_key("environment_metrics")
|
|
expect(second_entry["temperature"]).to be_within(1e-6).of(second_latest["environment_metrics"]["temperature"])
|
|
expect(second_entry["relative_humidity"]).to be_within(1e-6).of(second_latest["environment_metrics"]["relativeHumidity"])
|
|
expect(second_entry["barometric_pressure"]).to be_within(1e-6).of(second_latest["environment_metrics"]["barometricPressure"])
|
|
end
|
|
end
|
|
end
|