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:
dahoud
2026-04-21 12:42:35 +00:00
parent 33f5b5a707
commit 37db88672b
142 changed files with 27599 additions and 16068 deletions

View File

@@ -0,0 +1,498 @@
import 'dart:async';
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/communication/presentation/bloc/messaging_bloc.dart';
import 'package:unionflow_mobile_apps/features/communication/presentation/bloc/messaging_event.dart';
import 'package:unionflow_mobile_apps/features/communication/presentation/bloc/messaging_state.dart';
import 'package:unionflow_mobile_apps/features/communication/domain/repositories/messaging_repository.dart';
import 'package:unionflow_mobile_apps/features/communication/domain/entities/conversation.dart';
import 'package:unionflow_mobile_apps/features/communication/domain/entities/message.dart';
import 'package:unionflow_mobile_apps/core/websocket/websocket_service.dart';
@GenerateMocks([MessagingRepository, WebSocketService])
import 'messaging_bloc_test.mocks.dart';
void main() {
late MessagingBloc bloc;
late MockMessagingRepository mockRepository;
late MockWebSocketService mockWebSocketService;
late StreamController<WebSocketEvent> wsController;
// ── Fixtures ──────────────────────────────────────────────────────────────
ConversationSummary fakeConversationSummary({
String id = 'conv-1',
String type = 'DIRECTE',
}) =>
ConversationSummary(
id: id,
typeConversation: type,
titre: 'Conversation $id',
statut: 'ACTIVE',
nonLus: 0,
);
Conversation fakeConversation({String id = 'conv-1'}) => Conversation(
id: id,
typeConversation: 'DIRECTE',
titre: 'Conversation $id',
statut: 'ACTIVE',
messages: [],
participants: [],
);
Message fakeMessage({String id = 'msg-1'}) => Message(
id: id,
typeMessage: 'TEXTE',
contenu: 'Bonjour !',
dateEnvoi: DateTime.now(),
);
setUp(() {
wsController = StreamController<WebSocketEvent>.broadcast();
mockRepository = MockMessagingRepository();
mockWebSocketService = MockWebSocketService();
when(mockWebSocketService.eventStream).thenAnswer((_) => wsController.stream);
bloc = MessagingBloc(
repository: mockRepository,
webSocketService: mockWebSocketService,
);
});
tearDown(() async {
await bloc.close();
await wsController.close();
});
// ── Initial state ─────────────────────────────────────────────────────────
test('initial state is MessagingInitial', () {
expect(bloc.state, isA<MessagingInitial>());
});
// ── LoadMesConversations ──────────────────────────────────────────────────
group('LoadMesConversations', () {
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingLoading, MesConversationsLoaded] on success',
build: () {
when(mockRepository.getMesConversations()).thenAnswer(
(_) async => [
fakeConversationSummary(),
fakeConversationSummary(id: 'conv-2', type: 'ROLE_CANAL'),
],
);
return bloc;
},
act: (b) => b.add(const LoadMesConversations()),
expect: () => [
isA<MessagingLoading>(),
isA<MesConversationsLoaded>().having(
(s) => s.conversations.length,
'conversations.length',
2,
),
],
verify: (_) => verify(mockRepository.getMesConversations()).called(1),
);
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingLoading, MesConversationsLoaded] with empty list',
build: () {
when(mockRepository.getMesConversations()).thenAnswer((_) async => []);
return bloc;
},
act: (b) => b.add(const LoadMesConversations()),
expect: () => [
isA<MessagingLoading>(),
isA<MesConversationsLoaded>().having(
(s) => s.conversations,
'conversations',
isEmpty,
),
],
);
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingLoading, MessagingError] on failure',
build: () {
when(mockRepository.getMesConversations())
.thenThrow(Exception('network error'));
return bloc;
},
act: (b) => b.add(const LoadMesConversations()),
expect: () => [
isA<MessagingLoading>(),
isA<MessagingError>(),
],
);
});
// ── OpenConversation ──────────────────────────────────────────────────────
group('OpenConversation', () {
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingLoading, ConversationOuverte] on success',
build: () {
when(mockRepository.getConversation('conv-1'))
.thenAnswer((_) async => fakeConversation());
when(mockRepository.marquerLu('conv-1')).thenAnswer((_) async {});
return bloc;
},
act: (b) => b.add(const OpenConversation('conv-1')),
expect: () => [
isA<MessagingLoading>(),
isA<ConversationOuverte>().having(
(s) => s.conversation.id,
'conversation.id',
'conv-1',
),
],
verify: (_) {
verify(mockRepository.getConversation('conv-1')).called(1);
},
);
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingLoading, MessagingError] when conversation not found',
build: () {
when(mockRepository.getConversation(any))
.thenThrow(Exception('Conversation introuvable'));
return bloc;
},
act: (b) => b.add(const OpenConversation('conv-x')),
expect: () => [isA<MessagingLoading>(), isA<MessagingError>()],
);
});
// ── DemarrerConversationDirecte ───────────────────────────────────────────
group('DemarrerConversationDirecte', () {
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingLoading, ConversationCreee] on success',
build: () {
when(mockRepository.demarrerConversationDirecte(
destinataireId: 'usr-2',
organisationId: 'org-1',
premierMessage: anyNamed('premierMessage'),
)).thenAnswer((_) async => fakeConversation(id: 'conv-new'));
return bloc;
},
act: (b) => b.add(const DemarrerConversationDirecte(
destinataireId: 'usr-2',
organisationId: 'org-1',
premierMessage: 'Bonjour !',
)),
expect: () => [
isA<MessagingLoading>(),
isA<ConversationCreee>().having(
(s) => s.conversation.id,
'conversation.id',
'conv-new',
),
],
);
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingLoading, MessagingError] on failure',
build: () {
when(mockRepository.demarrerConversationDirecte(
destinataireId: anyNamed('destinataireId'),
organisationId: anyNamed('organisationId'),
premierMessage: anyNamed('premierMessage'),
)).thenThrow(Exception('Cannot start conversation'));
return bloc;
},
act: (b) => b.add(const DemarrerConversationDirecte(
destinataireId: 'usr-2',
organisationId: 'org-1',
)),
expect: () => [isA<MessagingLoading>(), isA<MessagingError>()],
);
});
// ── DemarrerConversationRole ──────────────────────────────────────────────
group('DemarrerConversationRole', () {
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingLoading, ConversationCreee] on success',
build: () {
when(mockRepository.demarrerConversationRole(
roleCible: 'TRESORIER',
organisationId: 'org-1',
premierMessage: anyNamed('premierMessage'),
)).thenAnswer((_) async => fakeConversation(id: 'conv-role'));
return bloc;
},
act: (b) => b.add(const DemarrerConversationRole(
roleCible: 'TRESORIER',
organisationId: 'org-1',
)),
expect: () => [
isA<MessagingLoading>(),
isA<ConversationCreee>().having(
(s) => s.conversation.id,
'conversation.id',
'conv-role',
),
],
);
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingLoading, MessagingError] on failure',
build: () {
when(mockRepository.demarrerConversationRole(
roleCible: anyNamed('roleCible'),
organisationId: anyNamed('organisationId'),
premierMessage: anyNamed('premierMessage'),
)).thenThrow(Exception('role not found'));
return bloc;
},
act: (b) => b.add(const DemarrerConversationRole(
roleCible: 'ADMIN',
organisationId: 'org-1',
)),
expect: () => [isA<MessagingLoading>(), isA<MessagingError>()],
);
});
// ── ArchiverConversation ──────────────────────────────────────────────────
group('ArchiverConversation', () {
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingActionOk, MessagingLoading, MesConversationsLoaded] on success',
build: () {
when(mockRepository.archiverConversation('conv-1'))
.thenAnswer((_) async {});
when(mockRepository.getMesConversations()).thenAnswer((_) async => []);
return bloc;
},
act: (b) => b.add(const ArchiverConversation('conv-1')),
expect: () => [
isA<MessagingActionOk>().having(
(s) => s.action,
'action',
'archiver',
),
isA<MessagingLoading>(),
isA<MesConversationsLoaded>(),
],
verify: (_) {
verify(mockRepository.archiverConversation('conv-1')).called(1);
verify(mockRepository.getMesConversations()).called(1);
},
);
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingError] on archive failure',
build: () {
when(mockRepository.archiverConversation(any))
.thenThrow(Exception('archive failed'));
return bloc;
},
act: (b) => b.add(const ArchiverConversation('conv-1')),
expect: () => [isA<MessagingError>()],
);
});
// ── EnvoyerMessageTexte ───────────────────────────────────────────────────
group('EnvoyerMessageTexte', () {
blocTest<MessagingBloc, MessagingState>(
'emits [MessageEnvoye, MessagesLoaded] on success',
build: () {
when(mockRepository.envoyerMessage(
'conv-1',
typeMessage: 'TEXTE',
contenu: 'Hello',
messageParentId: anyNamed('messageParentId'),
)).thenAnswer((_) async => fakeMessage());
when(mockRepository.getMessages('conv-1', page: anyNamed('page')))
.thenAnswer((_) async => [fakeMessage()]);
return bloc;
},
act: (b) => b.add(const EnvoyerMessageTexte(
conversationId: 'conv-1',
contenu: 'Hello',
)),
expect: () => [
isA<MessageEnvoye>()
.having((s) => s.conversationId, 'conversationId', 'conv-1')
.having((s) => s.message.id, 'message.id', 'msg-1'),
isA<MessagesLoaded>(),
],
);
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingError] on send failure',
build: () {
when(mockRepository.envoyerMessage(
any,
typeMessage: anyNamed('typeMessage'),
contenu: anyNamed('contenu'),
messageParentId: anyNamed('messageParentId'),
)).thenThrow(Exception('send failed'));
return bloc;
},
act: (b) => b.add(const EnvoyerMessageTexte(
conversationId: 'conv-1',
contenu: 'Hello',
)),
expect: () => [isA<MessagingError>()],
);
});
// ── LoadMessages ──────────────────────────────────────────────────────────
group('LoadMessages', () {
blocTest<MessagingBloc, MessagingState>(
'emits [MessagesLoaded] with hasMore=true when 20 messages returned',
build: () {
final messages = List.generate(
20,
(i) => fakeMessage(id: 'msg-$i'),
);
when(mockRepository.getMessages('conv-1', page: 0))
.thenAnswer((_) async => messages);
return bloc;
},
act: (b) => b.add(const LoadMessages(conversationId: 'conv-1')),
expect: () => [
isA<MessagesLoaded>()
.having((s) => s.messages.length, 'messages.length', 20)
.having((s) => s.hasMore, 'hasMore', true)
.having((s) => s.conversationId, 'conversationId', 'conv-1'),
],
);
blocTest<MessagingBloc, MessagingState>(
'emits [MessagesLoaded] with hasMore=false when less than 20 messages',
build: () {
when(mockRepository.getMessages('conv-1', page: 0))
.thenAnswer((_) async => [fakeMessage(), fakeMessage(id: 'msg-2')]);
return bloc;
},
act: (b) => b.add(const LoadMessages(conversationId: 'conv-1')),
expect: () => [
isA<MessagesLoaded>().having((s) => s.hasMore, 'hasMore', false),
],
);
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingError] on load messages failure',
build: () {
when(mockRepository.getMessages(any, page: anyNamed('page')))
.thenThrow(Exception('load error'));
return bloc;
},
act: (b) => b.add(const LoadMessages(conversationId: 'conv-1')),
expect: () => [isA<MessagingError>()],
);
});
// ── SupprimerMessage ──────────────────────────────────────────────────────
group('SupprimerMessage', () {
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingActionOk, MessagesLoaded] on delete success',
build: () {
when(mockRepository.supprimerMessage('conv-1', 'msg-1'))
.thenAnswer((_) async {});
when(mockRepository.getMessages('conv-1', page: anyNamed('page')))
.thenAnswer((_) async => []);
return bloc;
},
act: (b) => b.add(const SupprimerMessage(
conversationId: 'conv-1',
messageId: 'msg-1',
)),
expect: () => [
isA<MessagingActionOk>().having(
(s) => s.action,
'action',
'supprimer-message',
),
isA<MessagesLoaded>(),
],
);
blocTest<MessagingBloc, MessagingState>(
'emits [MessagingError] on delete failure',
build: () {
when(mockRepository.supprimerMessage(any, any))
.thenThrow(Exception('delete error'));
return bloc;
},
act: (b) => b.add(const SupprimerMessage(
conversationId: 'conv-1',
messageId: 'msg-1',
)),
expect: () => [isA<MessagingError>()],
);
});
// ── NouveauMessageWebSocket ───────────────────────────────────────────────
group('NouveauMessageWebSocket', () {
blocTest<MessagingBloc, MessagingState>(
'reloads messages when current conversation matches',
build: () {
// Open a conversation first so _currentConversationId is set
when(mockRepository.getConversation('conv-1'))
.thenAnswer((_) async => fakeConversation());
when(mockRepository.marquerLu(any)).thenAnswer((_) async {});
when(mockRepository.getMessages('conv-1', page: anyNamed('page')))
.thenAnswer((_) async => [fakeMessage()]);
return bloc;
},
act: (b) async {
b.add(const OpenConversation('conv-1'));
await Future<void>.delayed(const Duration(milliseconds: 50));
b.add(const NouveauMessageWebSocket('conv-1'));
},
expect: () => [
isA<MessagingLoading>(),
isA<ConversationOuverte>(),
isA<MessagesLoaded>(),
],
);
blocTest<MessagingBloc, MessagingState>(
'refreshes conversations list when ws message is for a different conversation',
build: () {
when(mockRepository.getMesConversations())
.thenAnswer((_) async => [fakeConversationSummary()]);
return bloc;
},
act: (b) => b.add(const NouveauMessageWebSocket('conv-other')),
expect: () => [
isA<MessagingLoading>(),
isA<MesConversationsLoaded>(),
],
verify: (_) {
verify(mockRepository.getMesConversations()).called(1);
},
);
});
// ── MarquerLu ─────────────────────────────────────────────────────────────
group('MarquerLu', () {
blocTest<MessagingBloc, MessagingState>(
'calls marquerLu on repository without state change',
build: () {
when(mockRepository.marquerLu('conv-1')).thenAnswer((_) async {});
return bloc;
},
act: (b) => b.add(const MarquerLu('conv-1')),
expect: () => [],
verify: (_) => verify(mockRepository.marquerLu('conv-1')).called(1),
);
});
}