<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Lucas Costa</title>
    <description>The latest articles on DEV Community by Lucas Costa (@lucasscosta).</description>
    <link>https://dev.to/lucasscosta</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F1976899%2F5e9d9c4a-c22c-41c9-9168-e781b9cecb28.jpg</url>
      <title>DEV Community: Lucas Costa</title>
      <link>https://dev.to/lucasscosta</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/lucasscosta"/>
    <language>en</language>
    <item>
      <title>Do Zero à Produção: EDA, Clean Arch e Observabilidade com Go e Kotlin</title>
      <dc:creator>Lucas Costa</dc:creator>
      <pubDate>Sun, 05 Apr 2026 19:48:18 +0000</pubDate>
      <link>https://dev.to/lucasscosta/do-zero-a-producao-eda-clean-arch-e-observabilidade-com-go-e-kotlin-23ln</link>
      <guid>https://dev.to/lucasscosta/do-zero-a-producao-eda-clean-arch-e-observabilidade-com-go-e-kotlin-23ln</guid>
      <description>&lt;blockquote&gt;
&lt;p&gt;"Ler sobre arquitetura é fácil. Tomar decisões reais de produção é onde o jogo acontece."&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;Neste projeto, decidi construir um pipeline de preços que simula um ambiente de alta disponibilidade. O objetivo não era apenas fazer o código funcionar, mas garantir que ele fosse &lt;strong&gt;resiliente, escalável e, acima de tudo, visível&lt;/strong&gt;.&lt;/p&gt;




&lt;h2&gt;
  
  
  🛠️ A Arquitetura do Sistema
&lt;/h2&gt;

&lt;p&gt;O fluxo de dados foi desenhado para ser totalmente desacoplado através de &lt;strong&gt;EDA (Event-Driven Architecture)&lt;/strong&gt;:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;graph LR
    A[POST /prices] --&amp;gt; B(price-crawler Go)
    B --&amp;gt; C{Kafka}
    C --&amp;gt; D(price-processor Kotlin)
    D --&amp;gt; E[(PostgreSQL)]
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;price-crawler (Go)&lt;/code&gt;: Responsável por receber o dado via HTTP, persistir localmente e garantir o envio ao Kafka.&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;price-processor (Kotlin)&lt;/code&gt;: Consome os eventos, aplica validações de negócio e persiste no banco de dados final.&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  🏗️ Clean Architecture: O Domínio como "Coração"
&lt;/h2&gt;

&lt;p&gt;No Kotlin (price-processor), utilizei uma abordagem de módulos Gradle para garantir o isolamento total:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Módulo Domain: Contém apenas POJOs e lógica pura, sem dependências de frameworks como Spring ou bibliotecas de mensageria.&lt;/li&gt;
&lt;li&gt;Módulo Infrastructure: Onde residem as implementações concretas de persistência e consumo de mensagens.
&lt;/li&gt;
&lt;/ul&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight kotlin"&gt;&lt;code&gt;&lt;span class="c1"&gt;// domain module — zero Spring, zero Kafka&lt;/span&gt;
&lt;span class="kd"&gt;data class&lt;/span&gt; &lt;span class="nc"&gt;PriceEvent&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;
    &lt;span class="kd"&gt;val&lt;/span&gt; &lt;span class="py"&gt;id&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nc"&gt;String&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="kd"&gt;val&lt;/span&gt; &lt;span class="py"&gt;product&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nc"&gt;String&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="kd"&gt;val&lt;/span&gt; &lt;span class="py"&gt;source&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nc"&gt;String&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="kd"&gt;val&lt;/span&gt; &lt;span class="py"&gt;price&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nc"&gt;BigDecimal&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="kd"&gt;val&lt;/span&gt; &lt;span class="py"&gt;timestamp&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nc"&gt;Instant&lt;/span&gt;
&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;fun&lt;/span&gt; &lt;span class="nf"&gt;validate&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
        &lt;span class="nf"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;product&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;isNotBlank&lt;/span&gt;&lt;span class="p"&gt;())&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="s"&gt;"product é obrigatório"&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt;
        &lt;span class="nf"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;price&lt;/span&gt; &lt;span class="p"&gt;&amp;gt;&lt;/span&gt; &lt;span class="nc"&gt;BigDecimal&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nc"&gt;ZERO&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="s"&gt;"price deve ser positivo"&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt;
        &lt;span class="nf"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;source&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="n"&gt;length&lt;/span&gt; &lt;span class="p"&gt;&amp;lt;=&lt;/span&gt; &lt;span class="mi"&gt;100&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="s"&gt;"source excede 100 caracteres"&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt;
    &lt;span class="p"&gt;}&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;






&lt;h2&gt;
  
  
  📦 Resiliência Extrema: Transactional Outbox + Circuit Breaker
&lt;/h2&gt;

&lt;p&gt;Para evitar a perda de dados caso o Kafka fique indisponível, implementei o Transactional Outbox no serviço em Go:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;O dado é salvo na tabela outbox dentro da mesma transação do banco de dados local. &lt;/li&gt;
&lt;li&gt;Um Relay Worker assíncrono lê essa tabela e realiza o despacho para o broker.&lt;/li&gt;
&lt;/ol&gt;

&lt;ul&gt;
&lt;li&gt;O pulo do gato: Circuit Breaker
Implementei um Circuit Breaker que monitora falhas de publicação. Se o Kafka cair, o circuito "abre", silenciando o worker temporariamente para preservar recursos e evitar logs de erro infinitos.&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  📈 Escalabilidade com &lt;code&gt;SKIP LOCKED&lt;/code&gt;
&lt;/h2&gt;

&lt;p&gt;Para escalar o worker sem duplicar envios, utilizei o &lt;code&gt;SELECT FOR UPDATE SKIP LOCKED&lt;/code&gt; do PostgreSQL:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight sql"&gt;&lt;code&gt;&lt;span class="k"&gt;SELECT&lt;/span&gt; &lt;span class="n"&gt;id&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="n"&gt;payload&lt;/span&gt; &lt;span class="k"&gt;FROM&lt;/span&gt; &lt;span class="n"&gt;outbox&lt;/span&gt;
&lt;span class="k"&gt;WHERE&lt;/span&gt; &lt;span class="n"&gt;status&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="s1"&gt;'PENDING'&lt;/span&gt;
&lt;span class="k"&gt;ORDER&lt;/span&gt; &lt;span class="k"&gt;BY&lt;/span&gt; &lt;span class="n"&gt;created_at&lt;/span&gt;
&lt;span class="k"&gt;FOR&lt;/span&gt; &lt;span class="k"&gt;UPDATE&lt;/span&gt; &lt;span class="n"&gt;SKIP&lt;/span&gt; &lt;span class="n"&gt;LOCKED&lt;/span&gt;
&lt;span class="k"&gt;LIMIT&lt;/span&gt; &lt;span class="mi"&gt;100&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Isso permite que múltiplas instâncias do worker rodem em paralelo, onde cada uma "pula" as linhas já bloqueadas por outra instância.&lt;/p&gt;




&lt;h2&gt;
  
  
  📊 Observabilidade: Onde o filho chora e a mãe não vê
&lt;/h2&gt;

&lt;p&gt;Utilizei a stack &lt;code&gt;Loki&lt;/code&gt; + &lt;code&gt;Prometheus&lt;/code&gt; + &lt;code&gt;Grafana&lt;/code&gt; para monitorar a saúde do pipeline:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Métricas: Comparação em tempo real entre o que o Go publica e o que o Kotlin consome.&lt;/li&gt;
&lt;li&gt;Alertas: Notificações de lag no Kafka e latência acima do esperado.&lt;/li&gt;
&lt;li&gt;Logs: Rastreamento centralizado para identificar falhas entre os containers através do Loki.&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  💣 Stress Test: Bombardeando o Sistema
&lt;/h2&gt;

&lt;p&gt;Para validar a resiliência, utilizei um script em Bash para saturar o endpoint de entrada e observar o comportamento do sistema sob carga:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;&lt;span class="c"&gt;# Exemplo do stress test utilizado&lt;/span&gt;
./bombardear.sh http://localhost:8080/prices 50
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Este teste foi vital para ajustar os limites do pool de conexões do Postgres e validar a recuperação automática do sistema sob pressão.&lt;/p&gt;




&lt;h2&gt;
  
  
  🔗 Código Fonte
&lt;/h2&gt;

&lt;p&gt;O projeto completo, incluindo as configurações do &lt;strong&gt;Docker Compose&lt;/strong&gt;, o script de &lt;strong&gt;stress test&lt;/strong&gt; e toda a infraestrutura de &lt;strong&gt;observabilidade&lt;/strong&gt;, está disponível no meu GitHub:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Repositório do Projeto Go:&lt;/strong&gt; &lt;a href="https://github.com/costa-lucas/price-crawler.git" rel="noopener noreferrer"&gt;https://github.com/costa-lucas/price-crawler.git&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Repositório do Projeto Kotlin:&lt;/strong&gt; &lt;a href="https://github.com/costa-lucas/price-consumer.git" rel="noopener noreferrer"&gt;https://github.com/costa-lucas/price-consumer.git&lt;/a&gt;
&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Se você está estudando &lt;strong&gt;System Design&lt;/strong&gt;, &lt;strong&gt;Go&lt;/strong&gt; ou &lt;strong&gt;Kotlin&lt;/strong&gt;, o código pode ajudar — especialmente a implementação do relay com &lt;code&gt;FOR UPDATE SKIP LOCKED&lt;/code&gt; e a separação rigorosa de módulos no Kotlin.&lt;/p&gt;




&lt;h2&gt;
  
  
  💡 O que aprendi nessa jornada
&lt;/h2&gt;

&lt;ol&gt;
&lt;li&gt; &lt;strong&gt;Resiliência por design:&lt;/strong&gt; Desacoplar serviços via eventos muda como você pensa em falhas. O sistema se torna tolerante por natureza.&lt;/li&gt;
&lt;li&gt; &lt;strong&gt;Garantia real:&lt;/strong&gt; O &lt;strong&gt;Transactional Outbox&lt;/strong&gt; é a diferença entre perder dados ou apenas atrasar o processamento em momentos de instabilidade.&lt;/li&gt;
&lt;li&gt; &lt;strong&gt;Métricas são vitais:&lt;/strong&gt; Sem monitoramento e alertas, você está voando às cegas. A stack Prometheus + Grafana + Loki é um divisor de águas.&lt;/li&gt;
&lt;/ol&gt;




&lt;p&gt;&lt;strong&gt;E você, como valida a carga e a resiliência dos seus sistemas em produção? Já utilizou o &lt;code&gt;SKIP LOCKED&lt;/code&gt; ou prefere outras estratégias de escalonamento de workers? Vamos conversar nos comentários!&lt;/strong&gt;&lt;/p&gt;

</description>
      <category>go</category>
      <category>kotlin</category>
      <category>architecture</category>
      <category>backend</category>
    </item>
  </channel>
</rss>
