mirror of
https://github.com/kuhyx/diet-guard.git
synced 2026-07-04 13:43:30 +02:00
Work-in-progress feature set accumulated ahead of the log-meal compact layout change: a food bank browser, a shared entry-edit screen, an app-wide settings service, and a substantially reworked history screen with filtering/sorting. Co-Authored-By: Claude Sonnet 5 <noreply@anthropic.com> Claude-Session: https://claude.ai/code/session_018UorgLvWJ4huH55tmXoUAZ
395 lines
12 KiB
Dart
395 lines
12 KiB
Dart
import 'dart:io';
|
|
|
|
import 'package:diet_guard_app/models/food_bank_record.dart';
|
|
import 'package:diet_guard_app/models/food_entry.dart';
|
|
import 'package:diet_guard_app/models/meal_component.dart';
|
|
import 'package:diet_guard_app/services/foodbank_service.dart';
|
|
import 'package:diet_guard_app/services/log_storage_service.dart';
|
|
import 'package:flutter_test/flutter_test.dart';
|
|
|
|
FoodEntry _entry({
|
|
required String id,
|
|
required String time,
|
|
required String desc,
|
|
double kcal = 100,
|
|
List<MealComponent>? components,
|
|
bool deleted = false,
|
|
}) => FoodEntry(
|
|
id: id,
|
|
time: time,
|
|
desc: desc,
|
|
grams: 100,
|
|
kcal: kcal,
|
|
proteinG: 1,
|
|
carbsG: 1,
|
|
fatG: 1,
|
|
source: components != null ? 'meal' : 'manual',
|
|
components: components,
|
|
deleted: deleted,
|
|
);
|
|
|
|
void main() {
|
|
group('FoodBankService.rebuild (pure)', () {
|
|
test('upserts a single-food entry by normalized name', () {
|
|
final log = {
|
|
'2026-06-22': [
|
|
_entry(id: '1', time: '2026-06-22T08:00:00+02:00', desc: 'Toast'),
|
|
],
|
|
};
|
|
final bank = FoodBankService.rebuild(log);
|
|
expect(bank['toast']!.desc, 'Toast');
|
|
expect(bank['toast']!.count, 1);
|
|
});
|
|
|
|
test('increments count and keeps the latest macros on repeat', () {
|
|
final log = {
|
|
'2026-06-22': [
|
|
_entry(
|
|
id: '1',
|
|
time: '2026-06-22T08:00:00+02:00',
|
|
desc: 'toast',
|
|
kcal: 100,
|
|
),
|
|
_entry(
|
|
id: '2',
|
|
time: '2026-06-22T12:00:00+02:00',
|
|
desc: 'toast',
|
|
kcal: 120,
|
|
),
|
|
],
|
|
};
|
|
final bank = FoodBankService.rebuild(log);
|
|
expect(bank['toast']!.count, 2);
|
|
expect(bank['toast']!.kcal, 120);
|
|
});
|
|
|
|
test('skips a tombstoned entry entirely', () {
|
|
final log = {
|
|
'2026-06-22': [
|
|
_entry(
|
|
id: '1',
|
|
time: '2026-06-22T08:00:00+02:00',
|
|
desc: 'toast',
|
|
deleted: true,
|
|
),
|
|
],
|
|
};
|
|
expect(FoodBankService.rebuild(log), isEmpty);
|
|
});
|
|
|
|
test('records a composite entry\'s components and total separately', () {
|
|
const components = [
|
|
MealComponent(
|
|
name: 'rice',
|
|
kcal: 200,
|
|
proteinG: 4,
|
|
carbsG: 44,
|
|
fatG: 1,
|
|
grams: 150,
|
|
),
|
|
];
|
|
final log = {
|
|
'2026-06-22': [
|
|
_entry(
|
|
id: '1',
|
|
time: '2026-06-22T20:00:00+02:00',
|
|
desc: 'dinner',
|
|
kcal: 200,
|
|
components: components,
|
|
),
|
|
],
|
|
};
|
|
final bank = FoodBankService.rebuild(log);
|
|
expect(bank['rice'], isNotNull);
|
|
expect(bank['dinner']!.components, ['rice']);
|
|
});
|
|
|
|
test('orders replay by time then id so two devices converge', () {
|
|
final logA = {
|
|
'2026-06-22': [
|
|
_entry(id: 'b', time: '2026-06-22T08:00:00+02:00', desc: 'x'),
|
|
_entry(id: 'a', time: '2026-06-22T08:00:00+02:00', desc: 'x'),
|
|
],
|
|
};
|
|
final logB = {
|
|
'2026-06-22': [
|
|
_entry(id: 'a', time: '2026-06-22T08:00:00+02:00', desc: 'x'),
|
|
_entry(id: 'b', time: '2026-06-22T08:00:00+02:00', desc: 'x'),
|
|
],
|
|
};
|
|
expect(
|
|
FoodBankService.rebuild(logA)['x']!.count,
|
|
FoodBankService.rebuild(logB)['x']!.count,
|
|
);
|
|
});
|
|
});
|
|
|
|
group('FoodBankService search/persistence', () {
|
|
late Directory tempDir;
|
|
|
|
setUp(() async {
|
|
tempDir = await Directory.systemTemp.createTemp('diet_guard_bank_');
|
|
FoodBankService.resetForTesting(testDir: tempDir);
|
|
LogStorageService.resetForTesting(testDir: tempDir);
|
|
});
|
|
|
|
tearDown(() async {
|
|
FoodBankService.resetForTesting();
|
|
LogStorageService.resetForTesting();
|
|
await tempDir.delete(recursive: true);
|
|
});
|
|
|
|
test('readBank returns empty map when no file exists', () async {
|
|
expect(await FoodBankService.instance.readBank(), isEmpty);
|
|
});
|
|
|
|
test('rebuildAndPersist writes the bank to disk', () async {
|
|
final log = {
|
|
'2026-06-22': [
|
|
_entry(id: '1', time: '2026-06-22T08:00:00+02:00', desc: 'egg'),
|
|
],
|
|
};
|
|
await FoodBankService.instance.rebuildAndPersist(log);
|
|
final reread = await FoodBankService.instance.readBank();
|
|
expect(reread['egg'], isNotNull);
|
|
});
|
|
|
|
test('search ranks an exact substring match above a fuzzy one', () async {
|
|
final log = {
|
|
'2026-06-22': [
|
|
_entry(id: '1', time: '2026-06-22T08:00:00+02:00', desc: 'banana'),
|
|
_entry(id: '2', time: '2026-06-22T08:01:00+02:00', desc: 'banama'),
|
|
],
|
|
};
|
|
await FoodBankService.instance.rebuildAndPersist(log);
|
|
final results = await FoodBankService.instance.search('banana');
|
|
expect(results.first.name, 'banana');
|
|
});
|
|
|
|
test('empty query returns the most-logged foods first', () async {
|
|
final log = {
|
|
'2026-06-22': [
|
|
_entry(id: '1', time: '2026-06-22T08:00:00+02:00', desc: 'rare'),
|
|
_entry(id: '2', time: '2026-06-22T08:01:00+02:00', desc: 'common'),
|
|
_entry(id: '3', time: '2026-06-22T08:02:00+02:00', desc: 'common'),
|
|
],
|
|
};
|
|
await FoodBankService.instance.rebuildAndPersist(log);
|
|
final results = await FoodBankService.instance.search('');
|
|
expect(results.first.name, 'common');
|
|
});
|
|
});
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Manual bank — addManualEntry / mergedEntries
|
|
// ---------------------------------------------------------------------------
|
|
|
|
group('FoodBankService manual bank', () {
|
|
late Directory tempDir;
|
|
|
|
setUp(() async {
|
|
tempDir = await Directory.systemTemp.createTemp('diet_guard_fb_manual_');
|
|
FoodBankService.resetForTesting(testDir: tempDir);
|
|
LogStorageService.resetForTesting(testDir: tempDir);
|
|
});
|
|
|
|
tearDown(() async {
|
|
FoodBankService.resetForTesting();
|
|
LogStorageService.resetForTesting();
|
|
await tempDir.delete(recursive: true);
|
|
});
|
|
|
|
test('addManualEntry persists and mergedEntries returns it', () async {
|
|
const record = FoodBankRecord(
|
|
desc: 'Manual oat',
|
|
kcal: 370,
|
|
proteinG: 13,
|
|
carbsG: 66,
|
|
fatG: 7,
|
|
grams: 100,
|
|
count: 0,
|
|
);
|
|
|
|
await FoodBankService.instance.addManualEntry(record);
|
|
final merged = await FoodBankService.instance.mergedEntries();
|
|
|
|
expect(merged.any((r) => r.desc == 'Manual oat'), isTrue);
|
|
});
|
|
|
|
test(
|
|
'mergedEntries: log-derived entry wins over manual on collision',
|
|
() async {
|
|
// Seed log with 'oat' (count=1, kcal=100).
|
|
final log = {
|
|
'2026-06-22': [
|
|
_entry(id: '1', time: '2026-06-22T08:00:00+02:00', desc: 'oat'),
|
|
],
|
|
};
|
|
await FoodBankService.instance.rebuildAndPersist(log);
|
|
|
|
// Add manual entry with same normalized key but different kcal.
|
|
await FoodBankService.instance.addManualEntry(
|
|
const FoodBankRecord(
|
|
desc: 'oat',
|
|
kcal: 999,
|
|
proteinG: 0,
|
|
carbsG: 0,
|
|
fatG: 0,
|
|
grams: 100,
|
|
count: 0,
|
|
),
|
|
);
|
|
|
|
final merged = await FoodBankService.instance.mergedEntries();
|
|
final oat = merged.firstWhere((r) => r.desc == 'oat');
|
|
// Log-derived entry (kcal=100) should win over manual (kcal=999).
|
|
expect(oat.kcal, 100);
|
|
},
|
|
);
|
|
|
|
test(
|
|
'mergedEntries includes both log-derived and manual entries',
|
|
() async {
|
|
final log = {
|
|
'2026-06-22': [
|
|
_entry(
|
|
id: '1',
|
|
time: '2026-06-22T08:00:00+02:00',
|
|
desc: 'toast',
|
|
),
|
|
],
|
|
};
|
|
await FoodBankService.instance.rebuildAndPersist(log);
|
|
|
|
await FoodBankService.instance.addManualEntry(
|
|
const FoodBankRecord(
|
|
desc: 'Quinoa',
|
|
kcal: 370,
|
|
proteinG: 14,
|
|
carbsG: 64,
|
|
fatG: 6,
|
|
grams: 100,
|
|
count: 0,
|
|
),
|
|
);
|
|
|
|
final merged = await FoodBankService.instance.mergedEntries();
|
|
final descs = merged.map((r) => r.desc).toSet();
|
|
expect(descs, containsAll(['toast', 'Quinoa']));
|
|
},
|
|
);
|
|
|
|
test('addManualEntry upserts by normalized key', () async {
|
|
await FoodBankService.instance.addManualEntry(
|
|
const FoodBankRecord(
|
|
desc: 'Oat',
|
|
kcal: 370,
|
|
proteinG: 13,
|
|
carbsG: 66,
|
|
fatG: 7,
|
|
grams: 100,
|
|
count: 0,
|
|
),
|
|
);
|
|
|
|
// Upsert same food with updated kcal.
|
|
await FoodBankService.instance.addManualEntry(
|
|
const FoodBankRecord(
|
|
desc: 'oat',
|
|
kcal: 400,
|
|
proteinG: 14,
|
|
carbsG: 68,
|
|
fatG: 8,
|
|
grams: 100,
|
|
count: 0,
|
|
),
|
|
);
|
|
|
|
final merged = await FoodBankService.instance.mergedEntries();
|
|
final oats = merged.where((r) => r.desc.toLowerCase() == 'oat').toList();
|
|
expect(oats.length, 1);
|
|
expect(oats.single.kcal, 400);
|
|
});
|
|
|
|
test('search includes manual entries', () async {
|
|
await FoodBankService.instance.addManualEntry(
|
|
const FoodBankRecord(
|
|
desc: 'Rare ingredient',
|
|
kcal: 50,
|
|
proteinG: 1,
|
|
carbsG: 10,
|
|
fatG: 0.5,
|
|
grams: 100,
|
|
count: 0,
|
|
),
|
|
);
|
|
|
|
final results = await FoodBankService.instance.search('Rare');
|
|
expect(results.any((r) => r.name == 'Rare ingredient'), isTrue);
|
|
});
|
|
});
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// IO error paths — FileSystemException and FormatException handlers
|
|
// ---------------------------------------------------------------------------
|
|
|
|
group('FoodBankService IO error paths', () {
|
|
late Directory tempDir;
|
|
|
|
setUp(() async {
|
|
tempDir = await Directory.systemTemp.createTemp('diet_guard_fb_err_');
|
|
FoodBankService.resetForTesting(testDir: tempDir);
|
|
LogStorageService.resetForTesting(testDir: tempDir);
|
|
});
|
|
|
|
tearDown(() async {
|
|
FoodBankService.resetForTesting();
|
|
LogStorageService.resetForTesting();
|
|
await tempDir.delete(recursive: true);
|
|
});
|
|
|
|
test('readBank returns empty on invalid JSON (FormatException)', () async {
|
|
await File(
|
|
'${tempDir.path}/food_bank.json',
|
|
).writeAsString('not valid json {{{');
|
|
expect(await FoodBankService.instance.readBank(), isEmpty);
|
|
});
|
|
|
|
test(
|
|
'readBank returns empty when file is unreadable (FileSystemException)',
|
|
() async {
|
|
final bankPath = '${tempDir.path}/food_bank.json';
|
|
await File(bankPath).writeAsString('{}');
|
|
await Process.run('chmod', ['000', bankPath]);
|
|
|
|
expect(await FoodBankService.instance.readBank(), isEmpty);
|
|
|
|
await Process.run('chmod', ['644', bankPath]);
|
|
},
|
|
);
|
|
|
|
test(
|
|
'mergedEntries handles invalid JSON in manual bank (FormatException)',
|
|
() async {
|
|
await File(
|
|
'${tempDir.path}/food_bank_manual.json',
|
|
).writeAsString('not valid json {{{');
|
|
expect(await FoodBankService.instance.mergedEntries(), isEmpty);
|
|
},
|
|
);
|
|
|
|
test(
|
|
'mergedEntries handles unreadable manual bank (FileSystemException)',
|
|
() async {
|
|
final manualPath = '${tempDir.path}/food_bank_manual.json';
|
|
await File(manualPath).writeAsString('{}');
|
|
await Process.run('chmod', ['000', manualPath]);
|
|
|
|
expect(await FoodBankService.instance.mergedEntries(), isEmpty);
|
|
|
|
await Process.run('chmod', ['644', manualPath]);
|
|
},
|
|
);
|
|
});
|
|
}
|