mirror of
https://github.com/l5yth/potato-mesh.git
synced 2026-07-05 17:31:30 +02:00
e733a0dd13
* web: add ?since= parameter to api/messages * app: add ?since= parameter to api/messages
242 lines
6.8 KiB
Dart
242 lines
6.8 KiB
Dart
// Copyright © 2025-26 l5yth & contributors
|
|
//
|
|
// 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.
|
|
|
|
import 'dart:convert';
|
|
|
|
import 'package:flutter_test/flutter_test.dart';
|
|
import 'package:http/http.dart' as http;
|
|
import 'package:http/testing.dart';
|
|
import 'package:potato_mesh_reader/main.dart';
|
|
import 'package:shared_preferences/shared_preferences.dart';
|
|
|
|
/// Unit tests for [MeshMessage] parsing and the sorting helper.
|
|
void main() {
|
|
TestWidgetsFlutterBinding.ensureInitialized();
|
|
|
|
setUp(() {
|
|
SharedPreferences.setMockInitialValues({});
|
|
NodeShortNameCache.instance.clear();
|
|
});
|
|
|
|
group('MeshMessage.fromJson', () {
|
|
test('parses fields and strips leading bang from sender', () {
|
|
final msg = MeshMessage.fromJson({
|
|
'id': '7',
|
|
'rx_iso': '2024-01-02T03:04:00Z',
|
|
'from_id': '!NICK',
|
|
'to_id': '^',
|
|
'channel': '1',
|
|
'channel_name': 'BerlinMesh',
|
|
'portnum': 'TEXT',
|
|
'text': 'Hello world',
|
|
'rssi': '-90',
|
|
'snr': '5.25',
|
|
'hop_limit': '3',
|
|
});
|
|
|
|
expect(msg.id, 7);
|
|
expect(msg.rxIso, '2024-01-02T03:04:00Z');
|
|
expect(msg.rxTime!.toUtc().hour, 3);
|
|
expect(msg.fromShort, 'NICK');
|
|
expect(msg.channelName, 'BerlinMesh');
|
|
expect(msg.text, 'Hello world');
|
|
expect(msg.rssi, -90);
|
|
expect(msg.snr, closeTo(5.25, 0.0001));
|
|
expect(msg.hopLimit, 3);
|
|
});
|
|
|
|
test('handles invalid timestamps and non-numeric fields', () {
|
|
final msg = MeshMessage.fromJson({
|
|
'id': null,
|
|
'rx_iso': 'not-a-date',
|
|
'from_id': '',
|
|
'to_id': '',
|
|
'channel': 'abc',
|
|
'portnum': 'TEXT',
|
|
'text': '',
|
|
'rssi': 'missing',
|
|
'snr': 'noise',
|
|
'hop_limit': null,
|
|
});
|
|
|
|
expect(msg.id, 0);
|
|
expect(msg.rxTime, isNull);
|
|
expect(msg.timeFormatted, '--:--');
|
|
expect(msg.fromShort, '?');
|
|
expect(msg.channel, isNull);
|
|
expect(msg.rssi, isNull);
|
|
expect(msg.snr, isNull);
|
|
expect(msg.hopLimit, isNull);
|
|
expect(msg.text, '');
|
|
});
|
|
});
|
|
|
|
group('sortMessagesByRxTime', () {
|
|
test('orders messages oldest to newest even with null timestamps', () {
|
|
final older = MeshMessage(
|
|
id: 1,
|
|
rxTime: DateTime.utc(2023, 12, 31, 23, 59),
|
|
rxIso: '2023-12-31T23:59:00Z',
|
|
fromId: 'A',
|
|
toId: 'B',
|
|
channel: 1,
|
|
channelName: 'Main',
|
|
portnum: 'TEXT',
|
|
text: 'Old',
|
|
rssi: -50,
|
|
snr: 1.0,
|
|
hopLimit: 1,
|
|
);
|
|
final unknownTime = MeshMessage(
|
|
id: 2,
|
|
rxTime: null,
|
|
rxIso: '',
|
|
fromId: 'B',
|
|
toId: 'A',
|
|
channel: 1,
|
|
channelName: 'Main',
|
|
portnum: 'TEXT',
|
|
text: 'Unknown',
|
|
rssi: -55,
|
|
snr: 1.5,
|
|
hopLimit: 1,
|
|
);
|
|
final newer = MeshMessage(
|
|
id: 3,
|
|
rxTime: DateTime.utc(2024, 01, 01, 0, 10),
|
|
rxIso: '2024-01-01T00:10:00Z',
|
|
fromId: 'C',
|
|
toId: 'D',
|
|
channel: 1,
|
|
channelName: 'Main',
|
|
portnum: 'TEXT',
|
|
text: 'New',
|
|
rssi: -60,
|
|
snr: 2.0,
|
|
hopLimit: 1,
|
|
);
|
|
|
|
final sorted = sortMessagesByRxTime([newer, unknownTime, older]);
|
|
|
|
expect(sorted.first.id, older.id);
|
|
expect(sorted.last.id, newer.id);
|
|
expect(sorted[1].id, unknownTime.id);
|
|
});
|
|
});
|
|
|
|
group('fetchMessages', () {
|
|
test('parses, sorts, and returns API messages', () async {
|
|
final calls = <Uri>[];
|
|
final client = MockClient((request) async {
|
|
calls.add(request.url);
|
|
return http.Response(
|
|
jsonEncode([
|
|
{
|
|
'id': 2,
|
|
'rx_iso': '2024-01-02T00:01:00Z',
|
|
'from_id': '!b',
|
|
'to_id': '^',
|
|
'channel': 1,
|
|
'portnum': 'TEXT',
|
|
'text': 'Later'
|
|
},
|
|
{
|
|
'id': 1,
|
|
'rx_iso': '2024-01-01T23:59:00Z',
|
|
'from_id': '!a',
|
|
'to_id': '^',
|
|
'channel': 1,
|
|
'portnum': 'TEXT',
|
|
'text': 'Earlier'
|
|
},
|
|
]),
|
|
200,
|
|
);
|
|
});
|
|
|
|
final messages = await fetchMessages(client: client);
|
|
|
|
expect(calls.single.queryParameters['limit'], '1000');
|
|
expect(calls.single.queryParameters['since'], '0');
|
|
expect(messages.first.id, 1);
|
|
expect(messages.last.id, 2);
|
|
expect(messages.first.fromShort, 'a');
|
|
});
|
|
|
|
test('throws on non-200 responses', () async {
|
|
final client = MockClient((request) async => http.Response('nope', 500));
|
|
|
|
expect(
|
|
() => fetchMessages(client: client),
|
|
throwsA(isA<Exception>()),
|
|
);
|
|
});
|
|
|
|
test('throws on unexpected response shapes', () async {
|
|
final client =
|
|
MockClient((request) async => http.Response('{"id":1}', 200));
|
|
|
|
expect(
|
|
() => fetchMessages(client: client),
|
|
throwsA(isA<Exception>()),
|
|
);
|
|
});
|
|
|
|
test('uses custom domains including full URLs', () async {
|
|
final calls = <Uri>[];
|
|
final client = MockClient((request) async {
|
|
calls.add(request.url);
|
|
return http.Response(jsonEncode([]), 200);
|
|
});
|
|
|
|
await fetchMessages(client: client, domain: 'mesh.example.org');
|
|
await fetchMessages(
|
|
client: client, domain: 'https://mesh.alt.example/api');
|
|
|
|
expect(calls[0].host, 'mesh.example.org');
|
|
expect(calls[0].path, '/api/messages');
|
|
expect(calls[1].scheme, 'https');
|
|
expect(calls[1].path, '/api/messages');
|
|
});
|
|
});
|
|
|
|
group('fetchInstances', () {
|
|
test('parses and sorts instance list', () async {
|
|
final client = MockClient((request) async {
|
|
return http.Response(
|
|
jsonEncode([
|
|
{'name': 'Bravo', 'domain': 'bravo.example'},
|
|
{'name': 'Alpha', 'domain': 'alpha.example'},
|
|
{'name': '', 'domain': ''},
|
|
]),
|
|
200,
|
|
);
|
|
});
|
|
|
|
final instances = await fetchInstances(client: client);
|
|
|
|
expect(instances.map((i) => i.displayName), ['Alpha', 'Bravo']);
|
|
expect(
|
|
instances.map((i) => i.domain), ['alpha.example', 'bravo.example']);
|
|
});
|
|
|
|
test('returns empty list on failed fetch', () async {
|
|
final client = MockClient((request) async => http.Response('oops', 500));
|
|
|
|
final instances = await fetchInstances(client: client);
|
|
expect(instances, isEmpty);
|
|
});
|
|
});
|
|
}
|