feat(backend): one-click setup script for 8 sheets in Google Sheet

This commit is contained in:
wasrusgen 2026-05-09 01:18:39 +03:00
parent 0c57f47ebf
commit 6b0b01e15e

93
backend/setup_database.gs Normal file
View File

@ -0,0 +1,93 @@
/**
* ЗОВ — База: одноразовый setup всех 8 листов с заголовками и формулами.
*
* Как запустить:
* 1. Открыть таблицу «ЗОВ — База» в Google Sheets.
* 2. Extensions (Расширения) → Apps Script.
* 3. В редакторе скопировать сюда содержимое этого файла.
* 4. Сохранить (Ctrl+S).
* 5. В верхней панели выбрать функцию `setupDatabase` → нажать Run.
* 6. Authorize при первом запуске (Google спросит разрешение на работу со Sheet).
* 7. Вернуться в таблицу — все 8 листов готовы.
*/
const SHEETS = {
Users: ["tg_id","tg_username","first_name","last_name","role","created_at","last_seen_at","invite_code_used"],
Managers: ["tg_id","full_name","email","phone","salon","city","is_zov_employee","status","last_order_date","active_until","total_leads","total_deals","conversion_rate","invite_code"],
Clients: ["tg_id","full_name","phone","email","address","city","budget_total","manager_tg_id","source","last_measurement_id"],
Measurements: ["id","created_at","client_tg_id","manager_tg_id","filled_by","layout","area_m2","ceiling_mm","walls_json","openings_json","infra_json","niches_json","photos_urls","notes","status"],
Leads: ["id","created_at","manager_tg_id","client_tg_id","client_name","measurement_id","checklist_json","ai_response","ai_model","ai_tokens_used","sent_to_tg","deal_status","deal_amount"],
Logs: ["timestamp","event","tg_id","payload"],
Settings: ["key","value","description"],
Dashboard: ["metric","value"],
};
const SETTINGS_DEFAULTS = [
["ACTIVE_PERIOD_DAYS", 90, "Дней active-статуса после сделки через куратора"],
["GRACE_PERIOD_DAYS", 14, "Grace-период перед переводом в lapsed"],
["AI_MODEL", "claude-haiku-4-5-20251001", "Модель Anthropic для подбора"],
["AI_TEMPERATURE", 0.3, "Температура генерации"],
["ADMIN_TG_ID", 5937498515, "Tg_id куратора (Руслан Васильев)"],
["PAID_PRICE_PER_LEAD", 500, "Цена pay-per-use для lapsed-менеджеров, ₽"],
["PAID_SUBSCRIPTION", 3000, "Цена месячной подписки, ₽"],
];
const DASHBOARD_ROWS = [
["Активных менеджеров", '=COUNTIF(Managers!H2:H; "active")'],
["Lapsed-менеджеров", '=COUNTIF(Managers!H2:H; "lapsed")'],
["Заявок за 30 дней", '=COUNTIFS(Leads!B2:B; ">="&(TODAY()-30))'],
["Сделок won", '=COUNTIF(Leads!L2:L; "won")'],
["Конверсия в сделку", '=IFERROR(COUNTIF(Leads!L2:L; "won")/COUNTA(Leads!A2:A); 0)'],
["Средний чек won-сделок, ₽", '=IFERROR(AVERAGEIF(Leads!L2:L; "won"; Leads!M2:M); 0)'],
["Замеров всего", '=COUNTA(Measurements!A2:A)'],
["Расход на AI, токенов", '=SUM(Leads!J2:J)'],
];
function setupDatabase() {
const ss = SpreadsheetApp.getActiveSpreadsheet();
// Удалить дефолтный пустой лист, если есть
const defaultSheet = ss.getSheetByName("Sheet1") || ss.getSheetByName("Лист1");
Object.keys(SHEETS).forEach((name, idx) => {
let sheet = ss.getSheetByName(name);
if (!sheet) {
sheet = ss.insertSheet(name, idx);
} else {
sheet.clear();
}
const headers = SHEETS[name];
sheet.getRange(1, 1, 1, headers.length).setValues([headers]).setFontWeight("bold").setBackground("#F0F9E8");
sheet.setFrozenRows(1);
sheet.autoResizeColumns(1, headers.length);
});
// Settings — заполнить дефолтами
const settings = ss.getSheetByName("Settings");
settings.getRange(2, 1, SETTINGS_DEFAULTS.length, 3).setValues(SETTINGS_DEFAULTS);
// Dashboard — заполнить формулами
const dashboard = ss.getSheetByName("Dashboard");
for (let i = 0; i < DASHBOARD_ROWS.length; i++) {
const [metric, formula] = DASHBOARD_ROWS[i];
dashboard.getRange(i + 2, 1).setValue(metric);
dashboard.getRange(i + 2, 2).setFormula(formula);
}
dashboard.setColumnWidth(1, 280);
dashboard.setColumnWidth(2, 140);
// Удалить пустой Sheet1 в самом конце (если был)
if (defaultSheet && ss.getSheets().length > 1) {
try { ss.deleteSheet(defaultSheet); } catch (e) {}
}
SpreadsheetApp.getActive().toast("✅ База готова — 8 листов созданы");
}
/** Удобный вызов через меню — необязательно */
function onOpen() {
SpreadsheetApp.getUi()
.createMenu("ЗОВ")
.addItem("Setup database", "setupDatabase")
.addToUi();
}