Files
potato-mesh/app/test/mesh_message_test.dart
T
l5y e733a0dd13 web: add ?since= parameter to api/messages (#512)
* web: add ?since= parameter to api/messages

* app: add ?since= parameter to api/messages
2025-11-25 22:11:20 +01:00

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);
});
});
}