Skip to content
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
70 commits
Select commit Hold shift + click to select a range
ef5414a
docs(roadmap): atualizar para Sprint 3 Parte 2
Dec 12, 2025
a57c34c
feat(locations): adicionar endpoints CRUD admin para AllowedCities
Dec 12, 2025
313a622
wip: migrar Commands/Queries/Handlers para implementação própria do M…
Dec 12, 2025
b0cb72f
test(locations): adicionar testes unitários e de integração para Allo…
Dec 12, 2025
ce1f3fa
test(locations): adicionar testes E2E para endpoints AllowedCities
Dec 12, 2025
2f81213
fix(locations): registrar handlers no DI container e corrigir parâmet…
Dec 12, 2025
4cbde3a
refactor(locations): IbgeService agora usa banco de dados ao invés de…
Dec 12, 2025
ea1a963
feat(locations): implementar exception handling com status codes HTTP…
Dec 12, 2025
0c48b7a
fix(locations): remover 15 warnings do módulo Locations
Dec 12, 2025
0f7ff0a
docs(roadmap): atualizar progresso do Sprint 3 Parte 2
Dec 12, 2025
d1ce745
fix: corrigir erros de compilação e exception handling em E2E tests
Dec 12, 2025
c21e2c3
docs: adicionar tarefas originais do roadmap à Sprint 3 Parte 4
Dec 12, 2025
e8683c0
refactor: substituir NSubstitute por Moq para padronização
Dec 12, 2025
0a44810
refactor: substituir Guid.CreateVersion7() por UuidGenerator.NewId()
Dec 12, 2025
1de5dc1
build: migrar solução para formato .slnx (.NET 9+)
Dec 12, 2025
ae6ef2d
feat(docs): automação completa de OpenAPI e GitHub Pages
Dec 12, 2025
ae65929
feat(scripts): adicionar scripts de seeding de dados
Dec 12, 2025
fe5a964
feat(shared): implementar seeding automático de dados de desenvolvimento
Dec 12, 2025
3d2b260
feat(aspire): migrar migrations para Aspire AppHost e remover Migrati…
Dec 12, 2025
53943da
feat(providers): implementar integração Providers ↔ ServiceCatalogs
Dec 12, 2025
32f854a
Merge branch 'master' into sprint3-parte2-admin-endpoints
Dec 12, 2025
e334c4d
fix: resolve build errors and code quality issues
Dec 12, 2025
7b37dc6
docs(roadmap): atualizar Sprint 3 como 100% completo
Dec 12, 2025
f4fac92
fix: ajustes finais em testes e cache service
Dec 12, 2025
fe94741
fix: corrigir assinaturas dos métodos de seeding
Dec 12, 2025
fe216b9
fix: corrigir erros de compilação e formatar código
Dec 12, 2025
5c16c0c
fix: implementar tuple (value, isCached) no ICacheService.GetAsync e …
Dec 12, 2025
7c2c0ea
fix: implementar melhorias do PR review
Dec 12, 2025
27e9113
feat: adicionar Bruno Collections completas para ServiceCatalogs
Dec 12, 2025
b09fcc8
fix: resolver feedback de code review e testes de integração
Dec 12, 2025
9bfa947
fix: usar idMap nos services + documentar scripts de coverage
Dec 12, 2025
b0b9470
docs: auditoria completa e documentação de todos os scripts
Dec 13, 2025
ef3aa47
refactor: simplificar scripts - remover redundâncias e over-engineering
Dec 13, 2025
afb3b17
refactor: simplificar infrastructure - remover redundâncias
Dec 13, 2025
18550b0
fix: aplicar correções de code review
Dec 13, 2025
618d193
chore: aplicar dotnet format e remover documento de trabalho
Dec 13, 2025
5b0d74a
fix: corrigir issues de code review
Dec 13, 2025
c5f79b4
fix(ci): corrigir falso positivo no check de formatação
Dec 13, 2025
fa50798
fix(ci): corrigir check de formatação em ambos workflows
Dec 13, 2025
d8bb00d
refactor: resolve SonarQube warnings (S1135 TODOs, S2139 exception ha…
Dec 13, 2025
dd768d0
fix(ci): resolve pr-validation workflow issues - fix trailing whitesp…
Dec 13, 2025
2a349f5
fix: apply code review feedback - improve cancellation handling, cach…
Dec 13, 2025
302c5e0
docs(roadmap): atualizar Sprint 3 como 100% concluída - todas as 4 pa…
Dec 13, 2025
9f7b09a
refactor: remover ExampleSchemaFilter problemático e seus testes
Dec 13, 2025
2559cb8
fix: corrigir erros de compilação críticos
Dec 13, 2025
5543174
fix: aplicar correções do code review
Dec 13, 2025
311a6fe
refactor: remover overengineering - ModuleTagsDocumentFilter e CacheW…
Dec 13, 2025
2e9e085
fix(ci): focar dotnet format apenas em style (não code quality analyz…
Dec 13, 2025
72115b6
fix(tests): UserProfileUpdatedDomainEventHandler deve re-lançar exceç…
Dec 13, 2025
6ca7dcc
fix(tests): Providers event handlers devem re-lançar exceção original
Dec 13, 2025
c76bc62
fix(code-review): apply GitHub bot suggestions - security, robustness…
Dec 13, 2025
4e26c1a
fix(tests): UserProfileUpdatedDomainEventHandler test deve esperar ex…
Dec 13, 2025
da56552
fix(code-review): aplicar sugestões críticas do CodeRabbit
Dec 13, 2025
0a666cf
fix(build): DapperConnection.HandleDapperError deve retornar exceção
Dec 13, 2025
bc2d0e9
fix(warnings): resolve critical SonarQube warnings without pragma sup…
Dec 14, 2025
0fd9c0c
fix(warnings): resolve all actionable warnings without pragma suppres…
Dec 14, 2025
0506176
fix(tests): update AzureBlobStorageService test name to match excepti…
Dec 14, 2025
8f5ff20
fix(editorconfig): add trailing newline to comply with insert_final_n…
Dec 14, 2025
95436e2
fix(code-quality): address security, exception handling, and document…
Dec 14, 2025
15d1263
fix(exceptions): preserve exception types to maintain HTTP status cod…
Dec 14, 2025
3bfede4
fix(security,performance,tests): prevent PII leakage, optimize token …
Dec 14, 2025
ff1fe58
fix(tests): update RequestLoggingMiddleware test for unwrapped except…
Dec 14, 2025
f88b246
fix(warnings,migrations,api): resolve build warnings and environment …
Dec 14, 2025
c9f268c
fix(security,tests): remove PII from logs and update ArgumentExceptio…
Dec 14, 2025
213bf76
fix(auth,tests): use dynamic token expiry and update ArgumentExceptio…
Dec 14, 2025
29543fd
fix(auth,security): correct HybridCache factory signature and prevent…
Dec 14, 2025
8b5d0f4
fix(auth,security): correct HybridCache factory signature and prevent…
Dec 14, 2025
3b67333
refactor(auth): simplify HybridCache factory with async ValueTask lambda
Dec 14, 2025
184ce63
fix(auth): use dynamic token expiration and improve cache factory sig…
Dec 14, 2025
3cbb5f8
fix(auth): remove redundant token fetch before cache check
Dec 14, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
feat(aspire): migrar migrations para Aspire AppHost e remover Migrati…
…onTool

- MigrationExtensions.cs no AppHost/Extensions:
  - WithMigrations() extension method para PostgreSQL resources
  - MigrationHostedService roda migrations na inicialização
  - Descobre todos os DbContexts automaticamente via reflection
  - Lê connection string de variáveis de ambiente Aspire
  - Logging detalhado de progresso
  - Retry automático em caso de falha
- Integrado em Program.cs:
  - Development: postgresql.MainDatabase.WithMigrations()
  - Testing: postgresql.MainDatabase.WithMigrations()
- Removida pasta tools/MigrationTool (obsoleta)

Vantagens sobre MigrationTool:
- Integração nativa com Aspire (WaitFor, recursos)
- Connection strings do Aspire (sem hardcoding)
- Logs no Aspire Dashboard
- Execução automática na ordem correta
- Sem necessidade de script separado

Migration agora é parte da orquestração Aspire
  • Loading branch information
Filipe Frigini committed Dec 12, 2025
commit 3d2b260b1188b2ad11ca4c0f86310b6f8c2dd7aa
196 changes: 196 additions & 0 deletions src/Aspire/MeAjudaAi.AppHost/Extensions/MigrationExtensions.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,196 @@
using System.Reflection;
using Aspire.Hosting;
using Aspire.Hosting.ApplicationModel;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;

namespace MeAjudaAi.AppHost.Extensions;

/// <summary>
/// Extensões para aplicar migrations automaticamente no Aspire
/// </summary>
public static class MigrationExtensions
{
/// <summary>
/// Adiciona e executa migrations de todos os módulos antes de iniciar a aplicação
/// </summary>
public static IResourceBuilder<T> WithMigrations<T>(
this IResourceBuilder<T> builder) where T : IResourceWithConnectionString
{
builder.ApplicationBuilder.Services.AddHostedService<MigrationHostedService>();
return builder;
}
}

/// <summary>
/// Hosted service que roda migrations na inicialização do AppHost
/// </summary>
internal class MigrationHostedService : IHostedService
{
private readonly ILogger<MigrationHostedService> _logger;
private readonly IServiceProvider _serviceProvider;

public MigrationHostedService(
ILogger<MigrationHostedService> logger,
IServiceProvider serviceProvider)
{
_logger = logger;
_serviceProvider = serviceProvider;
}
Comment thread
frigini marked this conversation as resolved.

public async Task StartAsync(CancellationToken cancellationToken)
{
_logger.LogInformation("🔄 Iniciando migrations de todos os módulos...");

try
{
var connectionString = GetConnectionString();
if (string.IsNullOrEmpty(connectionString))
{
_logger.LogWarning("⚠️ Connection string não encontrada, pulando migrations");
return;
}
Comment thread
frigini marked this conversation as resolved.

var dbContextTypes = DiscoverDbContextTypes();
_logger.LogInformation("📋 Encontrados {Count} DbContexts para migração", dbContextTypes.Count);

foreach (var contextType in dbContextTypes)
{
await MigrateDbContextAsync(contextType, connectionString, cancellationToken);
}

_logger.LogInformation("✅ Todas as migrations foram aplicadas com sucesso!");
}
catch (Exception ex)
{
_logger.LogError(ex, "❌ Erro ao aplicar migrations");
throw;
}
Comment thread
frigini marked this conversation as resolved.
}

public Task StopAsync(CancellationToken cancellationToken) => Task.CompletedTask;

private string? GetConnectionString()
{
// Tenta obter de variáveis de ambiente (padrão Aspire)
var host = Environment.GetEnvironmentVariable("POSTGRES_HOST")
?? Environment.GetEnvironmentVariable("DB_HOST")
?? "localhost";
var port = Environment.GetEnvironmentVariable("POSTGRES_PORT")
?? Environment.GetEnvironmentVariable("DB_PORT")
?? "5432";
var database = Environment.GetEnvironmentVariable("POSTGRES_DB")
?? Environment.GetEnvironmentVariable("MAIN_DATABASE")
?? "meajudaai";
var username = Environment.GetEnvironmentVariable("POSTGRES_USER")
?? Environment.GetEnvironmentVariable("DB_USERNAME")
?? "postgres";
var password = Environment.GetEnvironmentVariable("POSTGRES_PASSWORD")
?? Environment.GetEnvironmentVariable("DB_PASSWORD")
?? "test123";

return $"Host={host};Port={port};Database={database};Username={username};Password={password}";
}
Comment thread
frigini marked this conversation as resolved.

private List<Type> DiscoverDbContextTypes()
{
var dbContextTypes = new List<Type>();
var assemblies = AppDomain.CurrentDomain.GetAssemblies()
.Where(a => a.FullName?.Contains("MeAjudaAi.Modules") == true)
.ToList();

foreach (var assembly in assemblies)
{
try
{
var types = assembly.GetTypes()
.Where(t => t.IsClass && !t.IsAbstract && typeof(DbContext).IsAssignableFrom(t))
.Where(t => t.Name.EndsWith("DbContext"))
.ToList();

dbContextTypes.AddRange(types);
}
catch (Exception ex)
{
_logger.LogWarning(ex, "⚠️ Erro ao descobrir tipos no assembly {AssemblyName}", assembly.FullName);
}
}

return dbContextTypes;
}
Comment thread
frigini marked this conversation as resolved.

private async Task MigrateDbContextAsync(Type contextType, string connectionString, CancellationToken cancellationToken)
{
var moduleName = ExtractModuleName(contextType);
_logger.LogInformation("🔧 Aplicando migrations para {Module}...", moduleName);

try
{
// Criar DbContextOptionsBuilder dinamicamente
var optionsBuilderType = typeof(DbContextOptionsBuilder<>).MakeGenericType(contextType);
var optionsBuilder = Activator.CreateInstance(optionsBuilderType) as DbContextOptionsBuilder;

if (optionsBuilder == null)
{
throw new InvalidOperationException($"Não foi possível criar DbContextOptionsBuilder para {contextType.Name}");
}

// Configurar PostgreSQL
optionsBuilder.UseNpgsql(connectionString, npgsqlOptions =>
{
npgsqlOptions.MigrationsAssembly(contextType.Assembly.FullName);
npgsqlOptions.EnableRetryOnFailure(maxRetryCount: 3);
});

// Criar instância do DbContext
var context = Activator.CreateInstance(contextType, optionsBuilder.Options) as DbContext;

if (context == null)
{
throw new InvalidOperationException($"Não foi possível criar instância de {contextType.Name}");
}
Comment thread
frigini marked this conversation as resolved.
Comment thread
frigini marked this conversation as resolved.

using (context)
{
// Aplicar migrations
var pendingMigrations = (await context.Database.GetPendingMigrationsAsync(cancellationToken)).ToList();

if (pendingMigrations.Any())
{
_logger.LogInformation("📦 {Module}: {Count} migrations pendentes", moduleName, pendingMigrations.Count);
foreach (var migration in pendingMigrations)
{
_logger.LogDebug(" - {Migration}", migration);
}

await context.Database.MigrateAsync(cancellationToken);
_logger.LogInformation("✅ {Module}: Migrations aplicadas com sucesso", moduleName);
}
else
{
_logger.LogInformation("✓ {Module}: Nenhuma migration pendente", moduleName);
}
}
}
catch (Exception ex)
{
_logger.LogError(ex, "❌ Erro ao aplicar migrations para {Module}", moduleName);
throw;
}
}

private string ExtractModuleName(Type contextType)
{
// Extrai nome do módulo do namespace (ex: MeAjudaAi.Modules.Users.Infrastructure.Persistence.UsersDbContext -> Users)
var namespaceParts = contextType.Namespace?.Split('.') ?? Array.Empty<string>();
var moduleIndex = Array.IndexOf(namespaceParts, "Modules");

if (moduleIndex >= 0 && moduleIndex + 1 < namespaceParts.Length)
{
return namespaceParts[moduleIndex + 1];
}

return contextType.Name.Replace("DbContext", "");
}
}
6 changes: 6 additions & 0 deletions src/Aspire/MeAjudaAi.AppHost/Program.cs
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,9 @@ private static void ConfigureTestingEnvironment(IDistributedApplicationBuilder b
options.Password = testDbPassword;
});

// Aplicar migrations automaticamente (Testing também)
postgresql.MainDatabase.WithMigrations();

var redis = builder.AddRedis("redis");

_ = builder.AddProject<Projects.MeAjudaAi_ApiService>("apiservice")
Expand Down Expand Up @@ -107,6 +110,9 @@ private static void ConfigureDevelopmentEnvironment(IDistributedApplicationBuild
options.IncludePgAdmin = includePgAdmin;
});

// Aplicar migrations automaticamente
postgresql.MainDatabase.WithMigrations();

var redis = builder.AddRedis("redis");

var rabbitMq = builder.AddRabbitMQ("rabbitmq");
Expand Down
29 changes: 0 additions & 29 deletions tools/MigrationTool/MigrationTool.csproj

This file was deleted.

Loading