Update chat events/transcription flow and container startup fixes

This commit is contained in:
Ruslan Bakiev
2026-02-19 12:54:16 +07:00
parent 7cc86579b2
commit 3ac487c25b
27 changed files with 3888 additions and 780 deletions

View File

@@ -3,7 +3,7 @@ generator client {
}
datasource db {
provider = "sqlite"
provider = "postgresql"
url = env("DATABASE_URL")
}
@@ -263,22 +263,43 @@ model CalendarEvent {
}
model Deal {
id String @id @default(cuid())
teamId String
contactId String
title String
stage String
amount Int?
nextStep String?
summary String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
id String @id @default(cuid())
teamId String
contactId String
title String
stage String
amount Int?
nextStep String?
summary String?
currentStepId String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
team Team @relation(fields: [teamId], references: [id], onDelete: Cascade)
contact Contact @relation(fields: [contactId], references: [id], onDelete: Cascade)
team Team @relation(fields: [teamId], references: [id], onDelete: Cascade)
contact Contact @relation(fields: [contactId], references: [id], onDelete: Cascade)
steps DealStep[]
@@index([teamId, updatedAt])
@@index([contactId, updatedAt])
@@index([currentStepId])
}
model DealStep {
id String @id @default(cuid())
dealId String
title String
description String?
status String @default("todo")
dueAt DateTime?
order Int @default(0)
completedAt DateTime?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
deal Deal @relation(fields: [dealId], references: [id], onDelete: Cascade)
@@index([dealId, order])
@@index([status, dueAt])
}
model ChatConversation {

View File

@@ -32,7 +32,7 @@ const prisma = new PrismaClient();
const LOGIN_PHONE = "+15550000001";
const LOGIN_PASSWORD = "ConnectFlow#2026";
const LOGIN_NAME = "Connect Owner";
const LOGIN_NAME = "Владелец Connect";
const REF_DATE_ISO = "2026-02-20T12:00:00.000Z";
const SCRYPT_KEY_LENGTH = 64;
@@ -58,26 +58,26 @@ function plusMinutes(date, minutes) {
function buildOdooAiContacts(teamId) {
const prospects = [
{ name: "Olivia Reed", company: "RetailNova", country: "USA", location: "New York", email: "olivia.reed@retailnova.com", phone: "+1 555 120 0101" },
{ name: "Daniel Kim", company: "ForgePeak Manufacturing", country: "USA", location: "Chicago", email: "daniel.kim@forgepeak.com", phone: "+1 555 120 0102" },
{ name: "Marta Alonso", company: "Iberia Foods Group", country: "Spain", location: "Barcelona", email: "marta.alonso@iberiafoods.es", phone: "+34 91 555 0103" },
{ name: "Youssef Haddad", company: "GulfTrade Distribution", country: "UAE", location: "Dubai", email: "youssef.haddad@gulftrade.ae", phone: "+971 4 555 0104" },
{ name: "Emma Collins", company: "NorthBridge Logistics", country: "UK", location: "London", email: "emma.collins@northbridge.co.uk", phone: "+44 20 5550 0105" },
{ name: "Noah Fischer", company: "Bergmann Auto Parts", country: "Germany", location: "Munich", email: "noah.fischer@bergmann-auto.de", phone: "+49 89 5550 0106" },
{ name: "Ava Choi", company: "Pacific MedTech Supply", country: "Singapore", location: "Singapore", email: "ava.choi@pacificmedtech.sg", phone: "+65 6555 0107" },
{ name: "Liam Dubois", company: "HexaCommerce", country: "France", location: "Paris", email: "liam.dubois@hexacommerce.fr", phone: "+33 1 55 50 0108" },
{ name: "Maya Shah", company: "Zenith Consumer Brands", country: "Canada", location: "Toronto", email: "maya.shah@zenithbrands.ca", phone: "+1 416 555 0109" },
{ name: "Arman Petrosyan", company: "Ararat Electronics", country: "Armenia", location: "Yerevan", email: "arman.petrosyan@ararat-electronics.am", phone: "+374 10 555110" },
{ name: "Sophia Martinez", company: "Sunline Home Goods", country: "USA", location: "Austin", email: "sophia.martinez@sunlinehg.com", phone: "+1 555 120 0111" },
{ name: "Leo Novak", company: "CentralBuild Materials", country: "Germany", location: "Berlin", email: "leo.novak@centralbuild.de", phone: "+49 30 5550 0112" },
{ name: "Isla Grant", company: "BlueHarbor Pharma", country: "UK", location: "Manchester", email: "isla.grant@blueharbor.co.uk", phone: "+44 161 555 0113" },
{ name: "Mateo Rossi", company: "Milano Fashion House", country: "Italy", location: "Milan", email: "mateo.rossi@milanofh.it", phone: "+39 02 5550 0114" },
{ name: "Nina Volkova", company: "Polar AgriTech", country: "Kazakhstan", location: "Almaty", email: "nina.volkova@polaragri.kz", phone: "+7 727 555 0115" },
{ name: "Ethan Park", company: "Vertex Components", country: "South Korea", location: "Seoul", email: "ethan.park@vertexcomponents.kr", phone: "+82 2 555 0116" },
{ name: "Zara Khan", company: "Crescent Retail Chain", country: "UAE", location: "Abu Dhabi", email: "zara.khan@crescentretail.ae", phone: "+971 2 555 0117" },
{ name: "Hugo Silva", company: "Luso Industrial Systems", country: "Portugal", location: "Lisbon", email: "hugo.silva@lusois.pt", phone: "+351 21 555 0118" },
{ name: "Chloe Bernard", company: "Santex Clinics Network", country: "France", location: "Lyon", email: "chloe.bernard@santex.fr", phone: "+33 4 55 50 0119" },
{ name: "James Walker", company: "Metro Wholesale Group", country: "USA", location: "Los Angeles", email: "james.walker@metrowholesale.com", phone: "+1 555 120 0120" },
{ name: "Оливия Рид", company: "РитейлНова", country: "США", location: "Нью-Йорк", email: "olivia.reed@retailnova.com", phone: "+1 555 120 0101" },
{ name: "Даниэль Ким", company: "ФорджПик Производство", country: "США", location: "Чикаго", email: "daniel.kim@forgepeak.com", phone: "+1 555 120 0102" },
{ name: "Марта Алонсо", company: "Иберия Фудс Групп", country: "Испания", location: "Барселона", email: "marta.alonso@iberiafoods.es", phone: "+34 91 555 0103" },
{ name: "Юсеф Хаддад", company: "ГалфТрейд Дистрибуция", country: "ОАЭ", location: "Дубай", email: "youssef.haddad@gulftrade.ae", phone: "+971 4 555 0104" },
{ name: "Эмма Коллинз", company: "НортБридж Логистика", country: "Великобритания", location: "Лондон", email: "emma.collins@northbridge.co.uk", phone: "+44 20 5550 0105" },
{ name: "Ноа Фишер", company: "Бергман Автозапчасти", country: "Германия", location: "Мюнхен", email: "noah.fischer@bergmann-auto.de", phone: "+49 89 5550 0106" },
{ name: "Ава Чой", company: "Пасифик МедТех Сапплай", country: "Сингапур", location: "Сингапур", email: "ava.choi@pacificmedtech.sg", phone: "+65 6555 0107" },
{ name: "Лиам Дюбуа", company: "ГексаКоммерс", country: "Франция", location: "Париж", email: "liam.dubois@hexacommerce.fr", phone: "+33 1 55 50 0108" },
{ name: "Майя Шах", company: "Зенит Консьюмер Брендс", country: "Канада", location: "Торонто", email: "maya.shah@zenithbrands.ca", phone: "+1 416 555 0109" },
{ name: "Арман Петросян", company: "Арарат Электроникс", country: "Армения", location: "Ереван", email: "arman.petrosyan@ararat-electronics.am", phone: "+374 10 555110" },
{ name: "София Мартинес", company: "Санлайн Товары для дома", country: "США", location: "Остин", email: "sophia.martinez@sunlinehg.com", phone: "+1 555 120 0111" },
{ name: "Лео Новак", company: "ЦентралБилд Материалы", country: "Германия", location: "Берлин", email: "leo.novak@centralbuild.de", phone: "+49 30 5550 0112" },
{ name: "Айла Грант", company: "БлюХарбор Фарма", country: "Великобритания", location: "Манчестер", email: "isla.grant@blueharbor.co.uk", phone: "+44 161 555 0113" },
{ name: "Матео Росси", company: "Милано Фэшн Хаус", country: "Италия", location: "Милан", email: "mateo.rossi@milanofh.it", phone: "+39 02 5550 0114" },
{ name: "Нина Волкова", company: "Полар АгриТех", country: "Казахстан", location: "Алматы", email: "nina.volkova@polaragri.kz", phone: "+7 727 555 0115" },
{ name: "Итан Пак", company: "Вертекс Компонентс", country: "Южная Корея", location: "Сеул", email: "ethan.park@vertexcomponents.kr", phone: "+82 2 555 0116" },
{ name: "Зара Хан", company: "Кресент Ритейл Чейн", country: "ОАЭ", location: "Абу-Даби", email: "zara.khan@crescentretail.ae", phone: "+971 2 555 0117" },
{ name: "Уго Силва", company: "Лузо Индастриал Системс", country: "Португалия", location: "Лиссабон", email: "hugo.silva@lusois.pt", phone: "+351 21 555 0118" },
{ name: "Хлоя Бернар", company: "Сантекс Сеть Клиник", country: "Франция", location: "Лион", email: "chloe.bernard@santex.fr", phone: "+33 4 55 50 0119" },
{ name: "Джеймс Уокер", company: "Метро Оптовая Группа", country: "США", location: "Лос-Анджелес", email: "james.walker@metrowholesale.com", phone: "+1 555 120 0120" },
];
return prospects.map((p, idx) => {
@@ -113,8 +113,8 @@ async function main() {
const team = await prisma.team.upsert({
where: { id: "demo-team" },
update: { name: "Connect Workspace" },
create: { id: "demo-team", name: "Connect Workspace" },
update: { name: "Connect Рабочее пространство" },
create: { id: "demo-team", name: "Connect Рабочее пространство" },
});
await prisma.teamMember.upsert({
@@ -125,8 +125,8 @@ async function main() {
const conversation = await prisma.chatConversation.upsert({
where: { id: `pilot-${team.id}` },
update: { title: "Pilot" },
create: { id: `pilot-${team.id}`, teamId: team.id, createdByUserId: user.id, title: "Pilot" },
update: { title: "Пилот" },
create: { id: `pilot-${team.id}`, teamId: team.id, createdByUserId: user.id, title: "Пилот" },
});
await prisma.$transaction([
@@ -150,22 +150,22 @@ async function main() {
});
const integrationModules = [
"Sales + CRM + forecasting copilot",
"Inventory + demand prediction",
"Purchase + supplier risk scoring",
"Accounting + AI anomaly detection",
"Helpdesk + ticket triage assistant",
"Manufacturing + production planning AI",
"Продажи + CRM + копилот прогнозирования",
"Склад + прогноз спроса",
"Закупки + оценка рисков поставщиков",
"Бухгалтерия + AI-детекция аномалий",
"Поддержка + ассистент триажа заявок",
"Производство + AI-планирование мощностей",
];
await prisma.contactNote.createMany({
data: contacts.map((c, idx) => ({
contactId: c.id,
content:
`${c.company ?? c.name} is evaluating Odoo implementation with AI extensions. ` +
`Primary integration scope: ${integrationModules[idx % integrationModules.length]}. ` +
`Main buying trigger: reduce manual operations and shorten decision cycles. ` +
`Next milestone: run discovery workshop, confirm data owners, and approve pilot KPI pack.`,
`${c.company ?? c.name} рассматривает внедрение Odoo с AI-расширениями. ` +
`Основной контур интеграции: ${integrationModules[idx % integrationModules.length]}. ` +
`Ключевой драйвер покупки: сократить ручные операции и ускорить цикл принятия решений. ` +
`Следующая веха: провести сессию уточнения, согласовать владельцев данных и утвердить KPI пилота.`,
})),
});
@@ -180,7 +180,7 @@ async function main() {
kind: "MESSAGE",
direction: "IN",
channel: channels[i % channels.length],
content: `Hi, we are reviewing Odoo + AI rollout for ${contact.company}. Can we align on integration timeline this week?`,
content: `Здравствуйте! Мы рассматриваем запуск Odoo + AI для ${contact.company}. Можем согласовать план интеграции на этой неделе?`,
occurredAt: base,
});
@@ -189,7 +189,7 @@ async function main() {
kind: "MESSAGE",
direction: "OUT",
channel: channels[(i + 1) % channels.length],
content: "Sure. I suggest a 45-min discovery focused on workflows, API constraints, and pilot KPIs.",
content: "Да, предлагаю 45-минутный разбор: процессы, ограничения API и KPI пилота.",
occurredAt: plusMinutes(base, 22),
});
@@ -198,7 +198,7 @@ async function main() {
kind: "MESSAGE",
direction: i % 3 === 0 ? "OUT" : "IN",
channel: channels[(i + 2) % channels.length],
content: "Status update: technical scope is clear; blocker is budget owner approval and security questionnaire.",
content: "Обновление статуса: технический объём ясен; блокер — согласование бюджета и анкета по безопасности.",
occurredAt: plusMinutes(base, 65),
});
@@ -208,11 +208,11 @@ async function main() {
kind: "CALL",
direction: "OUT",
channel: "PHONE",
content: "Discovery call: Odoo modules, data flows, AI use-cases",
content: "Созвон по уточнению: модули Odoo, потоки данных и AI-сценарии",
durationSec: 180 + ((i * 23) % 420),
transcriptJson: [
`${contact.name}: We need phased rollout, starting from Sales and Inventory.`,
"You: Agreed. We can run a 6-week pilot with KPI baseline and weekly checkpoints.",
`${contact.name}: Нам нужен поэтапный запуск, начнём с продаж и склада.`,
"Вы: Согласен. Делаем пилот на 6 недель с базовыми KPI и еженедельными контрольными точками.",
],
occurredAt: plusMinutes(base, 110),
});
@@ -222,47 +222,47 @@ async function main() {
await prisma.calendarEvent.createMany({
data: contacts.flatMap((c, idx) => {
// Historical week ending on 20 Feb 2026: all seeded meetings are completed.
// Историческая неделя до 20 Feb 2026: все сидовые встречи завершены.
const firstStart = atOffset(-6 + (idx % 5), 10 + (idx % 6), (idx * 5) % 60);
const secondStart = atOffset(-5 + (idx % 5), 14 + (idx % 4), (idx * 3) % 60);
return [
{
teamId: team.id,
contactId: c.id,
title: `Discovery: Odoo + AI with ${c.company ?? c.name}`,
title: `Сессия уточнения: Odoo + AI с ${c.company ?? c.name}`,
startsAt: firstStart,
endsAt: plusMinutes(firstStart, 30),
note: "Confirm integration scope, current stack, and pilot success metrics.",
note: "Подтвердить рамки интеграции, текущий стек и метрики успеха пилота.",
status: "done",
},
{
teamId: team.id,
contactId: c.id,
title: `Architecture workshop: ${c.company ?? c.name}`,
title: `Архитектурный воркшоп: ${c.company ?? c.name}`,
startsAt: secondStart,
endsAt: plusMinutes(secondStart, 45),
note: "Review API mapping, ETL boundaries, and AI assistant guardrails.",
note: "Проверить маппинг API, границы ETL и ограничения для AI-ассистента.",
status: "done",
},
];
}),
});
const stages = ["Lead", "Discovery", "Solution Fit", "Proposal", "Negotiation", "Pilot", "Contract Review"];
const stages = ["Лид", "Уточнение", "Подбор решения", "Коммерческое предложение", "Переговоры", "Пилот", "Проверка договора"];
await prisma.deal.createMany({
data: contacts.map((c, idx) => ({
teamId: team.id,
contactId: c.id,
title: `${c.company ?? "Account"} Odoo + AI integration`,
title: `${c.company ?? "Клиент"}: интеграция Odoo + AI`,
stage: stages[idx % stages.length],
amount: 18000 + (idx % 8) * 7000,
nextStep:
idx % 4 === 0
? "Send pilot proposal and finalize integration backlog."
: "Run solution workshop and align commercial owner on timeline.",
? "Отправить предложение по пилоту и зафиксировать список задач интеграции."
: "Провести воркшоп по решению и согласовать сроки с коммерческим владельцем.",
summary:
"Potential deal for phased Odoo implementation with AI copilots for ops, sales, and planning. " +
"Commercial model: discovery + pilot + rollout.",
"Потенциальная сделка на поэтапное внедрение Odoo с AI-копилотами для операций, продаж и планирования. " +
"Коммерческая модель: уточнение + пилот + тиражирование.",
})),
});
@@ -272,8 +272,8 @@ async function main() {
contactId: c.id,
text:
idx % 3 === 0
? "Pinned: ask for ERP owner, data owner, and target go-live quarter."
: "Pinned: keep communication around one KPI and one next action.",
? "Закреплено: уточнить владельца ERP, владельца данных и целевой квартал запуска."
: "Закреплено: держать коммуникацию вокруг одного KPI и следующего шага.",
})),
});
@@ -287,14 +287,14 @@ async function main() {
contactId: c.id,
happenedAt: atOffset(-(idx % 6), 9 + (idx % 8), (idx * 9) % 60),
text:
`I reviewed ${c.company ?? c.name} account activity for the Odoo + AI opportunity. ` +
"There is enough momentum to move the deal one stage with a concrete next action.",
`Я проверил активность по аккаунту ${c.company ?? c.name} в рамках сделки Odoo + AI. ` +
"Есть достаточный импульс, чтобы перевести сделку на следующий этап при чётком следующем шаге.",
proposalJson: {
title: idx % 2 === 0 ? "Schedule pilot scoping call" : "Send unblock note for budget owner",
title: idx % 2 === 0 ? "Назначить созвон по рамкам пилота" : "Отправить сообщение для разблокировки у владельца бюджета",
details: [
`Contact: ${c.name}`,
idx % 2 === 0 ? "Timing: this week, 45 minutes" : "Timing: today in primary channel",
"Goal: confirm scope, owner, and next commercial checkpoint",
`Контакт: ${c.name}`,
idx % 2 === 0 ? "Когда: на этой неделе, 45 минут" : "Когда: сегодня в основном канале",
"Цель: подтвердить объём, владельца и следующую коммерческую контрольную точку",
],
key: proposalKeys[idx % proposalKeys.length],
},
@@ -305,62 +305,62 @@ async function main() {
data: [
{
teamId: team.id,
title: "Odoo integration discovery checklist",
title: "Чеклист уточнения для интеграции Odoo",
type: "Regulation",
owner: "Solution Team",
scope: "Pre-sale discovery",
summary: "Mandatory questions before estimation of Odoo + AI rollout.",
body: "## Must capture\n- Current ERP modules\n- Integration endpoints\n- Data owner per domain\n- Security constraints\n- Pilot KPI baseline",
owner: "Команда решений",
scope: "Предпродажное уточнение",
summary: "Обязательные вопросы перед оценкой запуска Odoo + AI.",
body: "## Нужно зафиксировать\n- Текущие модули ERP\n- Точки интеграции\n- Владельца данных по каждому домену\n- Ограничения безопасности\n- Базовые KPI пилота",
updatedAt: atOffset(-1, 11, 10),
},
{
teamId: team.id,
title: "AI copilot playbook for Odoo",
title: "Плейбук AI-копилота для Odoo",
type: "Playbook",
owner: "AI Practice Lead",
scope: "Use-case qualification",
summary: "How to position forecasting, assistant, and anomaly detection features.",
body: "## Flow\n1. Process pain\n2. Data quality\n3. Model target\n4. Success KPI\n5. Pilot scope",
owner: "Лид AI-практики",
scope: "Квалификация сценариев",
summary: "Как позиционировать прогнозирование, ассистента и детекцию аномалий.",
body: "## Поток\n1. Боль процесса\n2. Качество данных\n3. Целевая модель\n4. KPI успеха\n5. Объём пилота",
updatedAt: atOffset(-2, 15, 0),
},
{
teamId: team.id,
title: "Pilot pricing matrix",
title: "Матрица цен для пилота",
type: "Policy",
owner: "Commercial Ops",
scope: "Discovery and pilot contracts",
summary: "Price ranges for discovery, pilot, and production rollout phases.",
body: "## Typical ranges\n- Discovery: 5k-12k\n- Pilot: 15k-45k\n- Rollout: 50k+\n\nAlways tie cost to scope and timeline.",
owner: "Коммерческие операции",
scope: "Контракты уточнения и пилота",
summary: "Диапазоны цен для уточнения, пилота и продуктивной фазы.",
body: "## Типовые диапазоны\n- Уточнение: 5k-12k\n- Пилот: 15k-45k\n- Тиражирование: 50k+\n\nВсегда привязывай стоимость к объёму и срокам.",
updatedAt: atOffset(-3, 9, 30),
},
{
teamId: team.id,
title: "Security and compliance template",
title: "Шаблон по безопасности и комплаенсу",
type: "Template",
owner: "Delivery Office",
scope: "Enterprise prospects",
summary: "Template answers for data residency, RBAC, audit trail, and PII handling.",
body: "## Sections\n- Hosting model\n- Access control\n- Logging and audit\n- Data retention\n- Incident response",
owner: "Офис внедрения",
scope: "Крупные клиенты",
summary: "Шаблон ответов по data residency, RBAC, аудиту и обработке PII.",
body: "## Разделы\n- Модель хостинга\n- Контроль доступа\n- Логирование и аудит\n- Срок хранения данных\n- Реакция на инциденты",
updatedAt: atOffset(-4, 13, 45),
},
{
teamId: team.id,
title: "Integration architecture blueprint",
title: "Референс интеграционной архитектуры",
type: "Playbook",
owner: "Architecture Team",
scope: "Technical workshops",
summary: "Reference architecture for Odoo connectors, ETL, and AI service layer.",
body: "## Layers\n- Odoo core modules\n- Integration bus\n- Data warehouse\n- AI service endpoints\n- Monitoring",
owner: "Архитектурная команда",
scope: "Технические воркшопы",
summary: "Референс-архитектура для коннекторов Odoo, ETL и AI-сервисного слоя.",
body: "## Слои\n- Базовые модули Odoo\n- Интеграционная шина\n- Хранилище данных\n- Эндпоинты AI-сервиса\n- Мониторинг",
updatedAt: atOffset(-5, 10, 0),
},
{
teamId: team.id,
title: "Go-live readiness checklist",
title: "Чеклист готовности к запуску",
type: "Regulation",
owner: "PMO",
scope: "Pilot to production transition",
summary: "Checklist to move from pilot acceptance to production launch.",
body: "## Required\n- Pilot KPIs approved\n- Rollout backlog prioritized\n- Owners assigned\n- Support model defined",
scope: "Переход от пилота к продакшену",
summary: "Чеклист перехода от приёмки пилота к запуску в прод.",
body: "## Обязательно\n- KPI пилота утверждены\n- Backlog тиражирования приоритизирован\n- Владельцы назначены\n- Модель поддержки определена",
updatedAt: atOffset(-6, 16, 15),
},
],