feat(unionflow): ajout Spec-Kit, constitution, mission mutuelles

- Config Spec-Kit pour Spec-Driven Development
- CONSTITUTION.md + .specify/memory/constitution.md
- Commandes Cursor /speckit.*, règles projet
- Mission: associations + mutuelles d'épargne et de financement
- .gitignore: versionner config spec-kit unionflow

Made-with: Cursor
This commit is contained in:
dahoud
2026-02-27 14:41:07 +00:00
parent 144b68f8e7
commit b1957c1c81
631 changed files with 104070 additions and 0 deletions

View File

@@ -0,0 +1,296 @@
/// Repository pour la gestion des organisations
/// Interface avec l'API backend OrganizationResource
library organization_repository;
import 'package:dio/dio.dart';
import '../models/organization_model.dart';
/// Interface du repository des organisations
abstract class OrganizationRepository {
/// Récupère la liste des organisations avec pagination
Future<List<OrganizationModel>> getOrganizations({
int page = 0,
int size = 20,
String? recherche,
});
/// Récupère une organisation par son ID
Future<OrganizationModel?> getOrganizationById(String id);
/// Crée une nouvelle organisation
Future<OrganizationModel> createOrganization(OrganizationModel organization);
/// Met à jour une organisation
Future<OrganizationModel> updateOrganization(String id, OrganizationModel organization);
/// Supprime une organisation
Future<void> deleteOrganization(String id);
/// Active une organisation
Future<OrganizationModel> activateOrganization(String id);
/// Suspend une organisation
Future<OrganizationModel> suspendOrganization(String id);
/// Recherche avancée d'organisations
Future<List<OrganizationModel>> searchOrganizations({
String? nom,
TypeOrganization? type,
StatutOrganization? statut,
String? ville,
String? region,
String? pays,
int page = 0,
int size = 20,
});
/// Récupère les statistiques des organisations
Future<Map<String, dynamic>> getOrganizationsStats();
}
/// Implémentation du repository des organisations
class OrganizationRepositoryImpl implements OrganizationRepository {
final Dio _dio;
static const String _baseUrl = '/api/organisations';
OrganizationRepositoryImpl(this._dio);
@override
Future<List<OrganizationModel>> getOrganizations({
int page = 0,
int size = 20,
String? recherche,
}) async {
try {
final queryParams = <String, dynamic>{
'page': page,
'size': size,
};
if (recherche?.isNotEmpty == true) {
queryParams['recherche'] = recherche;
}
final response = await _dio.get(
_baseUrl,
queryParameters: queryParams,
);
if (response.statusCode == 200) {
final List<dynamic> data = response.data as List<dynamic>;
return data
.map((json) => OrganizationModel.fromJson(json as Map<String, dynamic>))
.toList();
} else {
throw Exception('Erreur lors de la récupération des organisations: ${response.statusCode}');
}
} on DioException catch (e) {
throw Exception('Erreur réseau lors de la récupération des organisations: ${e.message}');
} catch (e) {
throw Exception('Erreur inattendue lors de la récupération des organisations: $e');
}
}
@override
Future<OrganizationModel?> getOrganizationById(String id) async {
try {
final response = await _dio.get('$_baseUrl/$id');
if (response.statusCode == 200) {
return OrganizationModel.fromJson(response.data as Map<String, dynamic>);
} else if (response.statusCode == 404) {
return null;
} else {
throw Exception('Erreur lors de la récupération de l\'organisation: ${response.statusCode}');
}
} on DioException catch (e) {
if (e.response?.statusCode == 404) {
return null;
}
throw Exception('Erreur réseau lors de la récupération de l\'organisation: ${e.message}');
} catch (e) {
throw Exception('Erreur inattendue lors de la récupération de l\'organisation: $e');
}
}
@override
Future<OrganizationModel> createOrganization(OrganizationModel organization) async {
try {
final response = await _dio.post(
_baseUrl,
data: organization.toJson(),
);
if (response.statusCode == 201) {
return OrganizationModel.fromJson(response.data as Map<String, dynamic>);
} else {
throw Exception('Erreur lors de la création de l\'organisation: ${response.statusCode}');
}
} on DioException catch (e) {
if (e.response?.statusCode == 400) {
final errorData = e.response?.data;
if (errorData is Map<String, dynamic> && errorData.containsKey('error')) {
throw Exception('Données invalides: ${errorData['error']}');
}
} else if (e.response?.statusCode == 409) {
throw Exception('Une organisation avec ces informations existe déjà');
}
throw Exception('Erreur réseau lors de la création de l\'organisation: ${e.message}');
} catch (e) {
throw Exception('Erreur inattendue lors de la création de l\'organisation: $e');
}
}
@override
Future<OrganizationModel> updateOrganization(String id, OrganizationModel organization) async {
try {
final response = await _dio.put(
'$_baseUrl/$id',
data: organization.toJson(),
);
if (response.statusCode == 200) {
return OrganizationModel.fromJson(response.data as Map<String, dynamic>);
} else {
throw Exception('Erreur lors de la mise à jour de l\'organisation: ${response.statusCode}');
}
} on DioException catch (e) {
if (e.response?.statusCode == 404) {
throw Exception('Organisation non trouvée');
} else if (e.response?.statusCode == 400) {
final errorData = e.response?.data;
if (errorData is Map<String, dynamic> && errorData.containsKey('error')) {
throw Exception('Données invalides: ${errorData['error']}');
}
}
throw Exception('Erreur réseau lors de la mise à jour de l\'organisation: ${e.message}');
} catch (e) {
throw Exception('Erreur inattendue lors de la mise à jour de l\'organisation: $e');
}
}
@override
Future<void> deleteOrganization(String id) async {
try {
final response = await _dio.delete('$_baseUrl/$id');
if (response.statusCode != 200 && response.statusCode != 204) {
throw Exception('Erreur lors de la suppression de l\'organisation: ${response.statusCode}');
}
} on DioException catch (e) {
if (e.response?.statusCode == 404) {
throw Exception('Organisation non trouvée');
} else if (e.response?.statusCode == 400) {
final errorData = e.response?.data;
if (errorData is Map<String, dynamic> && errorData.containsKey('error')) {
throw Exception('Impossible de supprimer: ${errorData['error']}');
}
}
throw Exception('Erreur réseau lors de la suppression de l\'organisation: ${e.message}');
} catch (e) {
throw Exception('Erreur inattendue lors de la suppression de l\'organisation: $e');
}
}
@override
Future<OrganizationModel> activateOrganization(String id) async {
try {
final response = await _dio.post('$_baseUrl/$id/activer');
if (response.statusCode == 200) {
return OrganizationModel.fromJson(response.data as Map<String, dynamic>);
} else {
throw Exception('Erreur lors de l\'activation de l\'organisation: ${response.statusCode}');
}
} on DioException catch (e) {
if (e.response?.statusCode == 404) {
throw Exception('Organisation non trouvée');
}
throw Exception('Erreur réseau lors de l\'activation de l\'organisation: ${e.message}');
} catch (e) {
throw Exception('Erreur inattendue lors de l\'activation de l\'organisation: $e');
}
}
@override
Future<OrganizationModel> suspendOrganization(String id) async {
try {
final response = await _dio.post('$_baseUrl/$id/suspendre');
if (response.statusCode == 200) {
return OrganizationModel.fromJson(response.data as Map<String, dynamic>);
} else {
throw Exception('Erreur lors de la suspension de l\'organisation: ${response.statusCode}');
}
} on DioException catch (e) {
if (e.response?.statusCode == 404) {
throw Exception('Organisation non trouvée');
}
throw Exception('Erreur réseau lors de la suspension de l\'organisation: ${e.message}');
} catch (e) {
throw Exception('Erreur inattendue lors de la suspension de l\'organisation: $e');
}
}
@override
Future<List<OrganizationModel>> searchOrganizations({
String? nom,
TypeOrganization? type,
StatutOrganization? statut,
String? ville,
String? region,
String? pays,
int page = 0,
int size = 20,
}) async {
try {
final queryParams = <String, dynamic>{
'page': page,
'size': size,
};
if (nom?.isNotEmpty == true) queryParams['nom'] = nom;
if (type != null) queryParams['type'] = type.name.toUpperCase();
if (statut != null) queryParams['statut'] = statut.name.toUpperCase();
if (ville?.isNotEmpty == true) queryParams['ville'] = ville;
if (region?.isNotEmpty == true) queryParams['region'] = region;
if (pays?.isNotEmpty == true) queryParams['pays'] = pays;
final response = await _dio.get(
'$_baseUrl/recherche',
queryParameters: queryParams,
);
if (response.statusCode == 200) {
final List<dynamic> data = response.data as List<dynamic>;
return data
.map((json) => OrganizationModel.fromJson(json as Map<String, dynamic>))
.toList();
} else {
throw Exception('Erreur lors de la recherche d\'organisations: ${response.statusCode}');
}
} on DioException catch (e) {
throw Exception('Erreur réseau lors de la recherche d\'organisations: ${e.message}');
} catch (e) {
throw Exception('Erreur inattendue lors de la recherche d\'organisations: $e');
}
}
@override
Future<Map<String, dynamic>> getOrganizationsStats() async {
try {
final response = await _dio.get('$_baseUrl/statistiques');
if (response.statusCode == 200) {
return response.data as Map<String, dynamic>;
} else {
throw Exception('Erreur lors de la récupération des statistiques: ${response.statusCode}');
}
} on DioException catch (e) {
throw Exception('Erreur réseau lors de la récupération des statistiques: ${e.message}');
} catch (e) {
throw Exception('Erreur inattendue lors de la récupération des statistiques: $e');
}
}
}