feat: WebSocket temps réel + Finance Workflow + corrections

- Task #6: WebSocket /ws/dashboard + Kafka events (5 topics)
  * Backend: KafkaEventProducer, KafkaEventConsumer
  * Mobile: WebSocketService (reconnection, heartbeat, typed events)
  * DashboardBloc: Auto-refresh depuis WebSocket events

- Finance Workflow: approbations + budgets (backend + mobile)
  * Backend: entities, services, resources, migrations Flyway V6
  * Mobile: features finance_workflow complète avec BLoC

- Corrections DI: interfaces IRepository partout
  * IProfileRepository, IOrganizationRepository, IMembreRepository
  * GetIt configuré avec @injectable

- Spec-Kit: constitution + templates mis à jour
  * .specify/memory/constitution.md enrichie
  * Templates agent, plan, spec, tasks, checklist

- Nettoyage: fichiers temporaires supprimés

Signed-off-by: lions dev Team
This commit is contained in:
dahoud
2026-03-15 02:12:17 +00:00
parent bbc409de9d
commit e8ad874015
635 changed files with 58160 additions and 20674 deletions

View File

@@ -0,0 +1,75 @@
/// Tests unitaires pour DeleteAccount use case
library delete_account_test;
import 'package:flutter_test/flutter_test.dart';
import 'package:mockito/annotations.dart';
import 'package:mockito/mockito.dart';
import 'package:unionflow_mobile_apps/features/profile/domain/repositories/profile_repository.dart';
import 'package:unionflow_mobile_apps/features/profile/domain/usecases/delete_account.dart';
@GenerateMocks([IProfileRepository])
import 'delete_account_test.mocks.dart';
void main() {
late DeleteAccount useCase;
late MockIProfileRepository mockRepository;
setUp(() {
mockRepository = MockIProfileRepository();
useCase = DeleteAccount(mockRepository);
});
group('DeleteAccount Use Case', () {
const tMembreId = 'membre1';
test('should delete account successfully (soft delete)', () async {
// Arrange
when(mockRepository.deleteAccount(tMembreId))
.thenAnswer((_) async => Future.value());
// Act
await useCase(tMembreId);
// Assert
verify(mockRepository.deleteAccount(tMembreId));
verifyNoMoreInteractions(mockRepository);
});
test('should throw exception when account not found', () async {
// Arrange
when(mockRepository.deleteAccount(any))
.thenThrow(Exception('Compte non trouvé'));
// Act & Assert
expect(
() => useCase(tMembreId),
throwsA(isA<Exception>()),
);
verify(mockRepository.deleteAccount(tMembreId));
});
test('should throw exception when account is already deleted', () async {
// Arrange
when(mockRepository.deleteAccount(any))
.thenThrow(Exception('Compte déjà désactivé'));
// Act & Assert
expect(
() => useCase(tMembreId),
throwsA(isA<Exception>()),
);
});
test('should throw exception when deletion fails', () async {
// Arrange
when(mockRepository.deleteAccount(any))
.thenThrow(Exception('Deletion failed'));
// Act & Assert
expect(
() => useCase(tMembreId),
throwsException,
);
});
});
}

View File

@@ -0,0 +1,84 @@
/// Tests unitaires pour GetProfile use case
library get_profile_test;
import 'package:flutter_test/flutter_test.dart';
import 'package:mockito/annotations.dart';
import 'package:mockito/mockito.dart';
import 'package:unionflow_mobile_apps/features/profile/domain/repositories/profile_repository.dart';
import 'package:unionflow_mobile_apps/features/profile/domain/usecases/get_profile.dart';
import 'package:unionflow_mobile_apps/features/members/data/models/membre_complete_model.dart';
@GenerateMocks([IProfileRepository])
import 'get_profile_test.mocks.dart';
void main() {
late GetProfile useCase;
late MockIProfileRepository mockRepository;
setUp(() {
mockRepository = MockIProfileRepository();
useCase = GetProfile(mockRepository);
});
group('GetProfile Use Case', () {
final tMembre = MembreCompletModel(
id: 'membre1',
nom: 'Dupont',
prenom: 'Jean',
email: 'jean.dupont@example.com',
telephone: '+33612345678',
dateNaissance: DateTime(1990, 1, 1),
);
test('should return current user profile from repository', () async {
// Arrange
when(mockRepository.getMe()).thenAnswer((_) async => tMembre);
// Act
final result = await useCase();
// Assert
expect(result, equals(tMembre));
verify(mockRepository.getMe());
verifyNoMoreInteractions(mockRepository);
});
test('should return null when user is not authenticated', () async {
// Arrange
when(mockRepository.getMe()).thenAnswer((_) async => null);
// Act
final result = await useCase();
// Assert
expect(result, isNull);
verify(mockRepository.getMe());
});
test('should throw exception when repository throws', () async {
// Arrange
when(mockRepository.getMe()).thenThrow(Exception('Unauthorized'));
// Act & Assert
expect(
() => useCase(),
throwsA(isA<Exception>()),
);
verify(mockRepository.getMe());
});
test('should cache profile data on successful retrieval', () async {
// Arrange
when(mockRepository.getMe()).thenAnswer((_) async => tMembre);
// Act - Call twice
final result1 = await useCase();
final result2 = await useCase();
// Assert - Repository called twice (no caching at use case level)
expect(result1, equals(tMembre));
expect(result2, equals(tMembre));
verify(mockRepository.getMe()).called(2);
});
});
}

View File

@@ -0,0 +1,95 @@
/// Tests unitaires pour UpdateAvatar use case
library update_avatar_test;
import 'package:flutter_test/flutter_test.dart';
import 'package:mockito/annotations.dart';
import 'package:mockito/mockito.dart';
import 'package:unionflow_mobile_apps/features/profile/domain/repositories/profile_repository.dart';
import 'package:unionflow_mobile_apps/features/profile/domain/usecases/update_avatar.dart';
import 'package:unionflow_mobile_apps/features/members/data/models/membre_complete_model.dart';
@GenerateMocks([IProfileRepository])
import 'update_avatar_test.mocks.dart';
void main() {
late UpdateAvatar useCase;
late MockIProfileRepository mockRepository;
setUp(() {
mockRepository = MockIProfileRepository();
useCase = UpdateAvatar(mockRepository);
});
group('UpdateAvatar Use Case', () {
const tMembreId = 'membre1';
const tPhotoUrl = 'https://example.com/avatar.jpg';
final tUpdatedMembre = MembreCompletModel(
id: tMembreId,
nom: 'Dupont',
prenom: 'Jean',
email: 'jean.dupont@example.com',
photo: tPhotoUrl,
);
test('should update avatar successfully', () async {
// Arrange
when(mockRepository.updateAvatar(tMembreId, tPhotoUrl))
.thenAnswer((_) async => tUpdatedMembre);
// Act
final result = await useCase(tMembreId, tPhotoUrl);
// Assert
expect(result, equals(tUpdatedMembre));
expect(result.photo, equals(tPhotoUrl));
verify(mockRepository.updateAvatar(tMembreId, tPhotoUrl));
verifyNoMoreInteractions(mockRepository);
});
test('should handle empty photo URL', () async {
// Arrange
const emptyUrl = '';
final emptyPhotoMembre = MembreCompletModel(
id: tMembreId,
nom: 'Dupont',
prenom: 'Jean',
email: 'jean.dupont@example.com',
photo: emptyUrl,
);
when(mockRepository.updateAvatar(tMembreId, emptyUrl))
.thenAnswer((_) async => emptyPhotoMembre);
// Act
final result = await useCase(tMembreId, emptyUrl);
// Assert
expect(result.photo, equals(emptyUrl));
verify(mockRepository.updateAvatar(tMembreId, emptyUrl));
});
test('should throw exception when member not found', () async {
// Arrange
when(mockRepository.updateAvatar(any, any))
.thenThrow(Exception('Membre non trouvé'));
// Act & Assert
expect(
() => useCase(tMembreId, tPhotoUrl),
throwsA(isA<Exception>()),
);
});
test('should throw exception when upload fails', () async {
// Arrange
when(mockRepository.updateAvatar(any, any))
.thenThrow(Exception('Upload failed'));
// Act & Assert
expect(
() => useCase(tMembreId, tPhotoUrl),
throwsException,
);
});
});
}

View File

@@ -0,0 +1,93 @@
/// Tests unitaires pour UpdatePreferences use case
library update_preferences_test;
import 'package:flutter_test/flutter_test.dart';
import 'package:mockito/annotations.dart';
import 'package:mockito/mockito.dart';
import 'package:unionflow_mobile_apps/features/profile/domain/repositories/profile_repository.dart';
import 'package:unionflow_mobile_apps/features/profile/domain/usecases/update_preferences.dart';
@GenerateMocks([IProfileRepository])
import 'update_preferences_test.mocks.dart';
void main() {
late UpdatePreferences useCase;
late MockIProfileRepository mockRepository;
setUp(() {
mockRepository = MockIProfileRepository();
useCase = UpdatePreferences(mockRepository);
});
group('UpdatePreferences Use Case', () {
const tMembreId = 'membre1';
final tPreferences = {
'language': 'fr',
'theme': 'dark',
'notifications': true,
'emailNotifications': false,
};
final tUpdatedPreferences = {
...tPreferences,
'lastUpdated': '2026-03-14T10:00:00Z',
};
test('should update preferences successfully', () async {
// Arrange
when(mockRepository.updatePreferences(tMembreId, tPreferences))
.thenAnswer((_) async => tUpdatedPreferences);
// Act
final result = await useCase(tMembreId, tPreferences);
// Assert
expect(result, equals(tUpdatedPreferences));
expect(result['language'], equals('fr'));
expect(result['theme'], equals('dark'));
verify(mockRepository.updatePreferences(tMembreId, tPreferences));
verifyNoMoreInteractions(mockRepository);
});
test('should update partial preferences', () async {
// Arrange
final partialPrefs = {'theme': 'light'};
final expectedResult = {'theme': 'light'};
when(mockRepository.updatePreferences(tMembreId, partialPrefs))
.thenAnswer((_) async => expectedResult);
// Act
final result = await useCase(tMembreId, partialPrefs);
// Assert
expect(result['theme'], equals('light'));
verify(mockRepository.updatePreferences(tMembreId, partialPrefs));
});
test('should handle empty preferences map', () async {
// Arrange
final emptyPrefs = <String, dynamic>{};
when(mockRepository.updatePreferences(tMembreId, emptyPrefs))
.thenAnswer((_) async => emptyPrefs);
// Act
final result = await useCase(tMembreId, emptyPrefs);
// Assert
expect(result, isEmpty);
verify(mockRepository.updatePreferences(tMembreId, emptyPrefs));
});
test('should throw exception when update fails', () async {
// Arrange
when(mockRepository.updatePreferences(any, any))
.thenThrow(Exception('Failed to update preferences'));
// Act & Assert
expect(
() => useCase(tMembreId, tPreferences),
throwsException,
);
});
});
}

View File

@@ -0,0 +1,97 @@
/// Tests unitaires pour UpdateProfile use case
library update_profile_test;
import 'package:flutter_test/flutter_test.dart';
import 'package:mockito/annotations.dart';
import 'package:mockito/mockito.dart';
import 'package:unionflow_mobile_apps/features/profile/domain/repositories/profile_repository.dart';
import 'package:unionflow_mobile_apps/features/profile/domain/usecases/update_profile.dart';
import 'package:unionflow_mobile_apps/features/members/data/models/membre_complete_model.dart';
@GenerateMocks([IProfileRepository])
import 'update_profile_test.mocks.dart';
void main() {
late UpdateProfile useCase;
late MockIProfileRepository mockRepository;
setUp(() {
mockRepository = MockIProfileRepository();
useCase = UpdateProfile(mockRepository);
});
group('UpdateProfile Use Case', () {
const tMembreId = 'membre1';
final tMembre = MembreCompletModel(
id: tMembreId,
nom: 'Dupont',
prenom: 'Jean',
email: 'jean.dupont@example.com',
telephone: '+33612345678',
dateNaissance: DateTime(1990, 1, 1),
);
final tUpdatedMembre = MembreCompletModel(
id: tMembreId,
nom: 'Dupont',
prenom: 'Jean',
email: 'jean.dupont@example.com',
telephone: '+33698765432', // Updated phone
dateNaissance: DateTime(1990, 1, 1),
adresse: '123 Rue de Paris', // Added address
);
test('should update profile successfully', () async {
// Arrange
when(mockRepository.updateProfile(tMembreId, tMembre))
.thenAnswer((_) async => tUpdatedMembre);
// Act
final result = await useCase(tMembreId, tMembre);
// Assert
expect(result, equals(tUpdatedMembre));
verify(mockRepository.updateProfile(tMembreId, tMembre));
verifyNoMoreInteractions(mockRepository);
});
test('should update only specified fields', () async {
// Arrange
when(mockRepository.updateProfile(any, any))
.thenAnswer((_) async => tUpdatedMembre);
// Act
final result = await useCase(tMembreId, tMembre);
// Assert
expect(result.telephone, equals('+33698765432'));
expect(result.adresse, equals('123 Rue de Paris'));
verify(mockRepository.updateProfile(tMembreId, tMembre));
});
test('should throw exception when profile not found', () async {
// Arrange
when(mockRepository.updateProfile(any, any))
.thenThrow(Exception('Profil non trouvé'));
// Act & Assert
expect(
() => useCase(tMembreId, tMembre),
throwsA(isA<Exception>()),
);
verify(mockRepository.updateProfile(tMembreId, tMembre));
});
test('should throw exception when update fails', () async {
// Arrange
when(mockRepository.updateProfile(any, any))
.thenThrow(Exception('Network error'));
// Act & Assert
expect(
() => useCase(tMembreId, tMembre),
throwsException,
);
});
});
}