Files
life-helper/app/test/domain/streak/compute_streak_test.dart
joungmin 8fe6a8f378 [Developer] #204 Phase 1 MVP — Flutter app skeleton complete
- Drift 21 tables (8 catalog + 11 user + habit_dose_variants + meta_kv)
  with R1~R10 CHECK constraints and 19 indexes
- 8 hand-crafted seed JSON catalogs in app/assets/seed/
  (refs 84, protocols 34, methodologies 21, frame_patterns 30,
   reward_menu_items 30, break_protocols 8, common_frames 5, diet_patterns 5)
- 6 domain functions: recommend_variant, compute_streak,
  validate_frame_level, active_habit_quota, weekly_minimum_ratio,
  seed_importer (transactional, idempotent)
- 4 vertical-slice Riverpod screens: HabitList, HabitCreate, CheckIn, Streak
- 31 unit tests passing; flutter analyze clean
- OQ-5 streak semantics: missing entry ≠ explicit blank
  (missing = end of history; only TrackerValue.blank triggers Never-miss-twice)

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-06-12 10:33:03 +09:00

116 lines
3.7 KiB
Dart

import 'package:flutter_test/flutter_test.dart';
import 'package:life_helper/domain/models/tracker_entry.dart';
import 'package:life_helper/domain/streak/compute_streak.dart';
TrackerEntryModel _e(String date, TrackerValue v) =>
TrackerEntryModel(id: 'te_$date', habitId: 'hb', date: date, value: v);
void main() {
group('computeStreak', () {
test('empty → all zero, T0, not broken', () {
final s = computeStreak(
entries: const [],
asOf: DateTime(2026, 6, 11),
habitStartedAt: '2026-06-01',
);
expect(s.currentStreak, 0);
expect(s.currentTier, RewardTier.t0);
expect(s.neverMissTwiceBroken, false);
});
test('3 consecutive done → T1', () {
final s = computeStreak(
entries: [
_e('2026-06-09', TrackerValue.done),
_e('2026-06-10', TrackerValue.done),
_e('2026-06-11', TrackerValue.done),
],
asOf: DateTime(2026, 6, 11),
habitStartedAt: '2026-06-01',
);
expect(s.currentStreak, 3);
expect(s.currentTier, RewardTier.t1);
});
test('7 consecutive done → T2', () {
final entries = <TrackerEntryModel>[];
for (var i = 0; i < 7; i++) {
final d = DateTime(2026, 6, 5).add(Duration(days: i));
entries.add(_e(_ymd(d), TrackerValue.done));
}
final s = computeStreak(
entries: entries,
asOf: DateTime(2026, 6, 11),
habitStartedAt: '2026-06-01',
);
expect(s.currentStreak, 7);
expect(s.currentTier, RewardTier.t2);
});
test('OQ-5: 1 blank → streak=0, tier stays (not broken)', () {
// 6/9 done, 6/10 blank entry, 6/11 done.
final s = computeStreak(
entries: [
_e('2026-06-09', TrackerValue.done),
_e('2026-06-10', TrackerValue.blank),
_e('2026-06-11', TrackerValue.done),
],
asOf: DateTime(2026, 6, 11),
habitStartedAt: '2026-06-01',
);
// Walks back: 6/11 done (+1), 6/10 blank → streak resets to 0.
expect(s.currentStreak, 0);
expect(s.neverMissTwiceBroken, false);
});
test('OQ-5: 2 consecutive blank → neverMissTwiceBroken=true', () {
final s = computeStreak(
entries: [
_e('2026-06-09', TrackerValue.done),
_e('2026-06-10', TrackerValue.blank),
_e('2026-06-11', TrackerValue.blank),
],
asOf: DateTime(2026, 6, 11),
habitStartedAt: '2026-06-01',
);
expect(s.currentStreak, 0);
expect(s.neverMissTwiceBroken, true);
});
test('window30: 24/30 done → T3', () {
final entries = <TrackerEntryModel>[];
// 24 done in last 30 days, but not as a streak.
for (var i = 0; i < 24; i++) {
final d = DateTime(2026, 6, 11).subtract(Duration(days: i));
entries.add(_e(_ymd(d), TrackerValue.done));
}
final s = computeStreak(
entries: entries,
asOf: DateTime(2026, 6, 11),
habitStartedAt: '2026-05-01',
);
expect(s.doneCountInWindow30, 24);
expect(s.currentTier.rank, greaterThanOrEqualTo(RewardTier.t3.rank));
});
test('longestStreak picks largest run regardless of current', () {
final s = computeStreak(
entries: [
for (final d in ['2026-06-01', '2026-06-02', '2026-06-03', '2026-06-04'])
_e(d, TrackerValue.done),
_e('2026-06-05', TrackerValue.blank),
_e('2026-06-11', TrackerValue.done),
],
asOf: DateTime(2026, 6, 11),
habitStartedAt: '2026-06-01',
);
expect(s.longestStreak, 4);
});
});
}
String _ymd(DateTime d) =>
'${d.year.toString().padLeft(4, '0')}-'
'${d.month.toString().padLeft(2, '0')}-'
'${d.day.toString().padLeft(2, '0')}';