@semacode/cli 1.3.2 → 1.3.6

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 (96) hide show
  1. package/dist/index.js +33 -27
  2. package/dist/index.js.map +1 -1
  3. package/node_modules/@sema/gerador-css/package.json +10 -3
  4. package/node_modules/@sema/gerador-css/src/index.ts +605 -0
  5. package/node_modules/@sema/gerador-css/tsconfig.json +13 -0
  6. package/node_modules/@sema/gerador-css/tsconfig.tsbuildinfo +1 -0
  7. package/node_modules/@sema/gerador-dart/package.json +10 -3
  8. package/node_modules/@sema/gerador-dart/src/index.ts +52 -0
  9. package/node_modules/@sema/gerador-dart/tsconfig.json +13 -0
  10. package/node_modules/@sema/gerador-dart/tsconfig.tsbuildinfo +1 -0
  11. package/node_modules/@sema/gerador-html/package.json +10 -3
  12. package/node_modules/@sema/gerador-html/src/index.ts +185 -0
  13. package/node_modules/@sema/gerador-html/tsconfig.json +13 -0
  14. package/node_modules/@sema/gerador-html/tsconfig.tsbuildinfo +1 -0
  15. package/node_modules/@sema/gerador-javascript/package.json +10 -3
  16. package/node_modules/@sema/gerador-javascript/src/index.ts +461 -0
  17. package/node_modules/@sema/gerador-javascript/tsconfig.json +13 -0
  18. package/node_modules/@sema/gerador-javascript/tsconfig.tsbuildinfo +1 -0
  19. package/node_modules/@sema/gerador-lua/package.json +10 -3
  20. package/node_modules/@sema/gerador-lua/src/index.d.ts +3 -0
  21. package/node_modules/@sema/gerador-lua/src/index.js.map +1 -0
  22. package/node_modules/@sema/gerador-lua/src/index.ts +359 -0
  23. package/node_modules/@sema/gerador-lua/tsconfig.json +13 -0
  24. package/node_modules/@sema/gerador-lua/tsconfig.tsbuildinfo +1 -0
  25. package/node_modules/@sema/gerador-python/package.json +10 -3
  26. package/node_modules/@sema/gerador-python/src/index.ts +706 -0
  27. package/node_modules/@sema/gerador-python/tsconfig.json +13 -0
  28. package/node_modules/@sema/gerador-python/tsconfig.tsbuildinfo +1 -0
  29. package/node_modules/@sema/gerador-typescript/package.json +10 -3
  30. package/node_modules/@sema/gerador-typescript/src/index.ts +728 -0
  31. package/node_modules/@sema/gerador-typescript/tsconfig.json +13 -0
  32. package/node_modules/@sema/gerador-typescript/tsconfig.tsbuildinfo +1 -0
  33. package/node_modules/@sema/nucleo/package.json +6 -3
  34. package/node_modules/@sema/nucleo/src/ast/tipos.ts +191 -0
  35. package/node_modules/@sema/nucleo/src/diagnosticos/index.ts +43 -0
  36. package/node_modules/@sema/nucleo/src/formatador/index.ts +507 -0
  37. package/node_modules/@sema/nucleo/src/index.ts +133 -0
  38. package/node_modules/@sema/nucleo/src/ir/conversor.ts +912 -0
  39. package/node_modules/@sema/nucleo/src/ir/modelos.ts +331 -0
  40. package/node_modules/@sema/nucleo/src/lexer/lexer.ts +166 -0
  41. package/node_modules/@sema/nucleo/src/lexer/tokens.ts +64 -0
  42. package/node_modules/@sema/nucleo/src/parser/gramatica.ebnf +39 -0
  43. package/node_modules/@sema/nucleo/src/parser/parser.ts +790 -0
  44. package/node_modules/@sema/nucleo/src/semantico/analisador.ts +2692 -0
  45. package/node_modules/@sema/nucleo/src/semantico/estruturas.ts +632 -0
  46. package/node_modules/@sema/nucleo/src/semantico/seguranca.ts +362 -0
  47. package/node_modules/@sema/nucleo/src/util/arquivos.ts +28 -0
  48. package/node_modules/@sema/nucleo/tsconfig.json +9 -0
  49. package/node_modules/@sema/nucleo/tsconfig.tsbuildinfo +1 -0
  50. package/node_modules/@sema/padroes/package.json +6 -3
  51. package/node_modules/@sema/padroes/src/index.ts +251 -0
  52. package/node_modules/@sema/padroes/tsconfig.json +9 -0
  53. package/node_modules/@sema/padroes/tsconfig.tsbuildinfo +1 -0
  54. package/package.json +21 -33
  55. package/AGENTS.md +0 -272
  56. package/LICENSE +0 -22
  57. package/README.md +0 -73
  58. package/SEMA_BRIEF.curto.txt +0 -9
  59. package/SEMA_BRIEF.md +0 -49
  60. package/SEMA_BRIEF.micro.txt +0 -7
  61. package/SEMA_INDEX.json +0 -546
  62. package/docs/AGENT_STARTER.md +0 -102
  63. package/docs/como-ensinar-a-sema-para-ia.md +0 -149
  64. package/docs/fluxo-pratico-ia-sema.md +0 -177
  65. package/docs/instalacao-e-primeiro-uso.md +0 -196
  66. package/docs/integracao-com-ia.md +0 -228
  67. package/docs/pagamento-ponta-a-ponta.md +0 -155
  68. package/docs/prompt-base-ia-sema.md +0 -104
  69. package/docs/sintaxe.md +0 -361
  70. package/exemplos/agendamento.sema +0 -106
  71. package/exemplos/assinatura.sema +0 -136
  72. package/exemplos/auditoria.sema +0 -88
  73. package/exemplos/autenticacao.sema +0 -125
  74. package/exemplos/automacao.sema +0 -107
  75. package/exemplos/cadastro_usuario.sema +0 -54
  76. package/exemplos/calculadora.sema +0 -78
  77. package/exemplos/crud_simples.sema +0 -89
  78. package/exemplos/estoque.sema +0 -126
  79. package/exemplos/exportacao.sema +0 -94
  80. package/exemplos/fila.sema +0 -131
  81. package/exemplos/integracao_externa.sema +0 -94
  82. package/exemplos/multi_tenant.sema +0 -140
  83. package/exemplos/notificacao.sema +0 -98
  84. package/exemplos/operacao_estrategia.sema +0 -402
  85. package/exemplos/pagamento.sema +0 -222
  86. package/exemplos/pagamento_dominio.sema +0 -35
  87. package/exemplos/pedido.sema +0 -119
  88. package/exemplos/permissao.sema +0 -121
  89. package/exemplos/relatorio.sema +0 -93
  90. package/exemplos/testes_embutidos.sema +0 -45
  91. package/exemplos/tratamento_erro.sema +0 -157
  92. package/exemplos/upload_arquivo.sema +0 -93
  93. package/exemplos/webhook.sema +0 -96
  94. package/llms-full.txt +0 -34
  95. package/llms.txt +0 -17
  96. package/logo.png +0 -0
@@ -0,0 +1,632 @@
1
+ import type { IsolamentoEfeitoSemantico, PrivilegioEfeitoSemantico } from "./seguranca.js";
2
+
3
+ export type TipoExpressaoSemantica = "existe" | "comparacao" | "predicado" | "pertencimento" | "composta" | "negacao";
4
+ export type CategoriaEfeitoSemantico =
5
+ | "persistencia"
6
+ | "consulta"
7
+ | "evento"
8
+ | "notificacao"
9
+ | "auditoria"
10
+ | "db.read"
11
+ | "db.write"
12
+ | "queue.publish"
13
+ | "queue.consume"
14
+ | "fs.read"
15
+ | "fs.write"
16
+ | "network.egress"
17
+ | "secret.read"
18
+ | "shell.exec";
19
+ export type CriticidadeEfeitoSemantico = "baixa" | "media" | "alta" | "critica";
20
+
21
+ export interface ExpressaoBaseSemantica {
22
+ tipo: TipoExpressaoSemantica;
23
+ textoOriginal: string;
24
+ }
25
+
26
+ export interface ExpressaoExisteSemantica extends ExpressaoBaseSemantica {
27
+ tipo: "existe";
28
+ alvo: string;
29
+ }
30
+
31
+ export interface ExpressaoComparacaoSemantica extends ExpressaoBaseSemantica {
32
+ tipo: "comparacao";
33
+ alvo: string;
34
+ operador: "==" | "!=" | ">" | ">=" | "<" | "<=";
35
+ valor: string;
36
+ valorLiteral?: boolean;
37
+ }
38
+
39
+ export interface ExpressaoPredicadoSemantica extends ExpressaoBaseSemantica {
40
+ tipo: "predicado";
41
+ alvo: string;
42
+ predicado: string;
43
+ argumentos?: string;
44
+ }
45
+
46
+ export interface ExpressaoPertencimentoSemantica extends ExpressaoBaseSemantica {
47
+ tipo: "pertencimento";
48
+ alvo: string;
49
+ valores: string[];
50
+ }
51
+
52
+ export interface ExpressaoCompostaSemantica extends ExpressaoBaseSemantica {
53
+ tipo: "composta";
54
+ operadorLogico: "e" | "ou";
55
+ termos: ExpressaoSemantica[];
56
+ }
57
+
58
+ export interface ExpressaoNegacaoSemantica extends ExpressaoBaseSemantica {
59
+ tipo: "negacao";
60
+ termo: ExpressaoSemantica;
61
+ }
62
+
63
+ export type ExpressaoSemantica =
64
+ | ExpressaoExisteSemantica
65
+ | ExpressaoComparacaoSemantica
66
+ | ExpressaoPredicadoSemantica
67
+ | ExpressaoPertencimentoSemantica
68
+ | ExpressaoCompostaSemantica
69
+ | ExpressaoNegacaoSemantica;
70
+
71
+ export interface EfeitoSemantico {
72
+ textoOriginal: string;
73
+ categoria: CategoriaEfeitoSemantico;
74
+ alvo: string;
75
+ detalhe?: string;
76
+ criticidade?: CriticidadeEfeitoSemantico;
77
+ criticidadeTexto?: string;
78
+ privilegio?: PrivilegioEfeitoSemantico;
79
+ privilegioTexto?: string;
80
+ isolamento?: IsolamentoEfeitoSemantico;
81
+ isolamentoTexto?: string;
82
+ }
83
+
84
+ export interface TransicaoEstadoSemantica {
85
+ textoOriginal: string;
86
+ origem: string;
87
+ destino: string;
88
+ }
89
+
90
+ export interface EtapaFlowSemantica {
91
+ textoOriginal: string;
92
+ nome: string;
93
+ task?: string;
94
+ condicao?: ExpressaoSemantica;
95
+ dependencias: string[];
96
+ mapeamentos: Array<{ campo: string; valor: string }>;
97
+ emSucesso?: string;
98
+ emErro?: string;
99
+ porErro: Array<{ tipo: string; destino: string }>;
100
+ }
101
+
102
+ export interface ContratoErroRouteSemantico {
103
+ nome: string;
104
+ codigo: string;
105
+ mensagem?: string;
106
+ }
107
+
108
+ export interface ContratoRouteSemantico {
109
+ metodo?: string;
110
+ caminho?: string;
111
+ task?: string;
112
+ inputPublico: Array<{ nome: string; tipo: string; modificadores: string[] }>;
113
+ outputPublico: Array<{ nome: string; tipo: string; modificadores: string[] }>;
114
+ errosPublicos: ContratoErroRouteSemantico[];
115
+ effectsPublicos: EfeitoSemantico[];
116
+ }
117
+
118
+ const CATEGORIAS_EFEITO = new Set<CategoriaEfeitoSemantico>([
119
+ "persistencia",
120
+ "consulta",
121
+ "evento",
122
+ "notificacao",
123
+ "auditoria",
124
+ "db.read",
125
+ "db.write",
126
+ "queue.publish",
127
+ "queue.consume",
128
+ "fs.read",
129
+ "fs.write",
130
+ "network.egress",
131
+ "secret.read",
132
+ "shell.exec",
133
+ ]);
134
+
135
+ const CRITICIDADES_EFEITO = new Set<CriticidadeEfeitoSemantico>([
136
+ "baixa",
137
+ "media",
138
+ "alta",
139
+ "critica",
140
+ ]);
141
+
142
+ const MAPEAMENTO_EFEITOS_LEGADOS: Record<string, CategoriaEfeitoSemantico> = {
143
+ grava: "persistencia",
144
+ atualiza: "persistencia",
145
+ persiste: "persistencia",
146
+ consulta: "consulta",
147
+ le: "consulta",
148
+ acessa: "consulta",
149
+ emite: "evento",
150
+ notifica: "notificacao",
151
+ envia: "notificacao",
152
+ registra: "auditoria",
153
+ audita: "auditoria",
154
+ };
155
+
156
+ const PRIVILEGIOS_EFEITO = new Set<PrivilegioEfeitoSemantico>([
157
+ "leitura",
158
+ "escrita",
159
+ "publicacao",
160
+ "execucao",
161
+ "admin",
162
+ "egress",
163
+ ]);
164
+
165
+ const ISOLAMENTOS_EFEITO = new Set<IsolamentoEfeitoSemantico>([
166
+ "tenant",
167
+ "processo",
168
+ "host",
169
+ "vps",
170
+ "global",
171
+ ]);
172
+
173
+ const OPERADORES_COMPARACAO = new Set(["==", "!=", ">", ">=", "<", "<="]);
174
+
175
+ function removerParentesesExternos(texto: string): string {
176
+ let atual = texto.trim();
177
+ while (atual.startsWith("(") && atual.endsWith(")")) {
178
+ let profundidade = 0;
179
+ let removeu = true;
180
+ for (let indice = 0; indice < atual.length; indice += 1) {
181
+ const caractere = atual[indice]!;
182
+ if (caractere === "(") {
183
+ profundidade += 1;
184
+ } else if (caractere === ")") {
185
+ profundidade -= 1;
186
+ if (profundidade === 0 && indice < atual.length - 1) {
187
+ removeu = false;
188
+ break;
189
+ }
190
+ }
191
+ }
192
+
193
+ if (!removeu) {
194
+ break;
195
+ }
196
+ atual = atual.slice(1, -1).trim();
197
+ }
198
+
199
+ return atual;
200
+ }
201
+
202
+ function dividirNoNivelRaiz(texto: string, operador: " e " | " ou "): string[] {
203
+ const partes: string[] = [];
204
+ let profundidade = 0;
205
+ let inicio = 0;
206
+
207
+ for (let indice = 0; indice < texto.length; indice += 1) {
208
+ const caractere = texto[indice]!;
209
+ if (caractere === "(") {
210
+ profundidade += 1;
211
+ continue;
212
+ }
213
+ if (caractere === ")") {
214
+ profundidade -= 1;
215
+ continue;
216
+ }
217
+ if (profundidade === 0 && texto.slice(indice, indice + operador.length) === operador) {
218
+ partes.push(texto.slice(inicio, indice).trim());
219
+ inicio = indice + operador.length;
220
+ indice += operador.length - 1;
221
+ }
222
+ }
223
+
224
+ const ultimaParte = texto.slice(inicio).trim();
225
+ if (ultimaParte) {
226
+ partes.push(ultimaParte);
227
+ }
228
+
229
+ return partes;
230
+ }
231
+
232
+ function criarComparacaoDerivada(
233
+ base: ExpressaoComparacaoSemantica,
234
+ valor: string,
235
+ ): ExpressaoComparacaoSemantica | undefined {
236
+ const valorNormalizado = removerParentesesExternos(valor.trim());
237
+ if (!valorNormalizado) {
238
+ return undefined;
239
+ }
240
+
241
+ return {
242
+ tipo: "comparacao",
243
+ textoOriginal: `${base.alvo} ${base.operador} ${valorNormalizado}`,
244
+ alvo: base.alvo,
245
+ operador: base.operador,
246
+ valor: valorNormalizado,
247
+ valorLiteral: ehValorLiteralSemantico(valorNormalizado),
248
+ };
249
+ }
250
+
251
+ export function parsearExpressaoSemantica(texto: string): ExpressaoSemantica | undefined {
252
+ const normalizado = removerParentesesExternos(texto.trim());
253
+ if (!normalizado) {
254
+ return undefined;
255
+ }
256
+
257
+ const partesOu = dividirNoNivelRaiz(normalizado, " ou ");
258
+ if (partesOu.length > 1) {
259
+ const termos: ExpressaoSemantica[] = [];
260
+ let comparacaoBase: ExpressaoComparacaoSemantica | undefined;
261
+ let usouAtalhoComparacao = false;
262
+
263
+ for (const parte of partesOu) {
264
+ const termo = parsearExpressaoSemantica(parte);
265
+ if (termo) {
266
+ termos.push(termo);
267
+ if (termo.tipo === "comparacao" && (termo.operador === "==" || termo.operador === "!=")) {
268
+ comparacaoBase = termo;
269
+ }
270
+ continue;
271
+ }
272
+
273
+ if (comparacaoBase) {
274
+ const derivada = criarComparacaoDerivada(comparacaoBase, parte);
275
+ if (derivada) {
276
+ comparacaoBase.valorLiteral = true;
277
+ usouAtalhoComparacao = true;
278
+ termos.push(derivada);
279
+ continue;
280
+ }
281
+ }
282
+
283
+ return undefined;
284
+ }
285
+
286
+ if (usouAtalhoComparacao) {
287
+ for (const termo of termos) {
288
+ if (termo.tipo === "comparacao" && termo.alvo === comparacaoBase?.alvo && termo.operador === comparacaoBase?.operador) {
289
+ termo.valorLiteral = true;
290
+ }
291
+ }
292
+ }
293
+
294
+ return { tipo: "composta", textoOriginal: normalizado, operadorLogico: "ou", termos };
295
+ }
296
+
297
+ const partesE = dividirNoNivelRaiz(normalizado, " e ");
298
+ if (partesE.length > 1) {
299
+ const termos = partesE.map((parte) => parsearExpressaoSemantica(parte)).filter((parte): parte is ExpressaoSemantica => Boolean(parte));
300
+ return termos.length === partesE.length
301
+ ? { tipo: "composta", textoOriginal: normalizado, operadorLogico: "e", termos }
302
+ : undefined;
303
+ }
304
+
305
+ if (normalizado.startsWith("nao ")) {
306
+ const termo = parsearExpressaoSemantica(normalizado.slice("nao ".length).trim());
307
+ return termo
308
+ ? { tipo: "negacao", textoOriginal: normalizado, termo }
309
+ : undefined;
310
+ }
311
+
312
+ const correspondenciaMarcador = normalizado.match(/^(persistencia|estado|sucesso)\s+([A-Za-z_][A-Za-z0-9_]*)$/);
313
+ if (correspondenciaMarcador) {
314
+ return {
315
+ tipo: "predicado",
316
+ textoOriginal: normalizado,
317
+ alvo: correspondenciaMarcador[1]!,
318
+ predicado: correspondenciaMarcador[2]!,
319
+ };
320
+ }
321
+
322
+ const correspondenciaExiste = normalizado.match(/^([A-Za-z_][A-Za-z0-9_.]*)\s+existe$/);
323
+ if (correspondenciaExiste) {
324
+ return {
325
+ tipo: "existe",
326
+ textoOriginal: normalizado,
327
+ alvo: correspondenciaExiste[1]!,
328
+ };
329
+ }
330
+
331
+ const correspondenciaPredicado = normalizado.match(/^([A-Za-z_][A-Za-z0-9_.]*)\s+deve_ser\s+([A-Za-z_][A-Za-z0-9_]*)(?:\s+(.+))?$/);
332
+ if (correspondenciaPredicado) {
333
+ return {
334
+ tipo: "predicado",
335
+ textoOriginal: normalizado,
336
+ alvo: correspondenciaPredicado[1]!,
337
+ predicado: correspondenciaPredicado[2]!,
338
+ argumentos: correspondenciaPredicado[3]?.trim(),
339
+ };
340
+ }
341
+
342
+ const correspondenciaPertencimento = normalizado.match(/^([A-Za-z_][A-Za-z0-9_.]*)\s+em\s+\[(.+)\]$/);
343
+ if (correspondenciaPertencimento) {
344
+ const valores = correspondenciaPertencimento[2]!
345
+ .split(",")
346
+ .map((valor) => valor.trim())
347
+ .filter(Boolean);
348
+ if (valores.length === 0) {
349
+ return undefined;
350
+ }
351
+ return {
352
+ tipo: "pertencimento",
353
+ textoOriginal: normalizado,
354
+ alvo: correspondenciaPertencimento[1]!,
355
+ valores,
356
+ };
357
+ }
358
+
359
+ const partes = normalizado.split(/\s+/).filter(Boolean);
360
+ if (partes.length >= 3 && OPERADORES_COMPARACAO.has(partes[1]!)) {
361
+ const valorComparacao = partes.slice(2).join(" ");
362
+ return {
363
+ tipo: "comparacao",
364
+ textoOriginal: normalizado,
365
+ alvo: partes[0]!,
366
+ operador: partes[1] as ExpressaoComparacaoSemantica["operador"],
367
+ valor: valorComparacao,
368
+ valorLiteral: ehValorLiteralSemantico(valorComparacao),
369
+ };
370
+ }
371
+
372
+ return undefined;
373
+ }
374
+
375
+ export function parsearEfeitoSemantico(texto: string): EfeitoSemantico | undefined {
376
+ const normalizado = texto.trim();
377
+ if (!normalizado) {
378
+ return undefined;
379
+ }
380
+
381
+ const partes = normalizado.split(/\s+/).filter(Boolean);
382
+ if (partes.length < 2) {
383
+ return undefined;
384
+ }
385
+
386
+ const partesSemCriticidade = [...partes];
387
+ let criticidadeTexto: string | undefined;
388
+ let privilegioTexto: string | undefined;
389
+ let isolamentoTexto: string | undefined;
390
+
391
+ const extrairQualificador = (nome: string): string | undefined => {
392
+ const indiceInline = partesSemCriticidade.findIndex((parte) => parte.startsWith(`${nome}=`));
393
+ if (indiceInline !== -1) {
394
+ const valor = partesSemCriticidade[indiceInline]!.slice(`${nome}=`.length).trim();
395
+ partesSemCriticidade.splice(indiceInline, 1);
396
+ return valor;
397
+ }
398
+
399
+ const indiceSeparado = partesSemCriticidade.findIndex((parte) => parte === nome);
400
+ if (
401
+ indiceSeparado !== -1
402
+ && partesSemCriticidade[indiceSeparado + 1] === "="
403
+ && partesSemCriticidade[indiceSeparado + 2]
404
+ ) {
405
+ const valor = partesSemCriticidade[indiceSeparado + 2]!.trim();
406
+ partesSemCriticidade.splice(indiceSeparado, 3);
407
+ return valor;
408
+ }
409
+
410
+ return undefined;
411
+ };
412
+
413
+ criticidadeTexto = extrairQualificador("criticidade");
414
+ privilegioTexto = extrairQualificador("privilegio");
415
+ isolamentoTexto = extrairQualificador("isolamento");
416
+
417
+ if (partesSemCriticidade.length < 2) {
418
+ return undefined;
419
+ }
420
+
421
+ const categoriaNormalizada = partesSemCriticidade[0] as CategoriaEfeitoSemantico;
422
+ const criticidade = criticidadeTexto && CRITICIDADES_EFEITO.has(criticidadeTexto as CriticidadeEfeitoSemantico)
423
+ ? criticidadeTexto as CriticidadeEfeitoSemantico
424
+ : undefined;
425
+ const privilegio = privilegioTexto && PRIVILEGIOS_EFEITO.has(privilegioTexto as PrivilegioEfeitoSemantico)
426
+ ? privilegioTexto as PrivilegioEfeitoSemantico
427
+ : undefined;
428
+ const isolamento = isolamentoTexto && ISOLAMENTOS_EFEITO.has(isolamentoTexto as IsolamentoEfeitoSemantico)
429
+ ? isolamentoTexto as IsolamentoEfeitoSemantico
430
+ : undefined;
431
+ if (CATEGORIAS_EFEITO.has(categoriaNormalizada)) {
432
+ return {
433
+ textoOriginal: normalizado,
434
+ categoria: categoriaNormalizada,
435
+ alvo: partesSemCriticidade[1]!,
436
+ detalhe: partesSemCriticidade.slice(2).join(" ").trim() || undefined,
437
+ criticidade,
438
+ criticidadeTexto,
439
+ privilegio,
440
+ privilegioTexto,
441
+ isolamento,
442
+ isolamentoTexto,
443
+ };
444
+ }
445
+
446
+ const categoriaLegada = MAPEAMENTO_EFEITOS_LEGADOS[partesSemCriticidade[0]!.toLowerCase()];
447
+ if (categoriaLegada) {
448
+ return {
449
+ textoOriginal: normalizado,
450
+ categoria: categoriaLegada,
451
+ alvo: partesSemCriticidade[1]!,
452
+ detalhe: partesSemCriticidade.slice(2).join(" ").trim() || undefined,
453
+ criticidade,
454
+ criticidadeTexto,
455
+ privilegio,
456
+ privilegioTexto,
457
+ isolamento,
458
+ isolamentoTexto,
459
+ };
460
+ }
461
+
462
+ return {
463
+ textoOriginal: normalizado,
464
+ categoria: partesSemCriticidade[0]! as CategoriaEfeitoSemantico,
465
+ alvo: partesSemCriticidade[1]!,
466
+ detalhe: partesSemCriticidade.slice(2).join(" ").trim() || undefined,
467
+ criticidade,
468
+ criticidadeTexto,
469
+ privilegio,
470
+ privilegioTexto,
471
+ isolamento,
472
+ isolamentoTexto,
473
+ };
474
+ }
475
+
476
+ export function parsearTransicaoEstado(texto: string): TransicaoEstadoSemantica | undefined {
477
+ const normalizado = texto.trim();
478
+ if (!normalizado) {
479
+ return undefined;
480
+ }
481
+
482
+ const correspondencia = normalizado.match(/^([A-Za-z_][A-Za-z0-9_.]*)\s*->\s*([A-Za-z_][A-Za-z0-9_.]*)$/);
483
+ if (!correspondencia) {
484
+ return undefined;
485
+ }
486
+
487
+ return {
488
+ textoOriginal: normalizado,
489
+ origem: correspondencia[1]!,
490
+ destino: correspondencia[2]!,
491
+ };
492
+ }
493
+
494
+ export function parsearEtapaFlow(texto: string): EtapaFlowSemantica | undefined {
495
+ const normalizado = texto.trim();
496
+ if (!normalizado.startsWith("etapa ")) {
497
+ return undefined;
498
+ }
499
+
500
+ const semPrefixo = normalizado.slice("etapa ".length).trim();
501
+ const nome = semPrefixo.split(/\s+/)[0];
502
+ if (!nome) {
503
+ return undefined;
504
+ }
505
+
506
+ const resto = semPrefixo.slice(nome.length).trim();
507
+ const task = resto.match(/\busa\s+([A-Za-z_][A-Za-z0-9_.]*)/)?.[1];
508
+ const comTexto = resto.match(/\bcom\s+(.+?)(?=\s+(quando|depende_de|em_sucesso|em_erro|por_erro)\b|$)/)?.[1];
509
+ const dependenciasTexto = resto.match(/\bdepende_de\s+(.+?)(?=\s+(quando|com|em_sucesso|em_erro|por_erro)\b|$)/)?.[1];
510
+ const dependencias = dependenciasTexto
511
+ ? dependenciasTexto.split(",").map((parte) => parte.trim()).filter(Boolean)
512
+ : [];
513
+ const emSucesso = resto.match(/\bem_sucesso\s+([A-Za-z_][A-Za-z0-9_]*)/)?.[1];
514
+ const emErro = resto.match(/\bem_erro\s+([A-Za-z_][A-Za-z0-9_]*)/)?.[1];
515
+ const porErroTexto = resto.match(/\bpor_erro\s+(.+?)(?=\s+(quando|depende_de|em_sucesso|em_erro)\b|$)/)?.[1];
516
+ const mapeamentos = (comTexto
517
+ ? comTexto.split(",").map((parte) => parte.trim()).filter(Boolean)
518
+ : [])
519
+ .map((parte) => {
520
+ const [campo, ...restoValor] = parte.split("=");
521
+ return {
522
+ campo: campo?.trim() ?? "",
523
+ valor: restoValor.join("=").trim(),
524
+ };
525
+ })
526
+ .filter((item) => item.campo && item.valor);
527
+ const porErro = (porErroTexto
528
+ ? porErroTexto.split(",").map((parte) => parte.trim()).filter(Boolean)
529
+ : [])
530
+ .map((parte) => {
531
+ const [tipo, ...restoDestino] = parte.split("=");
532
+ return {
533
+ tipo: tipo?.trim() ?? "",
534
+ destino: restoDestino.join("=").trim(),
535
+ };
536
+ })
537
+ .filter((item) => item.tipo && item.destino);
538
+
539
+ const indiceQuando = resto.indexOf(" quando ");
540
+ const indicesTerminoCondicao = [
541
+ resto.indexOf(" depende_de "),
542
+ resto.indexOf(" em_sucesso "),
543
+ resto.indexOf(" em_erro "),
544
+ resto.indexOf(" por_erro "),
545
+ ].filter((indice) => indice !== -1 && indice > indiceQuando);
546
+ let condicao: ExpressaoSemantica | undefined;
547
+ if (indiceQuando !== -1) {
548
+ const fimCondicao = indicesTerminoCondicao.length > 0 ? Math.min(...indicesTerminoCondicao) : resto.length;
549
+ const textoCondicao = resto.slice(indiceQuando + " quando ".length, fimCondicao).trim();
550
+ condicao = parsearExpressaoSemantica(textoCondicao);
551
+ }
552
+
553
+ return {
554
+ textoOriginal: normalizado,
555
+ nome,
556
+ task,
557
+ condicao,
558
+ dependencias,
559
+ mapeamentos,
560
+ emSucesso,
561
+ emErro,
562
+ porErro,
563
+ };
564
+ }
565
+
566
+ export function extrairReferenciasDaExpressao(expressao: ExpressaoSemantica): string[] {
567
+ if (expressao.tipo === "composta") {
568
+ return expressao.termos.flatMap(extrairReferenciasDaExpressao);
569
+ }
570
+
571
+ if (expressao.tipo === "negacao") {
572
+ return extrairReferenciasDaExpressao(expressao.termo);
573
+ }
574
+
575
+ const referencias = [expressao.alvo];
576
+
577
+ if (
578
+ expressao.tipo === "comparacao"
579
+ && expressao.valor
580
+ && !expressao.valorLiteral
581
+ && pareceReferenciaSemantica(expressao.valor)
582
+ ) {
583
+ referencias.push(expressao.valor);
584
+ }
585
+
586
+ return referencias;
587
+ }
588
+
589
+ export function pareceReferenciaSemantica(valor: string): boolean {
590
+ const normalizado = valor.trim();
591
+ if (!normalizado) {
592
+ return false;
593
+ }
594
+
595
+ if (ehValorLiteralSemantico(normalizado)) {
596
+ return false;
597
+ }
598
+
599
+ return /^[A-Za-z_][A-Za-z0-9_.]*$/.test(normalizado);
600
+ }
601
+
602
+ function ehValorLiteralSemantico(valor: string): boolean {
603
+ const normalizado = valor.trim();
604
+ if (!normalizado) {
605
+ return false;
606
+ }
607
+
608
+ if (
609
+ (normalizado.startsWith("\"") && normalizado.endsWith("\""))
610
+ || (normalizado.startsWith("'") && normalizado.endsWith("'"))
611
+ ) {
612
+ return true;
613
+ }
614
+
615
+ if (/^-?\d+(?:\.\d+)?$/.test(normalizado)) {
616
+ return true;
617
+ }
618
+
619
+ if (["verdadeiro", "falso", "nulo"].includes(normalizado)) {
620
+ return true;
621
+ }
622
+
623
+ return /^[A-Z][A-Z0-9_]*$/.test(normalizado);
624
+ }
625
+
626
+ export function ehCategoriaEfeitoSemantico(valor: string): valor is CategoriaEfeitoSemantico {
627
+ return CATEGORIAS_EFEITO.has(valor as CategoriaEfeitoSemantico);
628
+ }
629
+
630
+ export function ehCriticidadeEfeitoSemantico(valor: string): valor is CriticidadeEfeitoSemantico {
631
+ return CRITICIDADES_EFEITO.has(valor as CriticidadeEfeitoSemantico);
632
+ }