refactor: Migrate from monolithic to modular architecture

- Restructure project into domain-specific modules (core, masterdata, members, horses, events, infrastructure)
- Create shared client components in common-ui module
- Implement CI/CD workflows with GitHub Actions
- Consolidate documentation in docs directory
- Remove deprecated modules and documentation files
- Add cleanup and migration scripts for transition
- Update README with new project structure and setup instructions
This commit is contained in:
stefan
2025-07-22 18:44:18 +02:00
parent 8229e8e571
commit a256622f37
314 changed files with 5930 additions and 19817 deletions
@@ -0,0 +1,10 @@
plugins {
kotlin("jvm")
}
dependencies {
implementation(projects.masterdata.masterdataDomain)
implementation(projects.core.coreDomain)
implementation(projects.core.coreUtils)
testImplementation(projects.platform.platformTesting)
}
@@ -0,0 +1,338 @@
package at.mocode.masterdata.application.usecase
import at.mocode.masterdata.domain.model.LandDefinition
import at.mocode.masterdata.domain.repository.LandRepository
import at.mocode.core.utils.validation.ValidationResult
import at.mocode.core.utils.validation.ValidationError
import com.benasher44.uuid.Uuid
import kotlinx.datetime.Clock
/**
* Use case for creating and updating country information.
*
* This use case encapsulates the business logic for country management
* including validation, duplicate checking, and persistence.
*/
class CreateCountryUseCase(
private val landRepository: LandRepository
) {
/**
* Request data for creating a new country.
*/
data class CreateCountryRequest(
val isoAlpha2Code: String,
val isoAlpha3Code: String,
val isoNumerischerCode: String? = null,
val nameDeutsch: String,
val nameEnglisch: String? = null,
val wappenUrl: String? = null,
val istEuMitglied: Boolean? = null,
val istEwrMitglied: Boolean? = null,
val istAktiv: Boolean = true,
val sortierReihenfolge: Int? = null
)
/**
* Request data for updating an existing country.
*/
data class UpdateCountryRequest(
val landId: Uuid,
val isoAlpha2Code: String,
val isoAlpha3Code: String,
val isoNumerischerCode: String? = null,
val nameDeutsch: String,
val nameEnglisch: String? = null,
val wappenUrl: String? = null,
val istEuMitglied: Boolean? = null,
val istEwrMitglied: Boolean? = null,
val istAktiv: Boolean = true,
val sortierReihenfolge: Int? = null
)
/**
* Response data for country creation.
*/
data class CreateCountryResponse(
val country: LandDefinition?,
val success: Boolean,
val errors: List<String> = emptyList()
)
/**
* Response data for country update.
*/
data class UpdateCountryResponse(
val country: LandDefinition?,
val success: Boolean,
val errors: List<String> = emptyList()
)
/**
* Response data for country deletion.
*/
data class DeleteCountryResponse(
val success: Boolean,
val errors: List<String> = emptyList()
)
/**
* Creates a new country after validation.
*
* @param request The country creation request
* @return CreateCountryResponse with the created country or validation errors
*/
suspend fun createCountry(request: CreateCountryRequest): CreateCountryResponse {
// Validate the request
val validationResult = validateCreateRequest(request)
if (!validationResult.isValid()) {
val errors = (validationResult as ValidationResult.Invalid).errors.map { it.message }
return CreateCountryResponse(
country = null,
success = false,
errors = errors
)
}
// Check for duplicates
val duplicateCheck = checkForDuplicates(request.isoAlpha2Code, request.isoAlpha3Code)
if (!duplicateCheck.isValid()) {
val errors = (duplicateCheck as ValidationResult.Invalid).errors.map { it.message }
return CreateCountryResponse(
country = null,
success = false,
errors = errors
)
}
// Create the domain object
val now = Clock.System.now()
val country = LandDefinition(
isoAlpha2Code = request.isoAlpha2Code.uppercase(),
isoAlpha3Code = request.isoAlpha3Code.uppercase(),
isoNumerischerCode = request.isoNumerischerCode,
nameDeutsch = request.nameDeutsch.trim(),
nameEnglisch = request.nameEnglisch?.trim(),
wappenUrl = request.wappenUrl?.trim(),
istEuMitglied = request.istEuMitglied,
istEwrMitglied = request.istEwrMitglied,
istAktiv = request.istAktiv,
sortierReihenfolge = request.sortierReihenfolge,
createdAt = now,
updatedAt = now
)
// Save to repository
val savedCountry = landRepository.save(country)
return CreateCountryResponse(
country = savedCountry,
success = true
)
}
/**
* Updates an existing country after validation.
*
* @param request The country update request
* @return ValidationResult containing the updated country or validation errors
*/
suspend fun updateCountry(request: UpdateCountryRequest): UpdateCountryResponse {
// Check if country exists
val existingCountry = landRepository.findById(request.landId)
if (existingCountry == null) {
return UpdateCountryResponse(
country = null,
success = false,
errors = listOf("Country with ID ${request.landId} not found")
)
}
// Validate the request
val validationResult = validateUpdateRequest(request)
if (!validationResult.isValid()) {
val errors = (validationResult as ValidationResult.Invalid).errors.map { it.message }
return UpdateCountryResponse(
country = null,
success = false,
errors = errors
)
}
// Check for duplicates (excluding current country)
val duplicateCheck = checkForDuplicatesExcluding(
request.isoAlpha2Code,
request.isoAlpha3Code,
request.landId
)
if (!duplicateCheck.isValid()) {
val errors = (duplicateCheck as ValidationResult.Invalid).errors.map { it.message }
return UpdateCountryResponse(
country = null,
success = false,
errors = errors
)
}
// Update the domain object
val updatedCountry = existingCountry.copy(
isoAlpha2Code = request.isoAlpha2Code.uppercase(),
isoAlpha3Code = request.isoAlpha3Code.uppercase(),
isoNumerischerCode = request.isoNumerischerCode,
nameDeutsch = request.nameDeutsch.trim(),
nameEnglisch = request.nameEnglisch?.trim(),
wappenUrl = request.wappenUrl?.trim(),
istEuMitglied = request.istEuMitglied,
istEwrMitglied = request.istEwrMitglied,
istAktiv = request.istAktiv,
sortierReihenfolge = request.sortierReihenfolge,
updatedAt = Clock.System.now()
)
// Save to repository
val savedCountry = landRepository.save(updatedCountry)
return UpdateCountryResponse(
country = savedCountry,
success = true
)
}
/**
* Deletes a country by ID.
*
* @param countryId The unique identifier of the country to delete
* @return DeleteCountryResponse indicating success or failure
*/
suspend fun deleteCountry(countryId: Uuid): DeleteCountryResponse {
val deleted = landRepository.delete(countryId)
return if (deleted) {
DeleteCountryResponse(success = true)
} else {
DeleteCountryResponse(
success = false,
errors = listOf("Country with ID $countryId not found or could not be deleted")
)
}
}
/**
* Validates a create country request.
*/
private fun validateCreateRequest(request: CreateCountryRequest): ValidationResult {
val errors = mutableListOf<ValidationError>()
// ISO Alpha-2 Code validation
if (request.isoAlpha2Code.isBlank()) {
errors.add(ValidationError("isoAlpha2Code", "ISO Alpha-2 code is required", "REQUIRED"))
} else if (request.isoAlpha2Code.length != 2) {
errors.add(ValidationError("isoAlpha2Code", "ISO Alpha-2 code must be exactly 2 characters", "INVALID_LENGTH"))
} else if (!request.isoAlpha2Code.all { it.isLetter() }) {
errors.add(ValidationError("isoAlpha2Code", "ISO Alpha-2 code must contain only letters", "INVALID_FORMAT"))
}
// ISO Alpha-3 Code validation
if (request.isoAlpha3Code.isBlank()) {
errors.add(ValidationError("isoAlpha3Code", "ISO Alpha-3 code is required", "REQUIRED"))
} else if (request.isoAlpha3Code.length != 3) {
errors.add(ValidationError("isoAlpha3Code", "ISO Alpha-3 code must be exactly 3 characters", "INVALID_LENGTH"))
} else if (!request.isoAlpha3Code.all { it.isLetter() }) {
errors.add(ValidationError("isoAlpha3Code", "ISO Alpha-3 code must contain only letters", "INVALID_FORMAT"))
}
// German name validation
if (request.nameDeutsch.isBlank()) {
errors.add(ValidationError("nameDeutsch", "German name is required", "REQUIRED"))
} else if (request.nameDeutsch.length > 100) {
errors.add(ValidationError("nameDeutsch", "German name must not exceed 100 characters", "MAX_LENGTH"))
}
// English name validation
request.nameEnglisch?.let { name ->
if (name.length > 100) {
errors.add(ValidationError("nameEnglisch", "English name must not exceed 100 characters", "MAX_LENGTH"))
}
}
// Sorting order validation
request.sortierReihenfolge?.let { order ->
if (order < 0) {
errors.add(ValidationError("sortierReihenfolge", "Sorting order must be non-negative", "INVALID_VALUE"))
}
}
return if (errors.isEmpty()) {
ValidationResult.Valid
} else {
ValidationResult.Invalid(errors)
}
}
/**
* Validates an update country request.
*/
private fun validateUpdateRequest(request: UpdateCountryRequest): ValidationResult {
// Use the same validation logic as create request
val createRequest = CreateCountryRequest(
isoAlpha2Code = request.isoAlpha2Code,
isoAlpha3Code = request.isoAlpha3Code,
isoNumerischerCode = request.isoNumerischerCode,
nameDeutsch = request.nameDeutsch,
nameEnglisch = request.nameEnglisch,
wappenUrl = request.wappenUrl,
istEuMitglied = request.istEuMitglied,
istEwrMitglied = request.istEwrMitglied,
istAktiv = request.istAktiv,
sortierReihenfolge = request.sortierReihenfolge
)
return validateCreateRequest(createRequest)
}
/**
* Checks for duplicate ISO codes.
*/
private suspend fun checkForDuplicates(isoAlpha2Code: String, isoAlpha3Code: String): ValidationResult {
val errors = mutableListOf<ValidationError>()
if (landRepository.existsByIsoAlpha2Code(isoAlpha2Code.uppercase())) {
errors.add(ValidationError("isoAlpha2Code", "Country with ISO Alpha-2 code '${isoAlpha2Code.uppercase()}' already exists", "DUPLICATE"))
}
if (landRepository.existsByIsoAlpha3Code(isoAlpha3Code.uppercase())) {
errors.add(ValidationError("isoAlpha3Code", "Country with ISO Alpha-3 code '${isoAlpha3Code.uppercase()}' already exists", "DUPLICATE"))
}
return if (errors.isEmpty()) {
ValidationResult.Valid
} else {
ValidationResult.Invalid(errors)
}
}
/**
* Checks for duplicate ISO codes excluding a specific country ID.
*/
private suspend fun checkForDuplicatesExcluding(
isoAlpha2Code: String,
isoAlpha3Code: String,
excludeId: Uuid
): ValidationResult {
val errors = mutableListOf<ValidationError>()
// Check Alpha-2 code
val existingAlpha2 = landRepository.findByIsoAlpha2Code(isoAlpha2Code.uppercase())
if (existingAlpha2 != null && existingAlpha2.landId != excludeId) {
errors.add(ValidationError("isoAlpha2Code", "Country with ISO Alpha-2 code '${isoAlpha2Code.uppercase()}' already exists", "DUPLICATE"))
}
// Check Alpha-3 code
val existingAlpha3 = landRepository.findByIsoAlpha3Code(isoAlpha3Code.uppercase())
if (existingAlpha3 != null && existingAlpha3.landId != excludeId) {
errors.add(ValidationError("isoAlpha3Code", "Country with ISO Alpha-3 code '${isoAlpha3Code.uppercase()}' already exists", "DUPLICATE"))
}
return if (errors.isEmpty()) {
ValidationResult.Valid
} else {
ValidationResult.Invalid(errors)
}
}
}
@@ -0,0 +1,120 @@
package at.mocode.masterdata.application.usecase
import at.mocode.masterdata.domain.model.LandDefinition
import at.mocode.masterdata.domain.repository.LandRepository
import com.benasher44.uuid.Uuid
/**
* Use case for retrieving country information.
*
* This use case encapsulates the business logic for fetching country data
* and provides a clean interface for the application layer.
*/
class GetCountryUseCase(
private val landRepository: LandRepository
) {
/**
* Retrieves a country by its unique ID.
*
* @param countryId The unique identifier of the country
* @return The country if found, null otherwise
*/
suspend fun getById(countryId: Uuid): LandDefinition? {
return landRepository.findById(countryId)
}
/**
* Retrieves a country by its ISO Alpha-2 code.
*
* @param isoCode The 2-letter ISO code (e.g., "AT", "DE")
* @return The country if found, null otherwise
*/
suspend fun getByIsoAlpha2Code(isoCode: String): LandDefinition? {
require(isoCode.length == 2) { "ISO Alpha-2 code must be exactly 2 characters" }
return landRepository.findByIsoAlpha2Code(isoCode.uppercase())
}
/**
* Retrieves a country by its ISO Alpha-3 code.
*
* @param isoCode The 3-letter ISO code (e.g., "AUT", "DEU")
* @return The country if found, null otherwise
*/
suspend fun getByIsoAlpha3Code(isoCode: String): LandDefinition? {
require(isoCode.length == 3) { "ISO Alpha-3 code must be exactly 3 characters" }
return landRepository.findByIsoAlpha3Code(isoCode.uppercase())
}
/**
* Searches for countries by name (partial match).
*
* @param searchTerm The search term to match against country names
* @param limit Maximum number of results to return (default: 50)
* @return List of matching countries
*/
suspend fun searchByName(searchTerm: String, limit: Int = 50): List<LandDefinition> {
require(searchTerm.isNotBlank()) { "Search term cannot be blank" }
require(limit > 0) { "Limit must be positive" }
return landRepository.findByName(searchTerm.trim(), limit)
}
/**
* Retrieves all active countries.
*
* @param orderBySortierung Whether to order by sortierReihenfolge field (default: true)
* @return List of active countries
*/
suspend fun getAllActive(orderBySortierung: Boolean = true): List<LandDefinition> {
return landRepository.findAllActive(orderBySortierung)
}
/**
* Retrieves all EU member countries.
*
* @return List of EU member countries
*/
suspend fun getEuMembers(): List<LandDefinition> {
return landRepository.findEuMembers()
}
/**
* Retrieves all EWR (European Economic Area) member countries.
*
* @return List of EWR member countries
*/
suspend fun getEwrMembers(): List<LandDefinition> {
return landRepository.findEwrMembers()
}
/**
* Checks if a country with the given ISO Alpha-2 code exists.
*
* @param isoCode The ISO Alpha-2 code to check
* @return true if a country with this code exists, false otherwise
*/
suspend fun existsByIsoAlpha2Code(isoCode: String): Boolean {
require(isoCode.length == 2) { "ISO Alpha-2 code must be exactly 2 characters" }
return landRepository.existsByIsoAlpha2Code(isoCode.uppercase())
}
/**
* Checks if a country with the given ISO Alpha-3 code exists.
*
* @param isoCode The ISO Alpha-3 code to check
* @return true if a country with this code exists, false otherwise
*/
suspend fun existsByIsoAlpha3Code(isoCode: String): Boolean {
require(isoCode.length == 3) { "ISO Alpha-3 code must be exactly 3 characters" }
return landRepository.existsByIsoAlpha3Code(isoCode.uppercase())
}
/**
* Counts the total number of active countries.
*
* @return The total count of active countries
*/
suspend fun countActive(): Long {
return landRepository.countActive()
}
}