@semacode/cli 1.5.14 → 1.5.16

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 (177) hide show
  1. package/README.md +2 -1
  2. package/dist/index.js +76 -43
  3. package/package.json +26 -35
  4. package/semacode-cli-1.3.1.tgz +0 -0
  5. package/src/angular-consumer-standalone.ts +312 -0
  6. package/src/cpp-symbols.ts +82 -0
  7. package/src/docs.ts +535 -0
  8. package/src/dotnet-http.ts +355 -0
  9. package/src/drift.ts +4933 -0
  10. package/src/go-http.ts +118 -0
  11. package/src/importador.ts +3891 -0
  12. package/src/index.ts +5641 -0
  13. package/src/java-http.ts +247 -0
  14. package/src/lua-symbols.ts +114 -0
  15. package/src/php-symbols.ts +462 -0
  16. package/src/projeto.ts +862 -0
  17. package/src/python-http.ts +258 -0
  18. package/src/rust-http.ts +125 -0
  19. package/src/tipos.ts +24 -0
  20. package/src/typescript-http.ts +1076 -0
  21. package/tsconfig.json +20 -0
  22. package/AGENTS.md +0 -272
  23. package/LICENSE +0 -22
  24. package/SEMA_BRIEF.curto.txt +0 -9
  25. package/SEMA_BRIEF.md +0 -63
  26. package/SEMA_BRIEF.micro.txt +0 -7
  27. package/SEMA_INDEX.json +0 -799
  28. package/dist/angular-consumer-standalone.d.ts +0 -6
  29. package/dist/angular-consumer-standalone.js.map +0 -1
  30. package/dist/cpp-symbols.d.ts +0 -10
  31. package/dist/cpp-symbols.js.map +0 -1
  32. package/dist/docs.d.ts +0 -56
  33. package/dist/docs.js.map +0 -1
  34. package/dist/dotnet-http.d.ts +0 -23
  35. package/dist/dotnet-http.js.map +0 -1
  36. package/dist/drift.d.ts +0 -225
  37. package/dist/drift.js.map +0 -1
  38. package/dist/go-http.d.ts +0 -23
  39. package/dist/go-http.js.map +0 -1
  40. package/dist/importador.d.ts +0 -31
  41. package/dist/importador.js.map +0 -1
  42. package/dist/index.d.ts +0 -2
  43. package/dist/index.js.map +0 -1
  44. package/dist/java-http.d.ts +0 -23
  45. package/dist/java-http.js.map +0 -1
  46. package/dist/lua-symbols.d.ts +0 -10
  47. package/dist/lua-symbols.js.map +0 -1
  48. package/dist/php-symbols.d.ts +0 -24
  49. package/dist/php-symbols.js.map +0 -1
  50. package/dist/projeto.d.ts +0 -53
  51. package/dist/projeto.js.map +0 -1
  52. package/dist/python-http.d.ts +0 -23
  53. package/dist/python-http.js.map +0 -1
  54. package/dist/rust-http.d.ts +0 -23
  55. package/dist/rust-http.js.map +0 -1
  56. package/dist/tipos.d.ts +0 -3
  57. package/dist/tipos.js.map +0 -1
  58. package/dist/typescript-http.d.ts +0 -35
  59. package/dist/typescript-http.js.map +0 -1
  60. package/docs/AGENT_STARTER.md +0 -102
  61. package/docs/cli.md +0 -117
  62. package/docs/como-ensinar-a-sema-para-ia.md +0 -149
  63. package/docs/deploy.md +0 -70
  64. package/docs/documentacao.md +0 -63
  65. package/docs/env.md +0 -56
  66. package/docs/extensao-vscode.md +0 -45
  67. package/docs/fluxo-pratico-ia-sema.md +0 -177
  68. package/docs/instalacao-e-primeiro-uso.md +0 -112
  69. package/docs/integracao-com-ia.md +0 -101
  70. package/docs/mcp.md +0 -53
  71. package/docs/pagamento-ponta-a-ponta.md +0 -155
  72. package/docs/persistencia-vendor-first.md +0 -145
  73. package/docs/prompt-base-ia-sema.md +0 -104
  74. package/docs/rollback.md +0 -47
  75. package/docs/sintaxe.md +0 -410
  76. package/exemplos/agendamento.sema +0 -106
  77. package/exemplos/assinatura.sema +0 -136
  78. package/exemplos/auditoria.sema +0 -88
  79. package/exemplos/autenticacao.sema +0 -125
  80. package/exemplos/automacao.sema +0 -107
  81. package/exemplos/cadastro_usuario.sema +0 -54
  82. package/exemplos/calculadora.sema +0 -78
  83. package/exemplos/crud_simples.sema +0 -89
  84. package/exemplos/estoque.sema +0 -126
  85. package/exemplos/exportacao.sema +0 -94
  86. package/exemplos/fila.sema +0 -131
  87. package/exemplos/integracao_externa.sema +0 -94
  88. package/exemplos/multi_tenant.sema +0 -140
  89. package/exemplos/notificacao.sema +0 -98
  90. package/exemplos/operacao_estrategia.sema +0 -402
  91. package/exemplos/pagamento.sema +0 -222
  92. package/exemplos/pagamento_dominio.sema +0 -35
  93. package/exemplos/pedido.sema +0 -119
  94. package/exemplos/permissao.sema +0 -121
  95. package/exemplos/persistencia_vendor_first.sema +0 -86
  96. package/exemplos/relatorio.sema +0 -93
  97. package/exemplos/testes_embutidos.sema +0 -45
  98. package/exemplos/tratamento_erro.sema +0 -157
  99. package/exemplos/upload_arquivo.sema +0 -93
  100. package/exemplos/webhook.sema +0 -96
  101. package/llms-full.txt +0 -34
  102. package/llms.txt +0 -17
  103. package/node_modules/@sema/gerador-css/dist/index.d.ts +0 -3
  104. package/node_modules/@sema/gerador-css/dist/index.js +0 -592
  105. package/node_modules/@sema/gerador-css/dist/index.js.map +0 -1
  106. package/node_modules/@sema/gerador-css/package.json +0 -7
  107. package/node_modules/@sema/gerador-dart/dist/index.d.ts +0 -3
  108. package/node_modules/@sema/gerador-dart/dist/index.js +0 -44
  109. package/node_modules/@sema/gerador-dart/dist/index.js.map +0 -1
  110. package/node_modules/@sema/gerador-dart/package.json +0 -7
  111. package/node_modules/@sema/gerador-html/dist/index.d.ts +0 -3
  112. package/node_modules/@sema/gerador-html/dist/index.js +0 -163
  113. package/node_modules/@sema/gerador-html/dist/index.js.map +0 -1
  114. package/node_modules/@sema/gerador-html/package.json +0 -7
  115. package/node_modules/@sema/gerador-javascript/dist/index.d.ts +0 -3
  116. package/node_modules/@sema/gerador-javascript/dist/index.js +0 -421
  117. package/node_modules/@sema/gerador-javascript/dist/index.js.map +0 -1
  118. package/node_modules/@sema/gerador-javascript/package.json +0 -7
  119. package/node_modules/@sema/gerador-lua/dist/index.d.ts +0 -3
  120. package/node_modules/@sema/gerador-lua/dist/index.js +0 -328
  121. package/node_modules/@sema/gerador-lua/dist/index.js.map +0 -1
  122. package/node_modules/@sema/gerador-lua/package.json +0 -7
  123. package/node_modules/@sema/gerador-python/dist/index.d.ts +0 -6
  124. package/node_modules/@sema/gerador-python/dist/index.js +0 -729
  125. package/node_modules/@sema/gerador-python/dist/index.js.map +0 -1
  126. package/node_modules/@sema/gerador-python/package.json +0 -7
  127. package/node_modules/@sema/gerador-typescript/dist/index.d.ts +0 -6
  128. package/node_modules/@sema/gerador-typescript/dist/index.js +0 -793
  129. package/node_modules/@sema/gerador-typescript/dist/index.js.map +0 -1
  130. package/node_modules/@sema/gerador-typescript/package.json +0 -7
  131. package/node_modules/@sema/nucleo/dist/ast/tipos.d.ts +0 -125
  132. package/node_modules/@sema/nucleo/dist/ast/tipos.js +0 -2
  133. package/node_modules/@sema/nucleo/dist/ast/tipos.js.map +0 -1
  134. package/node_modules/@sema/nucleo/dist/diagnosticos/index.d.ts +0 -21
  135. package/node_modules/@sema/nucleo/dist/diagnosticos/index.js +0 -12
  136. package/node_modules/@sema/nucleo/dist/diagnosticos/index.js.map +0 -1
  137. package/node_modules/@sema/nucleo/dist/formatador/index.d.ts +0 -9
  138. package/node_modules/@sema/nucleo/dist/formatador/index.js +0 -487
  139. package/node_modules/@sema/nucleo/dist/formatador/index.js.map +0 -1
  140. package/node_modules/@sema/nucleo/dist/index.d.ts +0 -35
  141. package/node_modules/@sema/nucleo/dist/index.js +0 -96
  142. package/node_modules/@sema/nucleo/dist/index.js.map +0 -1
  143. package/node_modules/@sema/nucleo/dist/ir/conversor.d.ts +0 -5
  144. package/node_modules/@sema/nucleo/dist/ir/conversor.js +0 -1058
  145. package/node_modules/@sema/nucleo/dist/ir/conversor.js.map +0 -1
  146. package/node_modules/@sema/nucleo/dist/ir/modelos.d.ts +0 -377
  147. package/node_modules/@sema/nucleo/dist/ir/modelos.js +0 -2
  148. package/node_modules/@sema/nucleo/dist/ir/modelos.js.map +0 -1
  149. package/node_modules/@sema/nucleo/dist/lexer/lexer.d.ts +0 -7
  150. package/node_modules/@sema/nucleo/dist/lexer/lexer.js +0 -122
  151. package/node_modules/@sema/nucleo/dist/lexer/lexer.js.map +0 -1
  152. package/node_modules/@sema/nucleo/dist/lexer/tokens.d.ts +0 -8
  153. package/node_modules/@sema/nucleo/dist/lexer/tokens.js +0 -82
  154. package/node_modules/@sema/nucleo/dist/lexer/tokens.js.map +0 -1
  155. package/node_modules/@sema/nucleo/dist/parser/parser.d.ts +0 -9
  156. package/node_modules/@sema/nucleo/dist/parser/parser.js +0 -807
  157. package/node_modules/@sema/nucleo/dist/parser/parser.js.map +0 -1
  158. package/node_modules/@sema/nucleo/dist/persistencia/contratos.d.ts +0 -39
  159. package/node_modules/@sema/nucleo/dist/persistencia/contratos.js +0 -294
  160. package/node_modules/@sema/nucleo/dist/persistencia/contratos.js.map +0 -1
  161. package/node_modules/@sema/nucleo/dist/semantico/analisador.d.ts +0 -58
  162. package/node_modules/@sema/nucleo/dist/semantico/analisador.js +0 -1912
  163. package/node_modules/@sema/nucleo/dist/semantico/analisador.js.map +0 -1
  164. package/node_modules/@sema/nucleo/dist/semantico/estruturas.d.ts +0 -104
  165. package/node_modules/@sema/nucleo/dist/semantico/estruturas.js +0 -445
  166. package/node_modules/@sema/nucleo/dist/semantico/estruturas.js.map +0 -1
  167. package/node_modules/@sema/nucleo/dist/semantico/seguranca.d.ts +0 -91
  168. package/node_modules/@sema/nucleo/dist/semantico/seguranca.js +0 -258
  169. package/node_modules/@sema/nucleo/dist/semantico/seguranca.js.map +0 -1
  170. package/node_modules/@sema/nucleo/dist/util/arquivos.d.ts +0 -2
  171. package/node_modules/@sema/nucleo/dist/util/arquivos.js +0 -25
  172. package/node_modules/@sema/nucleo/dist/util/arquivos.js.map +0 -1
  173. package/node_modules/@sema/nucleo/package.json +0 -7
  174. package/node_modules/@sema/padroes/dist/index.d.ts +0 -25
  175. package/node_modules/@sema/padroes/dist/index.js +0 -316
  176. package/node_modules/@sema/padroes/dist/index.js.map +0 -1
  177. package/node_modules/@sema/padroes/package.json +0 -7
package/src/docs.ts ADDED
@@ -0,0 +1,535 @@
1
+ import { mkdir, readFile, readdir, stat, writeFile } from "node:fs/promises";
2
+ import path from "node:path";
3
+
4
+ const LIMITE_CONTEUDO_DOC = 16_000;
5
+
6
+ type TipoDocumentoMudanca = "raiz" | "documentacao" | "runbook" | "pacote" | "contrato";
7
+
8
+ interface DocumentoPlanejado {
9
+ relativo: string;
10
+ tipo: TipoDocumentoMudanca;
11
+ motivo: string;
12
+ permitirCriacao: boolean;
13
+ }
14
+
15
+ export interface DocumentoObrigatorioMudanca {
16
+ caminho: string;
17
+ relativo: string;
18
+ tipo: TipoDocumentoMudanca;
19
+ motivo: string;
20
+ existe: boolean;
21
+ criado: boolean;
22
+ obrigatorio: boolean;
23
+ conteudo?: string;
24
+ truncado?: boolean;
25
+ template?: string;
26
+ }
27
+
28
+ export interface BloqueioDocumentacaoMudanca {
29
+ tipo: "documentacao_ausente" | "leitura_obrigatoria_nao_comprovada";
30
+ severidade: 4 | 5;
31
+ caminho: string;
32
+ mensagem: string;
33
+ }
34
+
35
+ export interface ResultadoDocumentacaoObrigatoria {
36
+ sucesso: boolean;
37
+ baseProjeto: string;
38
+ intencao: string;
39
+ categorias: string[];
40
+ arquivosAlvo: string[];
41
+ leituraObrigatoria: DocumentoObrigatorioMudanca[];
42
+ docsAusentes: DocumentoObrigatorioMudanca[];
43
+ docsCriadas: DocumentoObrigatorioMudanca[];
44
+ bloqueios: BloqueioDocumentacaoMudanca[];
45
+ instrucoes: string[];
46
+ }
47
+
48
+ export interface ResultadoVerificacaoDocumentacaoMudanca {
49
+ sucesso: boolean;
50
+ baseProjeto: string;
51
+ intencao: string;
52
+ categorias: string[];
53
+ docsLidas: string[];
54
+ leituraObrigatoria: DocumentoObrigatorioMudanca[];
55
+ docsNaoLidas: DocumentoObrigatorioMudanca[];
56
+ docsAusentes: DocumentoObrigatorioMudanca[];
57
+ diagnosticos: BloqueioDocumentacaoMudanca[];
58
+ instrucoes: string[];
59
+ }
60
+
61
+ interface RegraDocumentacao {
62
+ categoria: string;
63
+ padroes: RegExp[];
64
+ docs: DocumentoPlanejado[];
65
+ }
66
+
67
+ function normalizarTexto(valor: string): string {
68
+ return valor
69
+ .normalize("NFD")
70
+ .replace(/[\u0300-\u036f]/g, "")
71
+ .toLowerCase();
72
+ }
73
+
74
+ function normalizarRelativo(valor: string): string {
75
+ return valor.replace(/\\/g, "/").replace(/^\.\//, "");
76
+ }
77
+
78
+ function tituloDeRelativo(relativo: string): string {
79
+ const base = path.basename(relativo, path.extname(relativo));
80
+ return base
81
+ .split(/[-_\s]+/)
82
+ .filter(Boolean)
83
+ .map((parte) => `${parte.charAt(0).toUpperCase()}${parte.slice(1)}`)
84
+ .join(" ");
85
+ }
86
+
87
+ async function caminhoExiste(caminho: string): Promise<boolean> {
88
+ try {
89
+ await stat(caminho);
90
+ return true;
91
+ } catch {
92
+ return false;
93
+ }
94
+ }
95
+
96
+ function caminhoEstaDentro(base: string, alvo: string): boolean {
97
+ const relativo = path.relative(base, alvo);
98
+ return relativo === "" || (!relativo.startsWith("..") && !path.isAbsolute(relativo));
99
+ }
100
+
101
+ async function listarArquivosRecursivo(base: string, limite = 80): Promise<string[]> {
102
+ const encontrados: string[] = [];
103
+
104
+ async function visitar(pasta: string): Promise<void> {
105
+ if (encontrados.length >= limite || !(await caminhoExiste(pasta))) {
106
+ return;
107
+ }
108
+
109
+ const entradas = await readdir(pasta, { withFileTypes: true });
110
+ for (const entrada of entradas.sort((a, b) => a.name.localeCompare(b.name, "pt-BR"))) {
111
+ if (encontrados.length >= limite) {
112
+ return;
113
+ }
114
+ const caminho = path.join(pasta, entrada.name);
115
+ if (entrada.isDirectory()) {
116
+ if (!["node_modules", ".git", ".tmp", "dist"].includes(entrada.name)) {
117
+ await visitar(caminho);
118
+ }
119
+ continue;
120
+ }
121
+ encontrados.push(caminho);
122
+ }
123
+ }
124
+
125
+ await visitar(base);
126
+ return encontrados;
127
+ }
128
+
129
+ const REGRAS_DOCUMENTACAO: RegraDocumentacao[] = [
130
+ {
131
+ categoria: "deploy",
132
+ padroes: [/deploy/, /publicar/, /producao/, /release/, /pipeline/, /\bci\b/, /\bcd\b/, /vercel/, /cloudflare/, /docker/],
133
+ docs: [
134
+ { relativo: "docs/deploy.md", tipo: "runbook", motivo: "runbook de deploy antes de publicar ou alterar pipeline", permitirCriacao: true },
135
+ { relativo: "docs/env.md", tipo: "runbook", motivo: "variaveis, secrets e ambientes usados no deploy", permitirCriacao: true },
136
+ { relativo: "docs/rollback.md", tipo: "runbook", motivo: "plano de rollback obrigatorio antes de deploy", permitirCriacao: true },
137
+ ],
138
+ },
139
+ {
140
+ categoria: "api",
141
+ padroes: [/rota/, /endpoint/, /\bapi\b/, /controller/, /webhook/, /http/],
142
+ docs: [
143
+ { relativo: "docs/api.md", tipo: "documentacao", motivo: "contrato operacional de rotas e endpoints", permitirCriacao: true },
144
+ ],
145
+ },
146
+ {
147
+ categoria: "auth",
148
+ padroes: [/auth/, /login/, /permiss/, /autoriz/, /token/, /oauth/, /seguranca/],
149
+ docs: [
150
+ { relativo: "docs/auth.md", tipo: "documentacao", motivo: "regras de autenticacao, autorizacao e tokens", permitirCriacao: true },
151
+ { relativo: "docs/seguranca.md", tipo: "documentacao", motivo: "politicas de seguranca afetadas pela mudanca", permitirCriacao: true },
152
+ ],
153
+ },
154
+ {
155
+ categoria: "persistencia",
156
+ padroes: [/banco/, /database/, /persist/, /migration/, /migracao/, /prisma/, /postgres/, /redis/, /mongodb/],
157
+ docs: [
158
+ { relativo: "docs/persistencia-vendor-first.md", tipo: "documentacao", motivo: "regras de persistencia vendor-first", permitirCriacao: false },
159
+ { relativo: "docs/database.md", tipo: "runbook", motivo: "operacao de banco, migracoes e validacao de dados", permitirCriacao: true },
160
+ ],
161
+ },
162
+ {
163
+ categoria: "mcp",
164
+ padroes: [/\bmcp\b/, /tool/, /ferramenta/, /server/, /stdio/],
165
+ docs: [
166
+ { relativo: "pacotes/mcp/README.md", tipo: "pacote", motivo: "documentacao publica do pacote MCP", permitirCriacao: false },
167
+ { relativo: "docs/integracao-com-ia.md", tipo: "documentacao", motivo: "integracao do Sema com agentes IA", permitirCriacao: false },
168
+ { relativo: "docs/mcp.md", tipo: "documentacao", motivo: "runbook especifico de ferramentas MCP", permitirCriacao: true },
169
+ ],
170
+ },
171
+ {
172
+ categoria: "cli",
173
+ padroes: [/\bcli\b/, /comando/, /terminal/],
174
+ docs: [
175
+ { relativo: "pacotes/cli/README.md", tipo: "pacote", motivo: "documentacao publica da CLI", permitirCriacao: false },
176
+ { relativo: "docs/cli.md", tipo: "documentacao", motivo: "guia operacional dos comandos da CLI", permitirCriacao: false },
177
+ ],
178
+ },
179
+ {
180
+ categoria: "extensao",
181
+ padroes: [/vscode/, /extension/, /extensao/, /editor/, /\blsp\b/, /language server/],
182
+ docs: [
183
+ { relativo: "pacotes/editor-vscode/README.md", tipo: "pacote", motivo: "documentacao publica da extensao VS Code", permitirCriacao: false },
184
+ { relativo: "docs/extensao-vscode.md", tipo: "documentacao", motivo: "runbook da extensao VS Code", permitirCriacao: true },
185
+ ],
186
+ },
187
+ {
188
+ categoria: "contrato",
189
+ padroes: [/contrato/, /\.sema/, /semant/, /\bsema\b/, /drift/, /governanca/],
190
+ docs: [
191
+ { relativo: "docs/sintaxe.md", tipo: "documentacao", motivo: "sintaxe e blocos de contrato Sema", permitirCriacao: false },
192
+ { relativo: "docs/fluxo-pratico-ia-sema.md", tipo: "documentacao", motivo: "fluxo IA-first antes de editar contratos", permitirCriacao: false },
193
+ ],
194
+ },
195
+ {
196
+ categoria: "frontend",
197
+ padroes: [/frontend/, /\bui\b/, /react/, /vite/, /next/, /css/, /html/, /pagina/],
198
+ docs: [
199
+ { relativo: "docs/frontend.md", tipo: "documentacao", motivo: "decisoes e validacoes de interface", permitirCriacao: true },
200
+ ],
201
+ },
202
+ {
203
+ categoria: "testes",
204
+ padroes: [/teste/, /testar/, /qa/, /validacao/, /regressao/],
205
+ docs: [
206
+ { relativo: "docs/testes.md", tipo: "runbook", motivo: "estrategia de testes e regressao", permitirCriacao: true },
207
+ ],
208
+ },
209
+ {
210
+ categoria: "documentacao",
211
+ padroes: [/doc/, /readme/, /manual/, /runbook/, /onboarding/],
212
+ docs: [
213
+ { relativo: "docs/README.md", tipo: "documentacao", motivo: "indice de documentacao do projeto", permitirCriacao: false },
214
+ { relativo: "docs/documentacao.md", tipo: "documentacao", motivo: "politica de atualizacao documental", permitirCriacao: true },
215
+ ],
216
+ },
217
+ ];
218
+
219
+ function inferirCategorias(intencao: string, arquivosAlvo: string[]): string[] {
220
+ const texto = normalizarTexto(`${intencao} ${arquivosAlvo.join(" ")}`);
221
+ const categorias = REGRAS_DOCUMENTACAO
222
+ .filter((regra) => regra.padroes.some((padrao) => padrao.test(texto)))
223
+ .map((regra) => regra.categoria);
224
+ return categorias.length > 0 ? [...new Set(categorias)] : ["mudanca"];
225
+ }
226
+
227
+ function criarTemplateDoc(doc: DocumentoPlanejado, intencao: string, categorias: string[], arquivosAlvo: string[]): string {
228
+ const titulo = tituloDeRelativo(doc.relativo);
229
+ const linhas = [
230
+ `# ${titulo}`,
231
+ "",
232
+ "Documento criado pelo Sema porque uma IA declarou uma intencao que exige leitura operacional antes de agir.",
233
+ "",
234
+ "## Intencao",
235
+ "",
236
+ intencao || "Descrever a intencao operacional antes de executar a mudanca.",
237
+ "",
238
+ "## Escopo",
239
+ "",
240
+ arquivosAlvo.length > 0
241
+ ? arquivosAlvo.map((arquivo) => `- ${arquivo}`).join("\n")
242
+ : "- Declarar os arquivos, servicos ou contratos afetados.",
243
+ "",
244
+ "## Leitura obrigatoria antes de agir",
245
+ "",
246
+ "- Ler este documento inteiro.",
247
+ "- Ler contratos .sema relacionados antes de editar codigo.",
248
+ "- Ler README, runbooks e docs de pacote envolvidos.",
249
+ "",
250
+ "## Procedimento",
251
+ "",
252
+ "- Descrever os comandos ou passos seguros.",
253
+ "- Declarar pre-condicoes, variaveis e dependencias.",
254
+ "- Registrar como validar sucesso.",
255
+ "",
256
+ "## Validacao",
257
+ "",
258
+ "- Listar comandos de build, teste, verificacao ou smoke test.",
259
+ "- Registrar evidencias esperadas.",
260
+ "",
261
+ "## Rollback",
262
+ "",
263
+ "- Descrever como desfazer a mudanca.",
264
+ "- Declarar sinais que exigem rollback.",
265
+ "",
266
+ "## Atualizacao documental",
267
+ "",
268
+ "- Atualizar este arquivo quando o fluxo operacional mudar.",
269
+ "- Atualizar docs e contratos relacionados na mesma mudanca.",
270
+ "",
271
+ `Categorias inferidas: ${categorias.join(", ")}`,
272
+ "",
273
+ ];
274
+
275
+ return `${linhas.join("\n")}\n`;
276
+ }
277
+
278
+ function resumoConteudo(conteudo: string): { conteudo: string; truncado: boolean } {
279
+ if (conteudo.length <= LIMITE_CONTEUDO_DOC) {
280
+ return { conteudo, truncado: false };
281
+ }
282
+
283
+ return {
284
+ conteudo: `${conteudo.slice(0, LIMITE_CONTEUDO_DOC)}\n\n[TRUNCADO PELO SEMA: leia o arquivo completo antes de agir.]`,
285
+ truncado: true,
286
+ };
287
+ }
288
+
289
+ async function adicionarDocsRaizExistentes(baseProjeto: string, registrar: (doc: DocumentoPlanejado) => void): Promise<void> {
290
+ const docsRaiz = [
291
+ { relativo: "AGENTS.md", tipo: "raiz" as const, motivo: "regras obrigatorias do agente no projeto" },
292
+ { relativo: "SEMA_INDEX.json", tipo: "raiz" as const, motivo: "indice IA-first do projeto" },
293
+ { relativo: "README.md", tipo: "raiz" as const, motivo: "visao geral publica do projeto" },
294
+ { relativo: "docs/README.md", tipo: "documentacao" as const, motivo: "indice de documentacao" },
295
+ ];
296
+
297
+ for (const doc of docsRaiz) {
298
+ if (await caminhoExiste(path.join(baseProjeto, doc.relativo))) {
299
+ registrar({ ...doc, permitirCriacao: false });
300
+ }
301
+ }
302
+ }
303
+
304
+ async function adicionarReadmesDeArquivos(
305
+ baseProjeto: string,
306
+ arquivosAlvo: string[],
307
+ registrar: (doc: DocumentoPlanejado) => void,
308
+ ): Promise<void> {
309
+ for (const arquivo of arquivosAlvo) {
310
+ const absoluto = path.resolve(baseProjeto, arquivo);
311
+ if (!caminhoEstaDentro(baseProjeto, absoluto)) {
312
+ continue;
313
+ }
314
+
315
+ let pasta = (await caminhoExiste(absoluto)) && (await stat(absoluto)).isDirectory()
316
+ ? absoluto
317
+ : path.dirname(absoluto);
318
+
319
+ while (caminhoEstaDentro(baseProjeto, pasta)) {
320
+ const readme = path.join(pasta, "README.md");
321
+ if (await caminhoExiste(readme)) {
322
+ registrar({
323
+ relativo: normalizarRelativo(path.relative(baseProjeto, readme)),
324
+ tipo: "pacote",
325
+ motivo: `README mais proximo de ${normalizarRelativo(arquivo)}`,
326
+ permitirCriacao: false,
327
+ });
328
+ break;
329
+ }
330
+
331
+ const proxima = path.dirname(pasta);
332
+ if (proxima === pasta) {
333
+ break;
334
+ }
335
+ pasta = proxima;
336
+ }
337
+ }
338
+ }
339
+
340
+ async function adicionarContratosRelacionados(
341
+ baseProjeto: string,
342
+ categorias: string[],
343
+ intencao: string,
344
+ arquivosAlvo: string[],
345
+ registrar: (doc: DocumentoPlanejado) => void,
346
+ ): Promise<void> {
347
+ const pastaContratos = path.join(baseProjeto, "contratos");
348
+ if (!(await caminhoExiste(pastaContratos))) {
349
+ return;
350
+ }
351
+
352
+ const texto = normalizarTexto(`${intencao} ${arquivosAlvo.join(" ")} ${categorias.join(" ")}`);
353
+ const arquivos = (await listarArquivosRecursivo(pastaContratos)).filter((arquivo) => arquivo.endsWith(".sema"));
354
+
355
+ for (const arquivo of arquivos) {
356
+ const relativo = normalizarRelativo(path.relative(baseProjeto, arquivo));
357
+ const nome = normalizarTexto(relativo);
358
+ const relacionado =
359
+ texto.split(/\W+/).filter((parte) => parte.length >= 4).some((parte) => nome.includes(parte))
360
+ || (categorias.some((categoria) => ["mcp", "cli", "documentacao", "contrato"].includes(categoria)) && nome.includes("governanca_ia"));
361
+
362
+ if (relacionado) {
363
+ registrar({
364
+ relativo,
365
+ tipo: "contrato",
366
+ motivo: "contrato Sema relacionado a intencao declarada",
367
+ permitirCriacao: false,
368
+ });
369
+ }
370
+ }
371
+ }
372
+
373
+ function criarRegistrador(baseProjeto: string): {
374
+ docs: DocumentoPlanejado[];
375
+ registrar: (doc: DocumentoPlanejado) => void;
376
+ } {
377
+ const docs: DocumentoPlanejado[] = [];
378
+ const vistos = new Set<string>();
379
+
380
+ return {
381
+ docs,
382
+ registrar(doc) {
383
+ const relativo = normalizarRelativo(doc.relativo);
384
+ const absoluto = path.resolve(baseProjeto, relativo);
385
+ if (!caminhoEstaDentro(baseProjeto, absoluto) || vistos.has(relativo)) {
386
+ return;
387
+ }
388
+ vistos.add(relativo);
389
+ docs.push({ ...doc, relativo });
390
+ },
391
+ };
392
+ }
393
+
394
+ export async function resolverDocumentacaoObrigatoria(opcoes: {
395
+ baseProjeto?: string;
396
+ intencao: string;
397
+ arquivosAlvo?: string[];
398
+ criarAusentes?: boolean;
399
+ }): Promise<ResultadoDocumentacaoObrigatoria> {
400
+ const baseProjeto = path.resolve(opcoes.baseProjeto ?? process.cwd());
401
+ const intencao = opcoes.intencao.trim();
402
+ const arquivosAlvo = [...new Set((opcoes.arquivosAlvo ?? []).map(normalizarRelativo).filter(Boolean))];
403
+ const categorias = inferirCategorias(intencao, arquivosAlvo);
404
+ const { docs, registrar } = criarRegistrador(baseProjeto);
405
+
406
+ await adicionarDocsRaizExistentes(baseProjeto, registrar);
407
+
408
+ for (const regra of REGRAS_DOCUMENTACAO.filter((item) => categorias.includes(item.categoria))) {
409
+ for (const doc of regra.docs) {
410
+ registrar(doc);
411
+ }
412
+ }
413
+
414
+ await adicionarReadmesDeArquivos(baseProjeto, arquivosAlvo, registrar);
415
+ await adicionarContratosRelacionados(baseProjeto, categorias, intencao, arquivosAlvo, registrar);
416
+
417
+ const leituraObrigatoria: DocumentoObrigatorioMudanca[] = [];
418
+
419
+ for (const doc of docs) {
420
+ const caminho = path.resolve(baseProjeto, doc.relativo);
421
+ let existe = await caminhoExiste(caminho);
422
+ let criado = false;
423
+ const template = criarTemplateDoc(doc, intencao, categorias, arquivosAlvo);
424
+
425
+ if (!existe && opcoes.criarAusentes && doc.permitirCriacao) {
426
+ await mkdir(path.dirname(caminho), { recursive: true });
427
+ await writeFile(caminho, template, "utf8");
428
+ existe = true;
429
+ criado = true;
430
+ }
431
+
432
+ const item: DocumentoObrigatorioMudanca = {
433
+ caminho,
434
+ relativo: doc.relativo,
435
+ tipo: doc.tipo,
436
+ motivo: doc.motivo,
437
+ existe,
438
+ criado,
439
+ obrigatorio: true,
440
+ template: existe ? undefined : template,
441
+ };
442
+
443
+ if (existe) {
444
+ const conteudo = resumoConteudo(await readFile(caminho, "utf8"));
445
+ item.conteudo = conteudo.conteudo;
446
+ item.truncado = conteudo.truncado;
447
+ }
448
+
449
+ leituraObrigatoria.push(item);
450
+ }
451
+
452
+ const docsAusentes = leituraObrigatoria.filter((doc) => !doc.existe);
453
+ const docsCriadas = leituraObrigatoria.filter((doc) => doc.criado);
454
+ const bloqueios = docsAusentes.map((doc): BloqueioDocumentacaoMudanca => ({
455
+ tipo: "documentacao_ausente",
456
+ severidade: 4,
457
+ caminho: doc.relativo,
458
+ mensagem: `Documento obrigatorio ausente: ${doc.relativo}. Crie ou rode com --criar-ausentes antes de agir.`,
459
+ }));
460
+
461
+ return {
462
+ sucesso: bloqueios.length === 0,
463
+ baseProjeto,
464
+ intencao,
465
+ categorias,
466
+ arquivosAlvo,
467
+ leituraObrigatoria,
468
+ docsAusentes,
469
+ docsCriadas,
470
+ bloqueios,
471
+ instrucoes: [
472
+ "A IA deve ler todos os itens de leituraObrigatoria antes de executar a mudanca.",
473
+ "Se docsAusentes nao estiver vazio, crie ou preencha os documentos antes de editar codigo, contrato, deploy ou configuracao.",
474
+ "Antes de concluir, rode sema finalizar-mudanca informando as docs lidas.",
475
+ ],
476
+ };
477
+ }
478
+
479
+ function normalizarDocLida(baseProjeto: string, valor: string): string {
480
+ const semEspaco = valor.trim();
481
+ if (!semEspaco) {
482
+ return "";
483
+ }
484
+ const absoluto = path.isAbsolute(semEspaco) ? semEspaco : path.resolve(baseProjeto, semEspaco);
485
+ return normalizarRelativo(path.relative(baseProjeto, absoluto));
486
+ }
487
+
488
+ export async function verificarDocumentacaoMudanca(opcoes: {
489
+ baseProjeto?: string;
490
+ intencao: string;
491
+ arquivosAlvo?: string[];
492
+ docsLidas?: string[];
493
+ }): Promise<ResultadoVerificacaoDocumentacaoMudanca> {
494
+ const baseProjeto = path.resolve(opcoes.baseProjeto ?? process.cwd());
495
+ const impacto = await resolverDocumentacaoObrigatoria({
496
+ baseProjeto,
497
+ intencao: opcoes.intencao,
498
+ arquivosAlvo: opcoes.arquivosAlvo,
499
+ criarAusentes: false,
500
+ });
501
+
502
+ const docsLidas = [...new Set((opcoes.docsLidas ?? []).map((doc) => normalizarDocLida(baseProjeto, doc)).filter(Boolean))];
503
+ const setDocsLidas = new Set(docsLidas);
504
+ const docsNaoLidas = impacto.leituraObrigatoria.filter((doc) => doc.existe && !setDocsLidas.has(doc.relativo));
505
+
506
+ const diagnosticos: BloqueioDocumentacaoMudanca[] = [
507
+ ...impacto.docsAusentes.map((doc): BloqueioDocumentacaoMudanca => ({
508
+ tipo: "documentacao_ausente",
509
+ severidade: 4,
510
+ caminho: doc.relativo,
511
+ mensagem: `Documento obrigatorio ausente: ${doc.relativo}.`,
512
+ })),
513
+ ...docsNaoLidas.map((doc): BloqueioDocumentacaoMudanca => ({
514
+ tipo: "leitura_obrigatoria_nao_comprovada",
515
+ severidade: 4,
516
+ caminho: doc.relativo,
517
+ mensagem: `Leitura obrigatoria nao comprovada: ${doc.relativo}.`,
518
+ })),
519
+ ];
520
+
521
+ return {
522
+ sucesso: diagnosticos.length === 0,
523
+ baseProjeto,
524
+ intencao: opcoes.intencao,
525
+ categorias: impacto.categorias,
526
+ docsLidas,
527
+ leituraObrigatoria: impacto.leituraObrigatoria,
528
+ docsNaoLidas,
529
+ docsAusentes: impacto.docsAusentes,
530
+ diagnosticos,
531
+ instrucoes: diagnosticos.length === 0
532
+ ? ["Leitura documental comprovada. Continue com validar, drift e testes conforme a mudanca."]
533
+ : ["Nao conclua a mudanca enquanto houver diagnosticos de severidade 4+."],
534
+ };
535
+ }