feat: BLoC tests complets + sécurité production + freerasp 7.5.1 migration
## Tests BLoC (Task P2.4 Mobile) - 25 nouveaux fichiers *_bloc_test.dart + mocks générés (build_runner) - Features couvertes : authentication, admin_users, adhesions, backup, communication/messaging, contributions, dashboard, finance (approval/budget), events, explore/network, feed, logs_monitoring, notifications, onboarding, organizations (switcher/types/CRUD), profile, reports, settings, solidarity - ~380 tests, > 80% coverage BLoCs ## Sécurité Production (Task P2.2) - lib/core/security/app_integrity_service.dart (freerasp 7.5.1) - Migration API breaking changes freerasp 7.5.1 : - onRootDetected → onPrivilegedAccess - onDebuggerDetected → onDebug - onSignatureDetected → onAppIntegrity - onHookDetected → onHooks - onEmulatorDetected → onSimulator - onUntrustedInstallationSourceDetected → onUnofficialStore - onDeviceBindingDetected → onDeviceBinding - onObfuscationIssuesDetected → onObfuscationIssues - Talsec.start() split → start() + attachListener() - const AndroidConfig/IOSConfig → final (constructors call ConfigVerifier) - supportedAlternativeStores → supportedStores ## Pubspec - bloc_test: ^9.1.7 → ^10.0.0 (compat flutter_bloc ^9.0.0) - freerasp 7.5.1 ## Config - android/app/build.gradle : ajustements release - lib/core/config/environment.dart : URLs API actualisées - lib/main.dart + app_router : intégrations sécurité/BLoC ## Cleanup - Suppression docs intermédiaires (TACHES_*.md, TASK_*_COMPLETION_REPORT.md, TESTS_UNITAIRES_PROGRESS.md) - .g.dart régénérés (json_serializable) - .mocks.dart régénérés (mockito) ## Résultat - 142 fichiers, +27 596 insertions - Toutes les tâches P2 mobile complétées Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
302
test/features/explore/bloc/network_bloc_test.dart
Normal file
302
test/features/explore/bloc/network_bloc_test.dart
Normal file
@@ -0,0 +1,302 @@
|
||||
import 'package:bloc_test/bloc_test.dart';
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:mockito/annotations.dart';
|
||||
import 'package:mockito/mockito.dart';
|
||||
|
||||
import 'package:unionflow_mobile_apps/features/explore/presentation/bloc/network_bloc.dart';
|
||||
import 'package:unionflow_mobile_apps/features/explore/presentation/bloc/network_event.dart';
|
||||
import 'package:unionflow_mobile_apps/features/explore/presentation/bloc/network_state.dart';
|
||||
import 'package:unionflow_mobile_apps/features/explore/data/repositories/network_repository.dart';
|
||||
|
||||
@GenerateMocks([NetworkRepository])
|
||||
import 'network_bloc_test.mocks.dart';
|
||||
|
||||
void main() {
|
||||
late NetworkBloc bloc;
|
||||
late MockNetworkRepository mockRepository;
|
||||
|
||||
// ── Fixtures ──────────────────────────────────────────────────────────────
|
||||
|
||||
NetworkItem fakeMember({
|
||||
String id = 'mbr-1',
|
||||
bool isConnected = false,
|
||||
}) =>
|
||||
NetworkItem(
|
||||
id: id,
|
||||
name: 'Jean Dupont',
|
||||
subtitle: 'Comptable',
|
||||
type: 'Member',
|
||||
isConnected: isConnected,
|
||||
);
|
||||
|
||||
NetworkItem fakeOrg({String id = 'org-1'}) => NetworkItem(
|
||||
id: id,
|
||||
name: 'Association Test',
|
||||
subtitle: 'ASSOCIATION',
|
||||
type: 'Organization',
|
||||
isConnected: false,
|
||||
);
|
||||
|
||||
setUp(() {
|
||||
mockRepository = MockNetworkRepository();
|
||||
bloc = NetworkBloc(mockRepository);
|
||||
});
|
||||
|
||||
tearDown(() => bloc.close());
|
||||
|
||||
// ── Initial state ─────────────────────────────────────────────────────────
|
||||
|
||||
test('initial state is NetworkInitial', () {
|
||||
expect(bloc.state, isA<NetworkInitial>());
|
||||
});
|
||||
|
||||
// ── LoadNetworkRequested ──────────────────────────────────────────────────
|
||||
|
||||
group('LoadNetworkRequested', () {
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'emits [NetworkLoading, NetworkLoaded] on success with followed members',
|
||||
build: () {
|
||||
when(mockRepository.getFollowedIds())
|
||||
.thenAnswer((_) async => ['mbr-1', 'mbr-2']);
|
||||
when(mockRepository.search('', followedIds: anyNamed('followedIds')))
|
||||
.thenAnswer((_) async => [
|
||||
fakeMember(isConnected: true),
|
||||
fakeMember(id: 'mbr-2', isConnected: true),
|
||||
fakeOrg(),
|
||||
]);
|
||||
return bloc;
|
||||
},
|
||||
act: (b) => b.add(LoadNetworkRequested()),
|
||||
expect: () => [
|
||||
isA<NetworkLoading>(),
|
||||
isA<NetworkLoaded>()
|
||||
.having((s) => s.items.length, 'items.length', 3)
|
||||
.having((s) => s.currentQuery, 'currentQuery', ''),
|
||||
],
|
||||
verify: (_) {
|
||||
verify(mockRepository.getFollowedIds()).called(1);
|
||||
verify(mockRepository.search('', followedIds: anyNamed('followedIds'))).called(1);
|
||||
},
|
||||
);
|
||||
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'emits [NetworkLoading, NetworkLoaded] with empty list',
|
||||
build: () {
|
||||
when(mockRepository.getFollowedIds()).thenAnswer((_) async => []);
|
||||
when(mockRepository.search('', followedIds: anyNamed('followedIds')))
|
||||
.thenAnswer((_) async => []);
|
||||
return bloc;
|
||||
},
|
||||
act: (b) => b.add(LoadNetworkRequested()),
|
||||
expect: () => [
|
||||
isA<NetworkLoading>(),
|
||||
isA<NetworkLoaded>().having((s) => s.items, 'items', isEmpty),
|
||||
],
|
||||
);
|
||||
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'emits [NetworkLoading, NetworkError] on getFollowedIds failure',
|
||||
build: () {
|
||||
when(mockRepository.getFollowedIds()).thenThrow(Exception('auth error'));
|
||||
return bloc;
|
||||
},
|
||||
act: (b) => b.add(LoadNetworkRequested()),
|
||||
expect: () => [
|
||||
isA<NetworkLoading>(),
|
||||
isA<NetworkError>().having(
|
||||
(s) => s.message,
|
||||
'message',
|
||||
contains('Erreur'),
|
||||
),
|
||||
],
|
||||
);
|
||||
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'emits [NetworkLoading, NetworkError] on search failure',
|
||||
build: () {
|
||||
when(mockRepository.getFollowedIds()).thenAnswer((_) async => []);
|
||||
when(mockRepository.search(any, followedIds: anyNamed('followedIds')))
|
||||
.thenThrow(Exception('search error'));
|
||||
return bloc;
|
||||
},
|
||||
act: (b) => b.add(LoadNetworkRequested()),
|
||||
expect: () => [isA<NetworkLoading>(), isA<NetworkError>()],
|
||||
);
|
||||
});
|
||||
|
||||
// ── SearchNetworkRequested ────────────────────────────────────────────────
|
||||
|
||||
group('SearchNetworkRequested', () {
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'emits [NetworkLoading, NetworkLoaded] with search results',
|
||||
build: () {
|
||||
when(mockRepository.getFollowedIds()).thenAnswer((_) async => ['mbr-1']);
|
||||
when(mockRepository.search('Jean', followedIds: anyNamed('followedIds')))
|
||||
.thenAnswer((_) async => [fakeMember(isConnected: true)]);
|
||||
return bloc;
|
||||
},
|
||||
act: (b) => b.add(const SearchNetworkRequested('Jean')),
|
||||
expect: () => [
|
||||
isA<NetworkLoading>(),
|
||||
isA<NetworkLoaded>()
|
||||
.having((s) => s.currentQuery, 'currentQuery', 'Jean')
|
||||
.having((s) => s.items.length, 'items.length', 1),
|
||||
],
|
||||
verify: (_) {
|
||||
verify(mockRepository.search('Jean', followedIds: anyNamed('followedIds'))).called(1);
|
||||
},
|
||||
);
|
||||
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'emits [NetworkLoading, NetworkLoaded] with empty query (no search)',
|
||||
build: () {
|
||||
when(mockRepository.getFollowedIds()).thenAnswer((_) async => []);
|
||||
when(mockRepository.search('', followedIds: anyNamed('followedIds')))
|
||||
.thenAnswer((_) async => [fakeMember()]);
|
||||
return bloc;
|
||||
},
|
||||
act: (b) => b.add(const SearchNetworkRequested('')),
|
||||
expect: () => [
|
||||
isA<NetworkLoading>(),
|
||||
isA<NetworkLoaded>().having((s) => s.currentQuery, 'currentQuery', ''),
|
||||
],
|
||||
);
|
||||
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'emits [NetworkLoading, NetworkLoaded] with whitespace-only query (treated as empty)',
|
||||
build: () {
|
||||
when(mockRepository.getFollowedIds()).thenAnswer((_) async => []);
|
||||
when(mockRepository.search('', followedIds: anyNamed('followedIds')))
|
||||
.thenAnswer((_) async => []);
|
||||
return bloc;
|
||||
},
|
||||
act: (b) => b.add(const SearchNetworkRequested(' ')),
|
||||
expect: () => [
|
||||
isA<NetworkLoading>(),
|
||||
isA<NetworkLoaded>().having((s) => s.currentQuery, 'currentQuery', ''),
|
||||
],
|
||||
);
|
||||
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'emits [NetworkLoading, NetworkError] on search failure',
|
||||
build: () {
|
||||
when(mockRepository.getFollowedIds()).thenAnswer((_) async => []);
|
||||
when(mockRepository.search(any, followedIds: anyNamed('followedIds')))
|
||||
.thenThrow(Exception('search failed'));
|
||||
return bloc;
|
||||
},
|
||||
act: (b) => b.add(const SearchNetworkRequested('error')),
|
||||
expect: () => [
|
||||
isA<NetworkLoading>(),
|
||||
isA<NetworkError>(),
|
||||
],
|
||||
);
|
||||
});
|
||||
|
||||
// ── ToggleFollowRequested ─────────────────────────────────────────────────
|
||||
|
||||
group('ToggleFollowRequested — Member type', () {
|
||||
// Prerequisite: put bloc in NetworkLoaded state first
|
||||
final loadedState = NetworkLoaded(
|
||||
items: [
|
||||
fakeMember(id: 'mbr-1', isConnected: false),
|
||||
fakeOrg(),
|
||||
],
|
||||
currentQuery: 'test',
|
||||
);
|
||||
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'follows an unfollowed Member and emits NetworkLoaded with isConnected=true',
|
||||
build: () {
|
||||
when(mockRepository.follow('mbr-1')).thenAnswer((_) async => true);
|
||||
return bloc;
|
||||
},
|
||||
seed: () => loadedState,
|
||||
act: (b) => b.add(const ToggleFollowRequested('mbr-1')),
|
||||
expect: () => [
|
||||
isA<NetworkLoaded>().having(
|
||||
(s) => s.items.firstWhere((i) => i.id == 'mbr-1').isConnected,
|
||||
'mbr-1.isConnected',
|
||||
true,
|
||||
),
|
||||
],
|
||||
verify: (_) => verify(mockRepository.follow('mbr-1')).called(1),
|
||||
);
|
||||
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'unfollows a followed Member and emits NetworkLoaded with isConnected=false',
|
||||
build: () {
|
||||
when(mockRepository.unfollow('mbr-1')).thenAnswer((_) async => false);
|
||||
return bloc;
|
||||
},
|
||||
seed: () => NetworkLoaded(
|
||||
items: [fakeMember(id: 'mbr-1', isConnected: true)],
|
||||
currentQuery: '',
|
||||
),
|
||||
act: (b) => b.add(const ToggleFollowRequested('mbr-1')),
|
||||
expect: () => [
|
||||
isA<NetworkLoaded>().having(
|
||||
(s) => s.items.first.isConnected,
|
||||
'isConnected',
|
||||
false,
|
||||
),
|
||||
],
|
||||
verify: (_) => verify(mockRepository.unfollow('mbr-1')).called(1),
|
||||
);
|
||||
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'emits [NetworkError] when follow call fails',
|
||||
build: () {
|
||||
when(mockRepository.follow(any)).thenThrow(Exception('follow failed'));
|
||||
return bloc;
|
||||
},
|
||||
seed: () => loadedState,
|
||||
act: (b) => b.add(const ToggleFollowRequested('mbr-1')),
|
||||
expect: () => [isA<NetworkError>()],
|
||||
);
|
||||
});
|
||||
|
||||
group('ToggleFollowRequested — Organization type (local toggle only)', () {
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'toggles Organization follow locally without calling repository',
|
||||
build: () => bloc,
|
||||
seed: () => NetworkLoaded(
|
||||
items: [fakeOrg(id: 'org-1')],
|
||||
currentQuery: '',
|
||||
),
|
||||
act: (b) => b.add(const ToggleFollowRequested('org-1')),
|
||||
expect: () => [
|
||||
isA<NetworkLoaded>().having(
|
||||
(s) => s.items.first.isConnected,
|
||||
'isConnected',
|
||||
true,
|
||||
),
|
||||
],
|
||||
verify: (_) {
|
||||
verifyNever(mockRepository.follow(any));
|
||||
verifyNever(mockRepository.unfollow(any));
|
||||
},
|
||||
);
|
||||
});
|
||||
|
||||
group('ToggleFollowRequested — guard: item not in list', () {
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'does nothing when item id not found in current list',
|
||||
build: () => bloc,
|
||||
seed: () => NetworkLoaded(
|
||||
items: [fakeMember(id: 'mbr-1')],
|
||||
currentQuery: '',
|
||||
),
|
||||
act: (b) => b.add(const ToggleFollowRequested('non-existent')),
|
||||
expect: () => [],
|
||||
);
|
||||
|
||||
blocTest<NetworkBloc, NetworkState>(
|
||||
'does nothing when state is not NetworkLoaded',
|
||||
build: () => bloc,
|
||||
// Default initial state (NetworkInitial)
|
||||
act: (b) => b.add(const ToggleFollowRequested('mbr-1')),
|
||||
expect: () => [],
|
||||
);
|
||||
});
|
||||
}
|
||||
115
test/features/explore/bloc/network_bloc_test.mocks.dart
Normal file
115
test/features/explore/bloc/network_bloc_test.mocks.dart
Normal file
@@ -0,0 +1,115 @@
|
||||
// Mocks generated by Mockito 5.4.6 from annotations
|
||||
// in unionflow_mobile_apps/test/features/explore/bloc/network_bloc_test.dart.
|
||||
// Do not manually edit this file.
|
||||
|
||||
// ignore_for_file: no_leading_underscores_for_library_prefixes
|
||||
import 'dart:async' as _i3;
|
||||
|
||||
import 'package:mockito/mockito.dart' as _i1;
|
||||
import 'package:unionflow_mobile_apps/features/explore/data/repositories/network_repository.dart'
|
||||
as _i2;
|
||||
import 'package:unionflow_mobile_apps/features/explore/domain/entities/network_item.dart'
|
||||
as _i4;
|
||||
|
||||
// ignore_for_file: type=lint
|
||||
// ignore_for_file: avoid_redundant_argument_values
|
||||
// ignore_for_file: avoid_setters_without_getters
|
||||
// ignore_for_file: comment_references
|
||||
// ignore_for_file: deprecated_member_use
|
||||
// ignore_for_file: deprecated_member_use_from_same_package
|
||||
// ignore_for_file: implementation_imports
|
||||
// ignore_for_file: invalid_use_of_visible_for_testing_member
|
||||
// ignore_for_file: must_be_immutable
|
||||
// ignore_for_file: prefer_const_constructors
|
||||
// ignore_for_file: unnecessary_parenthesis
|
||||
// ignore_for_file: camel_case_types
|
||||
// ignore_for_file: subtype_of_sealed_class
|
||||
// ignore_for_file: invalid_use_of_internal_member
|
||||
|
||||
/// A class which mocks [NetworkRepository].
|
||||
///
|
||||
/// See the documentation for Mockito's code generation for more information.
|
||||
class MockNetworkRepository extends _i1.Mock implements _i2.NetworkRepository {
|
||||
MockNetworkRepository() {
|
||||
_i1.throwOnMissingStub(this);
|
||||
}
|
||||
|
||||
@override
|
||||
_i3.Future<List<_i4.NetworkItem>> searchMembers(
|
||||
String? query, {
|
||||
int? page = 0,
|
||||
int? size = 20,
|
||||
}) =>
|
||||
(super.noSuchMethod(
|
||||
Invocation.method(
|
||||
#searchMembers,
|
||||
[query],
|
||||
{#page: page, #size: size},
|
||||
),
|
||||
returnValue: _i3.Future<List<_i4.NetworkItem>>.value(
|
||||
<_i4.NetworkItem>[],
|
||||
),
|
||||
)
|
||||
as _i3.Future<List<_i4.NetworkItem>>);
|
||||
|
||||
@override
|
||||
_i3.Future<List<_i4.NetworkItem>> searchOrganizations(
|
||||
String? query, {
|
||||
int? page = 0,
|
||||
int? size = 20,
|
||||
}) =>
|
||||
(super.noSuchMethod(
|
||||
Invocation.method(
|
||||
#searchOrganizations,
|
||||
[query],
|
||||
{#page: page, #size: size},
|
||||
),
|
||||
returnValue: _i3.Future<List<_i4.NetworkItem>>.value(
|
||||
<_i4.NetworkItem>[],
|
||||
),
|
||||
)
|
||||
as _i3.Future<List<_i4.NetworkItem>>);
|
||||
|
||||
@override
|
||||
_i3.Future<List<_i4.NetworkItem>> search(
|
||||
String? query, {
|
||||
int? page = 0,
|
||||
int? size = 10,
|
||||
Set<String>? followedIds,
|
||||
}) =>
|
||||
(super.noSuchMethod(
|
||||
Invocation.method(
|
||||
#search,
|
||||
[query],
|
||||
{#page: page, #size: size, #followedIds: followedIds},
|
||||
),
|
||||
returnValue: _i3.Future<List<_i4.NetworkItem>>.value(
|
||||
<_i4.NetworkItem>[],
|
||||
),
|
||||
)
|
||||
as _i3.Future<List<_i4.NetworkItem>>);
|
||||
|
||||
@override
|
||||
_i3.Future<List<String>> getFollowedIds() =>
|
||||
(super.noSuchMethod(
|
||||
Invocation.method(#getFollowedIds, []),
|
||||
returnValue: _i3.Future<List<String>>.value(<String>[]),
|
||||
)
|
||||
as _i3.Future<List<String>>);
|
||||
|
||||
@override
|
||||
_i3.Future<bool> follow(String? memberId) =>
|
||||
(super.noSuchMethod(
|
||||
Invocation.method(#follow, [memberId]),
|
||||
returnValue: _i3.Future<bool>.value(false),
|
||||
)
|
||||
as _i3.Future<bool>);
|
||||
|
||||
@override
|
||||
_i3.Future<bool> unfollow(String? memberId) =>
|
||||
(super.noSuchMethod(
|
||||
Invocation.method(#unfollow, [memberId]),
|
||||
returnValue: _i3.Future<bool>.value(false),
|
||||
)
|
||||
as _i3.Future<bool>);
|
||||
}
|
||||
Reference in New Issue
Block a user