@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.
- package/README.md +2 -1
- package/dist/index.js +76 -43
- package/package.json +26 -35
- package/semacode-cli-1.3.1.tgz +0 -0
- package/src/angular-consumer-standalone.ts +312 -0
- package/src/cpp-symbols.ts +82 -0
- package/src/docs.ts +535 -0
- package/src/dotnet-http.ts +355 -0
- package/src/drift.ts +4933 -0
- package/src/go-http.ts +118 -0
- package/src/importador.ts +3891 -0
- package/src/index.ts +5641 -0
- package/src/java-http.ts +247 -0
- package/src/lua-symbols.ts +114 -0
- package/src/php-symbols.ts +462 -0
- package/src/projeto.ts +862 -0
- package/src/python-http.ts +258 -0
- package/src/rust-http.ts +125 -0
- package/src/tipos.ts +24 -0
- package/src/typescript-http.ts +1076 -0
- package/tsconfig.json +20 -0
- package/AGENTS.md +0 -272
- package/LICENSE +0 -22
- package/SEMA_BRIEF.curto.txt +0 -9
- package/SEMA_BRIEF.md +0 -63
- package/SEMA_BRIEF.micro.txt +0 -7
- package/SEMA_INDEX.json +0 -799
- package/dist/angular-consumer-standalone.d.ts +0 -6
- package/dist/angular-consumer-standalone.js.map +0 -1
- package/dist/cpp-symbols.d.ts +0 -10
- package/dist/cpp-symbols.js.map +0 -1
- package/dist/docs.d.ts +0 -56
- package/dist/docs.js.map +0 -1
- package/dist/dotnet-http.d.ts +0 -23
- package/dist/dotnet-http.js.map +0 -1
- package/dist/drift.d.ts +0 -225
- package/dist/drift.js.map +0 -1
- package/dist/go-http.d.ts +0 -23
- package/dist/go-http.js.map +0 -1
- package/dist/importador.d.ts +0 -31
- package/dist/importador.js.map +0 -1
- package/dist/index.d.ts +0 -2
- package/dist/index.js.map +0 -1
- package/dist/java-http.d.ts +0 -23
- package/dist/java-http.js.map +0 -1
- package/dist/lua-symbols.d.ts +0 -10
- package/dist/lua-symbols.js.map +0 -1
- package/dist/php-symbols.d.ts +0 -24
- package/dist/php-symbols.js.map +0 -1
- package/dist/projeto.d.ts +0 -53
- package/dist/projeto.js.map +0 -1
- package/dist/python-http.d.ts +0 -23
- package/dist/python-http.js.map +0 -1
- package/dist/rust-http.d.ts +0 -23
- package/dist/rust-http.js.map +0 -1
- package/dist/tipos.d.ts +0 -3
- package/dist/tipos.js.map +0 -1
- package/dist/typescript-http.d.ts +0 -35
- package/dist/typescript-http.js.map +0 -1
- package/docs/AGENT_STARTER.md +0 -102
- package/docs/cli.md +0 -117
- package/docs/como-ensinar-a-sema-para-ia.md +0 -149
- package/docs/deploy.md +0 -70
- package/docs/documentacao.md +0 -63
- package/docs/env.md +0 -56
- package/docs/extensao-vscode.md +0 -45
- package/docs/fluxo-pratico-ia-sema.md +0 -177
- package/docs/instalacao-e-primeiro-uso.md +0 -112
- package/docs/integracao-com-ia.md +0 -101
- package/docs/mcp.md +0 -53
- package/docs/pagamento-ponta-a-ponta.md +0 -155
- package/docs/persistencia-vendor-first.md +0 -145
- package/docs/prompt-base-ia-sema.md +0 -104
- package/docs/rollback.md +0 -47
- package/docs/sintaxe.md +0 -410
- package/exemplos/agendamento.sema +0 -106
- package/exemplos/assinatura.sema +0 -136
- package/exemplos/auditoria.sema +0 -88
- package/exemplos/autenticacao.sema +0 -125
- package/exemplos/automacao.sema +0 -107
- package/exemplos/cadastro_usuario.sema +0 -54
- package/exemplos/calculadora.sema +0 -78
- package/exemplos/crud_simples.sema +0 -89
- package/exemplos/estoque.sema +0 -126
- package/exemplos/exportacao.sema +0 -94
- package/exemplos/fila.sema +0 -131
- package/exemplos/integracao_externa.sema +0 -94
- package/exemplos/multi_tenant.sema +0 -140
- package/exemplos/notificacao.sema +0 -98
- package/exemplos/operacao_estrategia.sema +0 -402
- package/exemplos/pagamento.sema +0 -222
- package/exemplos/pagamento_dominio.sema +0 -35
- package/exemplos/pedido.sema +0 -119
- package/exemplos/permissao.sema +0 -121
- package/exemplos/persistencia_vendor_first.sema +0 -86
- package/exemplos/relatorio.sema +0 -93
- package/exemplos/testes_embutidos.sema +0 -45
- package/exemplos/tratamento_erro.sema +0 -157
- package/exemplos/upload_arquivo.sema +0 -93
- package/exemplos/webhook.sema +0 -96
- package/llms-full.txt +0 -34
- package/llms.txt +0 -17
- package/node_modules/@sema/gerador-css/dist/index.d.ts +0 -3
- package/node_modules/@sema/gerador-css/dist/index.js +0 -592
- package/node_modules/@sema/gerador-css/dist/index.js.map +0 -1
- package/node_modules/@sema/gerador-css/package.json +0 -7
- package/node_modules/@sema/gerador-dart/dist/index.d.ts +0 -3
- package/node_modules/@sema/gerador-dart/dist/index.js +0 -44
- package/node_modules/@sema/gerador-dart/dist/index.js.map +0 -1
- package/node_modules/@sema/gerador-dart/package.json +0 -7
- package/node_modules/@sema/gerador-html/dist/index.d.ts +0 -3
- package/node_modules/@sema/gerador-html/dist/index.js +0 -163
- package/node_modules/@sema/gerador-html/dist/index.js.map +0 -1
- package/node_modules/@sema/gerador-html/package.json +0 -7
- package/node_modules/@sema/gerador-javascript/dist/index.d.ts +0 -3
- package/node_modules/@sema/gerador-javascript/dist/index.js +0 -421
- package/node_modules/@sema/gerador-javascript/dist/index.js.map +0 -1
- package/node_modules/@sema/gerador-javascript/package.json +0 -7
- package/node_modules/@sema/gerador-lua/dist/index.d.ts +0 -3
- package/node_modules/@sema/gerador-lua/dist/index.js +0 -328
- package/node_modules/@sema/gerador-lua/dist/index.js.map +0 -1
- package/node_modules/@sema/gerador-lua/package.json +0 -7
- package/node_modules/@sema/gerador-python/dist/index.d.ts +0 -6
- package/node_modules/@sema/gerador-python/dist/index.js +0 -729
- package/node_modules/@sema/gerador-python/dist/index.js.map +0 -1
- package/node_modules/@sema/gerador-python/package.json +0 -7
- package/node_modules/@sema/gerador-typescript/dist/index.d.ts +0 -6
- package/node_modules/@sema/gerador-typescript/dist/index.js +0 -793
- package/node_modules/@sema/gerador-typescript/dist/index.js.map +0 -1
- package/node_modules/@sema/gerador-typescript/package.json +0 -7
- package/node_modules/@sema/nucleo/dist/ast/tipos.d.ts +0 -125
- package/node_modules/@sema/nucleo/dist/ast/tipos.js +0 -2
- package/node_modules/@sema/nucleo/dist/ast/tipos.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/diagnosticos/index.d.ts +0 -21
- package/node_modules/@sema/nucleo/dist/diagnosticos/index.js +0 -12
- package/node_modules/@sema/nucleo/dist/diagnosticos/index.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/formatador/index.d.ts +0 -9
- package/node_modules/@sema/nucleo/dist/formatador/index.js +0 -487
- package/node_modules/@sema/nucleo/dist/formatador/index.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/index.d.ts +0 -35
- package/node_modules/@sema/nucleo/dist/index.js +0 -96
- package/node_modules/@sema/nucleo/dist/index.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/ir/conversor.d.ts +0 -5
- package/node_modules/@sema/nucleo/dist/ir/conversor.js +0 -1058
- package/node_modules/@sema/nucleo/dist/ir/conversor.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/ir/modelos.d.ts +0 -377
- package/node_modules/@sema/nucleo/dist/ir/modelos.js +0 -2
- package/node_modules/@sema/nucleo/dist/ir/modelos.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/lexer/lexer.d.ts +0 -7
- package/node_modules/@sema/nucleo/dist/lexer/lexer.js +0 -122
- package/node_modules/@sema/nucleo/dist/lexer/lexer.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/lexer/tokens.d.ts +0 -8
- package/node_modules/@sema/nucleo/dist/lexer/tokens.js +0 -82
- package/node_modules/@sema/nucleo/dist/lexer/tokens.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/parser/parser.d.ts +0 -9
- package/node_modules/@sema/nucleo/dist/parser/parser.js +0 -807
- package/node_modules/@sema/nucleo/dist/parser/parser.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/persistencia/contratos.d.ts +0 -39
- package/node_modules/@sema/nucleo/dist/persistencia/contratos.js +0 -294
- package/node_modules/@sema/nucleo/dist/persistencia/contratos.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/semantico/analisador.d.ts +0 -58
- package/node_modules/@sema/nucleo/dist/semantico/analisador.js +0 -1912
- package/node_modules/@sema/nucleo/dist/semantico/analisador.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/semantico/estruturas.d.ts +0 -104
- package/node_modules/@sema/nucleo/dist/semantico/estruturas.js +0 -445
- package/node_modules/@sema/nucleo/dist/semantico/estruturas.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/semantico/seguranca.d.ts +0 -91
- package/node_modules/@sema/nucleo/dist/semantico/seguranca.js +0 -258
- package/node_modules/@sema/nucleo/dist/semantico/seguranca.js.map +0 -1
- package/node_modules/@sema/nucleo/dist/util/arquivos.d.ts +0 -2
- package/node_modules/@sema/nucleo/dist/util/arquivos.js +0 -25
- package/node_modules/@sema/nucleo/dist/util/arquivos.js.map +0 -1
- package/node_modules/@sema/nucleo/package.json +0 -7
- package/node_modules/@sema/padroes/dist/index.d.ts +0 -25
- package/node_modules/@sema/padroes/dist/index.js +0 -316
- package/node_modules/@sema/padroes/dist/index.js.map +0 -1
- 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
|
+
}
|