feat: alinea copy A3/A4 a activar y añade tests; actualiza env y docs
Co-authored-by: aider (openrouter/openai/gpt-5) <aider@aider.chat>webui
parent
b9252f4c49
commit
07bfa0f419
@ -0,0 +1,51 @@
|
||||
import { describe, it, expect, beforeAll, beforeEach } from 'bun:test';
|
||||
import { Database } from 'bun:sqlite';
|
||||
import { initializeDatabase } from '../../src/db';
|
||||
import { WebhookServer } from '../../src/server';
|
||||
import { ResponseQueue } from '../../src/services/response-queue';
|
||||
|
||||
describe('WebhookServer - DM "activar" (A4)', () => {
|
||||
let memdb: Database;
|
||||
|
||||
beforeAll(() => {
|
||||
memdb = new Database(':memory:');
|
||||
initializeDatabase(memdb);
|
||||
(WebhookServer as any).dbInstance = memdb;
|
||||
(ResponseQueue as any).dbInstance = memdb;
|
||||
});
|
||||
|
||||
beforeEach(() => {
|
||||
process.env.NODE_ENV = 'test';
|
||||
memdb.exec('DELETE FROM response_queue');
|
||||
memdb.exec('DELETE FROM users');
|
||||
});
|
||||
|
||||
function rowCount(): number {
|
||||
const r = memdb.query("SELECT COUNT(*) AS c FROM response_queue").get() as any;
|
||||
return Number(r?.c || 0);
|
||||
}
|
||||
|
||||
it('al recibir "activar" por DM, asegura usuario y encola confirmación', async () => {
|
||||
const data = {
|
||||
key: { remoteJid: '7001@s.whatsapp.net', fromMe: false },
|
||||
message: { conversation: 'activar' }
|
||||
};
|
||||
await WebhookServer.handleMessageUpsert(data);
|
||||
|
||||
expect(rowCount()).toBe(1);
|
||||
const row = memdb.query("SELECT recipient, message FROM response_queue ORDER BY id").get() as any;
|
||||
expect(row.recipient).toBe('7001');
|
||||
expect(String(row.message).toLowerCase()).toContain('listo');
|
||||
});
|
||||
|
||||
it('es idempotente: si envía "activar" de nuevo, se vuelve a encolar', async () => {
|
||||
const data = {
|
||||
key: { remoteJid: '8002@s.whatsapp.net', fromMe: false },
|
||||
message: { conversation: 'activar' }
|
||||
};
|
||||
await WebhookServer.handleMessageUpsert(data);
|
||||
await WebhookServer.handleMessageUpsert(data);
|
||||
|
||||
expect(rowCount()).toBe(2);
|
||||
});
|
||||
});
|
||||
@ -0,0 +1,73 @@
|
||||
import { describe, it, expect, beforeAll, beforeEach } from 'bun:test';
|
||||
import { Database } from 'bun:sqlite';
|
||||
import { initializeDatabase } from '../../../src/db';
|
||||
import { TaskService } from '../../../src/tasks/service';
|
||||
import { CommandService } from '../../../src/services/command';
|
||||
import { Metrics } from '../../../src/services/metrics';
|
||||
|
||||
describe('CommandService - A4 JIT DM al asignador', () => {
|
||||
let memdb: Database;
|
||||
|
||||
beforeAll(() => {
|
||||
memdb = new Database(':memory:');
|
||||
initializeDatabase(memdb);
|
||||
TaskService.dbInstance = memdb as any;
|
||||
CommandService.dbInstance = memdb as any;
|
||||
});
|
||||
|
||||
beforeEach(() => {
|
||||
process.env.NODE_ENV = 'test';
|
||||
process.env.METRICS_ENABLED = 'true';
|
||||
process.env.ONBOARDING_ENABLE_IN_TEST = 'true';
|
||||
process.env.CHATBOT_PHONE_NUMBER = '555111222';
|
||||
Metrics.reset();
|
||||
|
||||
memdb.exec(`
|
||||
DELETE FROM task_assignments;
|
||||
DELETE FROM tasks;
|
||||
DELETE FROM users;
|
||||
`);
|
||||
});
|
||||
|
||||
it('envía un único DM JIT al asignador con la lista y enlace wa.me cuando hay tokens no resolubles', async () => {
|
||||
const res = await CommandService.handle({
|
||||
sender: '111',
|
||||
groupId: '', // DM
|
||||
message: '/t n Mixta @34600123456 @lid-opaque',
|
||||
mentions: [],
|
||||
});
|
||||
|
||||
// Debe existir al menos el ACK y el JIT
|
||||
expect(res.length).toBeGreaterThan(0);
|
||||
|
||||
const toSender = res.filter(r => r.recipient === '111');
|
||||
expect(toSender.length).toBeGreaterThan(0);
|
||||
|
||||
const jit = toSender.find(r => r.message.includes('activar'));
|
||||
expect(jit).toBeTruthy();
|
||||
expect(jit!.message).toContain('lid-opaque');
|
||||
expect(jit!.message).toContain('https://wa.me/555111222');
|
||||
|
||||
const prom = Metrics.render('prom');
|
||||
expect(prom).toContain('onboarding_prompts_sent_total');
|
||||
expect(prom).toContain('source="jit_assignee_failure"');
|
||||
});
|
||||
|
||||
it('no envía JIT si falta CHATBOT_PHONE_NUMBER y contabiliza skipped:missing_bot_number', async () => {
|
||||
process.env.CHATBOT_PHONE_NUMBER = '';
|
||||
|
||||
const res = await CommandService.handle({
|
||||
sender: '222',
|
||||
groupId: '',
|
||||
message: '/t n Solo opaco @alias-xyz',
|
||||
mentions: [],
|
||||
});
|
||||
|
||||
const anyJit = res.find(r => r.recipient === '222' && r.message.includes('activar'));
|
||||
expect(anyJit).toBeUndefined();
|
||||
|
||||
const prom = Metrics.render('prom');
|
||||
expect(prom).toContain('onboarding_prompts_skipped_total');
|
||||
expect(prom).toContain('reason="missing_bot_number"');
|
||||
});
|
||||
});
|
||||
Loading…
Reference in New Issue