@luanpdd/kit-mcp 1.20.0 → 1.22.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (51) hide show
  1. package/README.md +1 -1
  2. package/gates/dept-cycle-prevention.md +179 -0
  3. package/gates/multi-tenant-rls-coverage.md +102 -0
  4. package/gates/service-role-not-in-user-facing.md +113 -0
  5. package/kit/README.md +24 -0
  6. package/kit/agents/audit-log-implementer.md +175 -0
  7. package/kit/agents/auditor-consistencia-isolamento.md +380 -0
  8. package/kit/agents/b2b-saas-architect.md +156 -0
  9. package/kit/agents/crm-pipeline-implementer.md +167 -0
  10. package/kit/agents/detector-tenant-quente.md +337 -0
  11. package/kit/agents/evolution-go-integrator.md +179 -0
  12. package/kit/agents/invite-flow-implementer.md +137 -0
  13. package/kit/agents/lgpd-compliance-auditor.md +206 -0
  14. package/kit/agents/multi-tenant-isolation-auditor.md +253 -0
  15. package/kit/agents/multi-tenant-rls-writer.md +262 -0
  16. package/kit/agents/org-onboarding-implementer.md +202 -0
  17. package/kit/agents/supabase-architect.md +10 -0
  18. package/kit/agents/supabase-migration-writer.md +12 -0
  19. package/kit/agents/super-admin-implementer.md +182 -0
  20. package/kit/agents/validador-evolucao-schema.md +335 -0
  21. package/kit/commands/dados-distribuidos.md +188 -0
  22. package/kit/commands/multi-tenant.md +163 -0
  23. package/kit/file-manifest.json +48 -9
  24. package/kit/skills/_shared-dados-distribuidos/glossary.md +224 -0
  25. package/kit/skills/_shared-multi-tenant/glossary.md +186 -0
  26. package/kit/skills/armadilhas-sistemas-distribuidos/SKILL.md +447 -0
  27. package/kit/skills/audit-log-multi-tenant/SKILL.md +340 -0
  28. package/kit/skills/b2b-saas-architecture/SKILL.md +300 -0
  29. package/kit/skills/cascading-failures/SKILL.md +4 -0
  30. package/kit/skills/consistencia-leitura-replica/SKILL.md +385 -0
  31. package/kit/skills/crm-lead-pipeline-patterns/SKILL.md +343 -0
  32. package/kit/skills/escolha-modelo-consistencia/SKILL.md +495 -0
  33. package/kit/skills/evolucao-schema-compativel/SKILL.md +448 -0
  34. package/kit/skills/evolution-go-whatsapp-integration/SKILL.md +322 -0
  35. package/kit/skills/lgpd-multi-tenant-compliance/SKILL.md +340 -0
  36. package/kit/skills/member-invite-flow/SKILL.md +305 -0
  37. package/kit/skills/member-management-react-shadcn/SKILL.md +328 -0
  38. package/kit/skills/multi-tenant-performance-scaling/SKILL.md +316 -0
  39. package/kit/skills/multi-tenant-rls-hierarchy/SKILL.md +342 -0
  40. package/kit/skills/org-onboarding-flow/SKILL.md +257 -0
  41. package/kit/skills/org-switcher-react-pattern/SKILL.md +349 -0
  42. package/kit/skills/permission-gate-react-pattern/SKILL.md +271 -0
  43. package/kit/skills/postgres-isolamento-concorrencia/SKILL.md +552 -0
  44. package/kit/skills/rbac-permissions-matrix-supabase/SKILL.md +301 -0
  45. package/kit/skills/streams-eventos-cdc/SKILL.md +712 -0
  46. package/kit/skills/supabase-cron-queues/SKILL.md +9 -0
  47. package/kit/skills/supabase-migrations/SKILL.md +10 -0
  48. package/kit/skills/super-admin-platform-pattern/SKILL.md +326 -0
  49. package/kit/skills/tenant-quente-mitigacao/SKILL.md +605 -0
  50. package/kit/skills/whatsapp-conversation-state-machine/SKILL.md +287 -0
  51. package/package.json +1 -1
@@ -0,0 +1,385 @@
1
+ ---
2
+ name: consistencia-leitura-replica
3
+ description: Use ao usar Supabase read replicas via Supavisor (porta 6543) ou ao combinar Realtime broadcast + leitura DB — 3 problemas canônicos DDIA Ch 5 (read-after-write inconsistente, leituras não-monotônicas, prefixo causal violado), 3 soluções para Supabase (leitura no líder após escrita, sticky session por user_id, detecção stale via pg_last_wal_replay_lsn), padrão "ler o próprio broadcast" para evitar re-fetch após broadcast.
4
+ ---
5
+
6
+ # Consistência Leitura Réplica — Supabase + Supavisor + Realtime
7
+
8
+ ## Quando usar
9
+
10
+ LLM carrega esta skill ao usar Supabase Pro+ com **read replicas** ou ao combinar Realtime broadcast com leitura subsequente do DB. Trigger phrases:
11
+
12
+ - "Supabase read replica", "réplica de leitura"
13
+ - "porta 6543 vs 5432", "Supavisor session vs transaction"
14
+ - "read-after-write", "leitura após escrita inconsistente"
15
+ - "monotonic reads", "leituras não-monotônicas"
16
+ - "consistent prefix reads", "prefixo causal violado"
17
+ - "replication lag Supabase", "atraso de replicação"
18
+ - "broadcast Realtime + SELECT stale"
19
+ - "pg_last_wal_replay_lsn", "WAL position detection"
20
+
21
+ Esta skill aplica **DDIA Ch 5 "Problems With Replication Lag"** ao stack Supabase. Cross-referenciada por `supabase-realtime` (v1.8) ao bundlear broadcast + leitura, e por `multi-tenant-performance-scaling` (v1.21) ao escalar Postgres em Pro+.
22
+
23
+ Termos canônicos (`read-after-write consistency`, `monotonic reads`, `consistent prefix reads`, `replication lag`, `leader-follower replication`) definidos em [`_shared-dados-distribuidos/glossary.md`](../_shared-dados-distribuidos/glossary.md) seções (a) e (b) — esta skill **não duplica**, apenas linka.
24
+
25
+ ## Regras absolutas
26
+
27
+ **REGRA #1 (read-after-write own data):** Para leituras do **próprio dado do usuário** dentro de janela de **5s após write**, rotear para **porta 5432 (líder)** — não 6543. Sem isso, usuário cria post → GET retorna 404 → percepção de bug. Justify: DDIA p. 156 "always read the user's own profile from the leader".
28
+
29
+ **REGRA #2 (sticky session monotonic):** Para usuários ativos lendo de réplicas, escolher réplica via `hash(user_id) mod N` — **não round-robin**. Round-robin viola monotonic reads (DDIA p. 158). Mitigação obrigatória: fallback para líder se réplica down.
30
+
31
+ **REGRA #3 (broadcast trust payload):** Após receber Realtime broadcast com `payload.record`, **NÃO** fazer SELECT subsequente. Confiar no payload — server é a fonte canônica do evento. SELECT pode atingir réplica que ainda não replicou (lag típico 50-500ms).
32
+
33
+ **REGRA #4 (causal partition):** Writes causalmente relacionados (pergunta + resposta em chat, parent + child em árvore) **DEVEM** ir para a mesma partição lógica. Em Supabase: usar mesmo `org_id` ou `conversation_id` como partition key. DDIA p. 159 "any writes which are causally related to each other are written to the same partition".
34
+
35
+ **REGRA #5 (LSN wait com timeout):** Quando usar `pg_last_wal_replay_lsn() >= captured_lsn`, **SEMPRE** com timeout (3-5s). Sem timeout, query trava se réplica falhou. Após timeout, fallback para líder.
36
+
37
+ ## Patterns canônicos
38
+
39
+ ### Problema 1: Read-after-write inconsistente (DDIA Ch 5, p. 156)
40
+
41
+ **Cenário canônico:** usuário cria post via form submit → tela mostra "criado com sucesso" → usuário clica "ver post" → request vai para réplica → réplica ainda não replicou → 404. Da perspectiva do usuário: "perdi meu dado".
42
+
43
+ ```
44
+ User 1234 ──INSERT─→ Leader (5432) ┐
45
+ │ │ replication lag (50-500ms)
46
+ └──WAL──────────────→ Follower (6543, replica)
47
+
48
+ User 1234 ─SELECT────────────────────────────→ ❌ 404
49
+ (vai para replica via Supavisor)
50
+ ```
51
+
52
+ **Solução A — leitura no líder após write do mesmo usuário:**
53
+
54
+ ```typescript
55
+ // PT-BR: client mantém timestamp do último write em memória
56
+ class SupabaseRouter {
57
+ private lastWriteAt: Map<string, number> = new Map() // userId → timestamp ms
58
+ private readonly STICKY_WINDOW_MS = 5000 // 5s leitura no líder
59
+
60
+ async write(userId: string, table: string, payload: unknown) {
61
+ const result = await this.leaderClient.from(table).insert(payload)
62
+ this.lastWriteAt.set(userId, Date.now())
63
+ return result
64
+ }
65
+
66
+ async read(userId: string, table: string, filter: object) {
67
+ const lastWrite = this.lastWriteAt.get(userId) ?? 0
68
+ const elapsedMs = Date.now() - lastWrite
69
+
70
+ // PT-BR: dentro da janela 5s, ler do líder (porta 5432)
71
+ if (elapsedMs < this.STICKY_WINDOW_MS) {
72
+ return this.leaderClient.from(table).select().match(filter)
73
+ }
74
+ // PT-BR: fora da janela, OK ler de replica via pooler 6543
75
+ return this.replicaClient.from(table).select().match(filter)
76
+ }
77
+ }
78
+ ```
79
+
80
+ **Trade-off:** dentro da janela, perde benefício do read scaling. DDIA recomenda janela curta (1-5s) — cobre 99% dos casos UX sem sobrecarregar líder.
81
+
82
+ ### Problema 2: Leituras não-monotônicas (DDIA Ch 5, p. 158)
83
+
84
+ **Cenário canônico:** usuário 2345 abre lista de comentários — primeira leitura vai para réplica 1 (lag 100ms) → vê comentário X. Segundo refresh vai para réplica 2 (lag 800ms) → comentário X **desapareceu**. Da perspectiva do usuário: "dado voltou no tempo".
85
+
86
+ ```
87
+ User 2345 ─SELECT(1)────→ Replica 1 (lag 100ms) → 1 result ✅
88
+ User 2345 ─SELECT(2)────→ Replica 2 (lag 800ms) → 0 results ❌ (parecia ter sumido)
89
+ ```
90
+
91
+ **Solução B — sticky session por `user_id` em routing:**
92
+
93
+ ```typescript
94
+ // PT-BR: hash determinístico do user_id → escolhe replica fixa para esse user
95
+ import { createHash } from 'node:crypto'
96
+
97
+ function pickReplica(userId: string, replicas: ReadonlyArray<SupabaseClient>): SupabaseClient {
98
+ const hash = createHash('sha256').update(userId).digest()
99
+ const idx = hash.readUInt32BE(0) % replicas.length
100
+ return replicas[idx]
101
+ }
102
+
103
+ // PT-BR: usage com fallback para líder se replica falhar
104
+ async function readWithStickyReplica(userId: string, query: () => Promise<Result>) {
105
+ const replica = pickReplica(userId, REPLICAS)
106
+ try {
107
+ return await query.call(replica)
108
+ } catch (err) {
109
+ if (isReplicaDownError(err)) {
110
+ // PT-BR: fallback obrigatório — REGRA #2
111
+ return await query.call(LEADER)
112
+ }
113
+ throw err
114
+ }
115
+ }
116
+ ```
117
+
118
+ **Pitfall:** réplica fica down → todos os usuários alocados a ela ficam sem leitura. Mitigação: detectar via health check + reroute para líder até réplica voltar.
119
+
120
+ ### Problema 3: Prefixo causal violado (DDIA Ch 5, p. 159)
121
+
122
+ **Cenário canônico chat:** Mr Poons pergunta "How far into the future can you see, Mrs Cake?" → write vai para partição A. Mrs Cake responde "About ten seconds usually, Mr Poons." → write vai para partição B. Observador lê de B (lag baixo) e A (lag alto): **vê resposta antes da pergunta**.
123
+
124
+ ```
125
+ Partição A (lag 800ms): "How far into the future..."
126
+ Partição B (lag 100ms): "About ten seconds..."
127
+
128
+ Observer lê B → vê resposta ✅
129
+ Observer lê A → vê pergunta ✅
130
+ Observer ordem percebida: resposta → pergunta ❌
131
+ ```
132
+
133
+ **Solução parcial — particionamento por chave causal:**
134
+
135
+ ```sql
136
+ -- PT-BR: ambas msgs (pergunta + resposta) particionadas por conversation_id
137
+ -- garante mesma partição lógica = mesma ordem WAL
138
+ create table public.messages (
139
+ id uuid primary key default gen_random_uuid(),
140
+ conversation_id uuid not null,
141
+ author_id uuid not null,
142
+ body text not null,
143
+ created_at timestamptz not null default now()
144
+ ) partition by hash (conversation_id);
145
+
146
+ -- PT-BR: indexes ajudam ordering
147
+ create index messages_conv_created_idx
148
+ on public.messages (conversation_id, created_at);
149
+ ```
150
+
151
+ **Limitação:** garante consistent prefix dentro de uma partição. Cross-partition (ex: usuário em duas conversações simultâneas), DDIA p. 159 conclui "in general, ensuring consistent prefix reads requires snapshot isolation". Em Supabase prático: **manter conversação em uma tabela com chave causal explícita**.
152
+
153
+ ### Solução C — Detecção stale via `pg_last_wal_replay_lsn()`
154
+
155
+ Quando precisa de garantia "este read viu meu write" sem rotear ao líder:
156
+
157
+ ```sql
158
+ -- PT-BR: capturar LSN no líder após write (chamada do app via RPC)
159
+ create or replace function public.get_current_lsn()
160
+ returns text
161
+ language sql
162
+ security invoker
163
+ set search_path = ''
164
+ as $$
165
+ -- PT-BR: pg_current_wal_lsn() retorna posição atual do WAL no líder
166
+ select pg_current_wal_lsn()::text;
167
+ $$;
168
+
169
+ -- PT-BR: na replica, esperar até replay alcançar o LSN capturado
170
+ create or replace function public.wait_for_lsn(target_lsn text, timeout_ms int default 5000)
171
+ returns boolean
172
+ language plpgsql
173
+ security invoker
174
+ set search_path = ''
175
+ as $$
176
+ declare
177
+ start_at timestamptz := clock_timestamp();
178
+ elapsed_ms int;
179
+ begin
180
+ loop
181
+ -- PT-BR: pg_last_wal_replay_lsn() na replica = última posição replayed
182
+ if pg_last_wal_replay_lsn() >= target_lsn::pg_lsn then
183
+ return true;
184
+ end if;
185
+
186
+ elapsed_ms := extract(milliseconds from (clock_timestamp() - start_at))::int;
187
+ if elapsed_ms >= timeout_ms then
188
+ return false; -- PT-BR: REGRA #5 — timeout sem bloquear infinito
189
+ end if;
190
+
191
+ perform pg_sleep(0.05); -- PT-BR: 50ms entre polls
192
+ end loop;
193
+ end;
194
+ $$;
195
+ ```
196
+
197
+ **Uso típico no client:**
198
+
199
+ ```typescript
200
+ // PT-BR: 1) write no líder, captura LSN
201
+ const { data: lsn } = await leaderClient.rpc('get_current_lsn')
202
+ await leaderClient.from('orders').insert(order)
203
+
204
+ // PT-BR: 2) read no líder, espera replay
205
+ const { data: ready } = await replicaClient.rpc('wait_for_lsn', {
206
+ target_lsn: lsn,
207
+ timeout_ms: 3000,
208
+ })
209
+
210
+ if (ready) {
211
+ // PT-BR: replica caught up, leitura é safe
212
+ return replicaClient.from('orders').select().eq('id', order.id)
213
+ }
214
+ // PT-BR: timeout — fallback para líder (REGRA #5)
215
+ return leaderClient.from('orders').select().eq('id', order.id)
216
+ ```
217
+
218
+ ### Supavisor read replica routing
219
+
220
+ | Porta | Modo | Use case | Connection string |
221
+ |---|---|---|---|
222
+ | **6543** | Transaction (default Pro+) | Apps com pooler já configurado, edge runtimes, serverless | `postgresql://postgres.[ref]:pwd@aws-0-region.pooler.supabase.com:6543/postgres` |
223
+ | **5432** | Session (líder) | Reads críticas (read-after-write), writes, prepared statements, advisory locks | `postgresql://postgres.[ref]:pwd@aws-0-region.pooler.supabase.com:5432/postgres` |
224
+ | `pooler.read.*` | Réplica routing | Read-heavy workloads em Pro+ com replicas habilitadas | (futuro Supabase feature — placeholder hoje) |
225
+
226
+ **Decisão por tipo de query:**
227
+
228
+ ```
229
+ SELECT do próprio dado dentro 5s do write? → 5432 (líder, REGRA #1)
230
+ SELECT cross-user, sem janela sticky? → 6543 (replica via Supavisor)
231
+ INSERT / UPDATE / DELETE? → 5432 (sempre líder)
232
+ SELECT FOR UPDATE / advisory lock? → 5432 (transaction precisa session mode)
233
+ ```
234
+
235
+ Cross-ref ATIVO: [`multi-tenant-performance-scaling/SKILL.md`](../multi-tenant-performance-scaling/SKILL.md) (v1.21) cobre Supavisor REGRA #1 sob lente de connection pooling — esta skill cobre a mesma porta sob lente de consistência.
236
+
237
+ ### Realtime broadcast + leitura DB — padrão "ler o próprio broadcast"
238
+
239
+ **Cenário canônico:** client A faz INSERT em `orders` → server emite Realtime broadcast `new_order` no canal `org:orders:org_42` → client B recebe broadcast → client B faz SELECT para refresh da lista. **Pode receber dado stale** (replica não replicou ainda).
240
+
241
+ **Sequência do bug:**
242
+
243
+ ```
244
+ t=0ms Client A INSERT → Leader
245
+ t=10ms Server emite broadcast → todos clients no canal recebem
246
+ t=15ms Client B recebe broadcast → triggers re-fetch
247
+ t=20ms Client B SELECT → Replica (lag ainda 80ms)
248
+ t=20ms Replica retorna lista SEM o novo order ❌
249
+ t=80ms Replica finalmente replicou (mas client B já desenhou stale)
250
+ ```
251
+
252
+ **Padrão correto — confiar no payload broadcast:**
253
+
254
+ ```typescript
255
+ // PT-BR: client confia no payload, NÃO faz SELECT subsequente
256
+ import { createClient } from '@supabase/supabase-js'
257
+
258
+ const supabase = createClient(SUPABASE_URL, SUPABASE_ANON_KEY)
259
+
260
+ const channel = supabase
261
+ .channel('org:orders:org_42', { config: { private: true } })
262
+ .on('broadcast', { event: 'new_order' }, ({ payload }) => {
263
+ // PT-BR: REGRA #3 — confie no payload, NÃO faça SELECT
264
+ setOrders((prev) => [...prev, payload.record])
265
+ })
266
+ .subscribe()
267
+
268
+ // PT-BR: cleanup obrigatório — pattern de supabase-realtime v1.8
269
+ return () => {
270
+ supabase.removeChannel(channel)
271
+ }
272
+ ```
273
+
274
+ **Server side — emitir payload completo no broadcast:**
275
+
276
+ ```typescript
277
+ // PT-BR: Edge Function que cria order e broadcast com record completo
278
+ Deno.serve(async (req) => {
279
+ const supabase = createClient(SUPABASE_URL, SERVICE_ROLE_KEY)
280
+ const order = await req.json()
281
+
282
+ // PT-BR: 1) INSERT no líder
283
+ const { data: created } = await supabase
284
+ .from('orders')
285
+ .insert(order)
286
+ .select()
287
+ .single()
288
+
289
+ // PT-BR: 2) broadcast com record completo — clients confiam neste payload
290
+ await supabase
291
+ .channel(`org:orders:${order.org_id}`)
292
+ .send({
293
+ type: 'broadcast',
294
+ event: 'new_order',
295
+ payload: { record: created }, // PT-BR: payload canônico
296
+ })
297
+
298
+ return new Response(JSON.stringify(created), { status: 201 })
299
+ })
300
+ ```
301
+
302
+ **Cross-ref ATIVO:** [`supabase-realtime/SKILL.md`](../supabase-realtime/SKILL.md) (v1.8) define padrão de canal (`scope:entity:id`, `private:true`, `removeChannel` cleanup). Esta skill estende com o padrão `payload.record` específico para evitar replica lag bug.
303
+
304
+ ## Anti-patterns
305
+
306
+ ### Anti-pattern 1: Round-robin entre réplicas para o mesmo usuário
307
+
308
+ **Errado:**
309
+ ```typescript
310
+ // PT-BR: pegar replica aleatoriamente cada read
311
+ const replica = REPLICAS[Math.floor(Math.random() * REPLICAS.length)]
312
+ return replica.from('messages').select()
313
+ ```
314
+
315
+ **Por quê:** viola monotonic reads (DDIA p. 158). User vê mensagem X em uma leitura, depois X some na próxima (replica 2 ainda não replicou). Leituras "voltam no tempo".
316
+
317
+ **Certo:** sticky session por `hash(user_id) mod N` (Solução B acima).
318
+
319
+ ### Anti-pattern 2: Re-fetch após broadcast
320
+
321
+ **Errado:**
322
+ ```typescript
323
+ .on('broadcast', { event: 'new_order' }, async () => {
324
+ // PT-BR: re-fetch que pode atingir replica stale
325
+ const { data } = await supabase.from('orders').select() // ❌
326
+ setOrders(data)
327
+ })
328
+ ```
329
+
330
+ **Por quê:** broadcast chega em 10-15ms, mas replication lag tipicamente 50-500ms. SELECT no callback **garantido** vai chegar antes da replica replicar. Bug "intermittent missing data".
331
+
332
+ **Certo:** confiar no `payload.record` enviado pelo server (REGRA #3 + Solução padrão "ler o próprio broadcast").
333
+
334
+ ### Anti-pattern 3: `pg_last_wal_replay_lsn()` sem timeout
335
+
336
+ **Errado:**
337
+ ```sql
338
+ -- PT-BR: loop infinito se replica falhou
339
+ do $$
340
+ begin
341
+ loop
342
+ if pg_last_wal_replay_lsn() >= captured_lsn::pg_lsn then exit; end if;
343
+ perform pg_sleep(0.05);
344
+ end loop;
345
+ end$$;
346
+ ```
347
+
348
+ **Por quê:** se replica desconectou do WAL stream (network partition, disk full), `pg_last_wal_replay_lsn()` nunca alcança o LSN do líder. Query trava indefinidamente, esgota connection pool.
349
+
350
+ **Certo:** timeout 3-5s + fallback explícito para líder (REGRA #5 + função `wait_for_lsn` acima).
351
+
352
+ ### Anti-pattern 4: Cross-partition para conversação causal
353
+
354
+ **Errado:**
355
+ ```sql
356
+ -- PT-BR: messages particionadas por created_at (range temporal)
357
+ create table public.messages (...) partition by range (created_at);
358
+ ```
359
+
360
+ **Por quê:** pergunta e resposta em uma conversação podem cair em partições diferentes (se virada de mês entre as duas). Viola consistent prefix reads — observador vê resposta antes da pergunta.
361
+
362
+ **Certo:** particionar por `conversation_id` (HASH), garante que toda a conversação fica na mesma partição = mesma ordem WAL = consistent prefix.
363
+
364
+ ### Anti-pattern 5: Porta 6543 para `SELECT FOR UPDATE`
365
+
366
+ **Errado:**
367
+ ```typescript
368
+ // PT-BR: tentando lock pessimista via Supavisor transaction mode
369
+ const { data } = await client6543.rpc('lock_order', { id })
370
+ ```
371
+
372
+ **Por quê:** Supavisor 6543 (transaction mode) não preserva sessão entre statements — `SELECT FOR UPDATE` libera o lock na próxima query. Lock vira no-op silencioso.
373
+
374
+ **Certo:** porta 5432 (session mode) para qualquer operação que precisa estado de sessão (locks, prepared statements, `SET LOCAL`, advisory locks).
375
+
376
+ ## Ver também
377
+
378
+ - [`_shared-dados-distribuidos/glossary.md`](../_shared-dados-distribuidos/glossary.md) — termos canônicos `read-after-write consistency`, `monotonic reads`, `consistent prefix reads`, `replication lag`, `leader-follower replication` (Phase 117)
379
+ - [`supabase-realtime/SKILL.md`](../supabase-realtime/SKILL.md) — broadcast com `private:true`, naming `scope:entity:id`, cleanup `removeChannel` (v1.8)
380
+ - [`multi-tenant-performance-scaling/SKILL.md`](../multi-tenant-performance-scaling/SKILL.md) — Supavisor connection string canônica, REGRA #1 porta 6543 (v1.21)
381
+ - [`supabase-database-functions/SKILL.md`](../supabase-database-functions/SKILL.md) — padrões PG functions (security invoker, search_path) usados em `get_current_lsn` e `wait_for_lsn`
382
+ - [Designing Data-Intensive Applications, Martin Kleppmann (O'Reilly 2017)](https://www.oreilly.com/library/view/designing-data-intensive-applications/9781491903063/) — Ch 5 "Problems With Replication Lag" (p. 155-160)
383
+ - [PostgreSQL Documentation — pg_last_wal_replay_lsn](https://www.postgresql.org/docs/current/functions-admin.html#FUNCTIONS-RECOVERY-INFO)
384
+ - [Supabase Read Replicas](https://supabase.com/docs/guides/platform/read-replicas)
385
+ - [Supabase Supavisor 1M Connections](https://supabase.com/blog/supavisor-1-million)