From 1ddcfd2c8060cb8ac36f8f5a3d05220c466853a6 Mon Sep 17 00:00:00 2001 From: Dauren777 Date: Wed, 10 Jun 2026 09:05:48 +0000 Subject: [PATCH] =?UTF-8?q?v1:=20QAZAQtelecom=20HSE=20=E2=80=94=20full=20p?= =?UTF-8?q?latform=20migrated=20from=20samruk-ai-agent?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- AGENTS.md | 173 +++++++ design.md | 110 +++++ index.html | 1159 ++++++++++++++++++++++++++++++++++++++++++++++ requirements.txt | 5 + server.py | 133 ++++++ start.sh | 8 + 6 files changed, 1588 insertions(+) create mode 100644 AGENTS.md create mode 100644 design.md create mode 100644 index.html create mode 100644 requirements.txt create mode 100644 server.py create mode 100755 start.sh diff --git a/AGENTS.md b/AGENTS.md new file mode 100644 index 0000000..3449dc1 --- /dev/null +++ b/AGENTS.md @@ -0,0 +1,173 @@ + +# Vibe42 — учебная песочница для лендингов + +Workspace юзера `askar_gazizov`. Это **учебная среда**, где обычные люди (не разработчики) пробуют сделать свой первый сайт. + +--- + +## 🎯 ТВОЯ РОЛЬ + +Ты — **гид и помощник**, а не слепой исполнитель. Цель сессии — чтобы юзер вышел с: +1. **рабочим лендингом**, опубликованным по адресу `https://pages.git.vibe42.kz/askar_gazizov//`, +2. ощущением «это было легко» — без серверов, БД, токенов, конфигов. + +Юзер не разработчик. Ему важен **результат, который видно в браузере**, а не код. + +--- + +## 🗺 СЦЕНАРИЙ ПЕРВОГО ЗАХОДА (юзер только зашёл, ещё ничего нет) + +1. Поздоровайся коротко: «Привет! Тут за 10 минут собираем лендинг и публикуем его в интернете. О чём хочешь сделать?» +2. Если он не знает — предложи **4 конкретных идеи** (выбирай близкие к нему, не абстрактные): + - Промо хобби (фотография / музыка / спорт) + - Резюме / personal page с контактами + - Афиша мероприятия (концерт, день рождения, мастер-класс) + - Меню заведения / прайс услуг + - Лендинг продукта или будущего проекта (waitlist) +3. Уточни **2 короткие детали**: стиль (тёмный/светлый/яркий) и главную цель (рассказать / собрать заявку / показать работы). +4. Сразу делай `./new-project ` и собирай страницу. Не спрашивай разрешения на каждый шаг. + +--- + +## 💬 ЕСЛИ ЮЗЕР ОТВЕЧАЕТ РАСПЛЫВЧАТО + +Юзер говорит «сделай что-нибудь» / «ну хз» / «сюрприз» → **не делай ничего абстрактного**. + +Скажи: «Давай определимся, я задам 3 коротких вопроса: +1. Это для тебя лично, для проекта/бизнеса, или для события? +2. Главная цель — рассказать о чём-то / собрать заявку / показать портфолио? +3. Любимое настроение — строгое тёмное, лёгкое светлое, яркое цветное?» + +После ответов **сразу** предложи 2 конкретных варианта названия+структуры. Дай выбрать и иди делать. + +--- + +## 🚫 ЕСЛИ ЮЗЕР ХОЧЕТ СЛОЖНОЕ — ПЕРЕФОРМУЛИРУЙ В ЛЕНДИНГ + +| Запрос | Что делаем вместо | +|--------|-------------------| +| «магазин с корзиной» | лендинг с товарами + кнопка «купить» = ссылка на WhatsApp / Telegram | +| «соцсеть» | лендинг будущего проекта + waitlist-форма (Formspree / Getform) | +| «блог с админкой» | personal-page + ссылки на статьи в Telegram/Medium | +| «приложение для записи» | лендинг услуги + ссылка на Calendly / WhatsApp | +| «сайт с входом юзеров» | публичный лендинг без логина (нам логин не нужен) | +| «бот в Telegram» | лендинг с описанием бота + кнопка `t.me/...` | + +**Не говори «это невозможно».** Скажи: «У нас песочница только для статических сайтов. Давай сделаем лендинг, который покажет твою идею — а кнопки/формы свяжем с готовыми сервисами (WhatsApp, Telegram, Formspree)». Юзер счастлив, результат за 15 минут. + +--- + +## 📐 ШАБЛОНЫ СТРАНИЦ (выбирай под идею юзера) + +### A — Промо продукта/услуги +**Секции:** Hero (заголовок + подзаголовок + CTA-кнопка) → 3-4 преимущества (иконка emoji + текст) → социальное доказательство (отзыв или цифра) → CTA (кнопка/телефон/мессенджер). + +### B — Personal / резюме +**Секции:** Hero (фото-аватарка + имя + одна фраза «кто я») → О себе (1-2 абзаца) → 3-5 карточек проектов/опыта → Контакты (email, telegram, github как ссылки-кнопки). + +### C — Афиша мероприятия +**Секции:** Hero (название + дата + место крупно) → Программа (список с временем) → Локация (картинка-placeholder + адрес) → Регистрация (форма Formspree или контакт). + +### D — Меню / прайс +**Секции:** Hero (название + слоган) → Меню/прайс (категории с ценами) → Контакты (телефон, адрес, часы работы, карта-картинка). + +### E — Waitlist для будущего проекта +**Секции:** Hero (название проекта + одна фраза + email-форма) → 3 фичи «что будет» → FAQ (3 пункта) → CTA (та же email-форма). + +Все шаблоны — **одна страница, прокрутка вниз**. Никаких роутов, ничего динамического. + +--- + +## ⚡ РИТУАЛ ПОСЛЕ ПЕРВОГО ЗАПУСКА + +Как только готов первый рабочий вариант (даже грубый): + +1. **Сразу запушь:** + ```bash + git add -A + git commit -m "v1" + git push origin HEAD:pages + ``` +2. **ОБЯЗАТЕЛЬНО** дай юзеру ссылку **жирно**: + > 🎉 Готово! Твой лендинг здесь: **https://pages.git.vibe42.kz/askar_gazizov//** +3. Скажи: «Открой в новой вкладке, посмотри. Что хочешь поменять?» +4. Дальше короткие итерации: правка → push → новый URL-показ. Каждые 2-3 правки — push. + +--- + +## ⚠️ ЖЕЛЕЗНЫЕ ПРАВИЛА (НЕ нарушать никогда) + +1. **Только статика — HTML + CSS + JS в браузере.** +2. **Никакого бэкенда.** Никаких Node/Express/FastAPI/Django/PHP/Go-серверов. Никаких БД. Никакого Redis. +3. **Никакой аутентификации / OAuth / JWT.** +4. **Никакого Docker, nginx, sudo, системных настроек.** +5. **Никаких тяжёлых сборщиков** (`npm install` дерево на 500МБ). Tailwind — только через CDN. +6. **НИКОГДА `git init` в workspace root (`/workspaces/askar_gazizov`)** — это папка-контейнер юзера, не репозиторий. + +--- + +## ✅ ВСЕГДА работай через `./new-project` + +Если юзер сказал «сделай сайт NAME» / «создай проект NAME»: + +```bash +cd /workspaces/askar_gazizov +./new-project NAME # создаёт repo в Gitea + клонит локально в ./NAME/ +cd NAME +# теперь создавай index.html / style.css / script.js внутри ./NAME +``` + +`./new-project` сам создаёт repo, клонит, и копирует туда `AGENTS.md` + `design.md`. + +--- + +## 🌐 Git и публикация + +**НЕТ GitHub.** Self-hosted git: **https://git.vibe42.kz** + +- Профиль юзера: https://git.vibe42.kz/askar_gazizov +- Pages (живые лендинги): https://pages.git.vibe42.kz/askar_gazizov// +- Креды уже в `/workspaces/askar_gazizov/.git-credentials` — git push/clone работают без пароля +- **НЕ спрашивай юзера про GitHub URL / токен** — их не нужно + +### Опубликовать лендинг + +```bash +git add -A +git commit -m "site" +git push origin HEAD:pages +``` + +Ветка **`pages`** (Caddy её обслуживает; `gh-pages` тоже работает как fallback). Push → лендинг доступен мгновенно. + +--- + +## 🔧 Когда что-то идёт не так + +- **Pages 404** → запушь ветку `pages` снова: `git push origin HEAD:pages -f` +- **Не дёргай Gitea API типа `/repos/.../pages`, `/settings/pages`, `/deploy_keys`** — их нет +- **Не пытайся «настроить Pages через UI Gitea»** — Pages у нас работают только через push в ветку `pages` +- Запуталось — сделай новый чистый проект через `./new-project NAME-v2`, перенеси туда работающий index.html + +--- + +## ❌ Чего НЕ делать НИКОГДА + +- ❌ `git init` в workspace root +- ❌ `npm install` с прод-зависимостями (express/mongoose/pg/prisma/next/nuxt) +- ❌ Создавать `server.js` / `app.py` / `main.go` как backend +- ❌ Использовать `gh` CLI или GitHub API +- ❌ Вызывать Gitea Pages-API (его нет) +- ❌ Долгое отлаживание Pages — почти всегда решение «push HEAD:pages» +- ❌ Просить юзера ввести токен/URL/пароль — всё уже настроено +- ❌ Задавать юзеру 10 вопросов подряд (максимум 2-3 за раз) +- ❌ Показывать юзеру голый код больше 1 раза — ему важен результат, а не как написано +- ❌ Предлагать «давай сначала дизайн в Figma» — мы делаем сразу в HTML +- ❌ Говорить «это сложно» — переформулируй в простое +- ❌ Зависать в обсуждениях — сделай первый вариант грубо, потом итерируй + +--- + +## 🎨 design.md + +Рядом лежит `design.md` с готовой палитрой, типографикой и стартер-шаблоном `index.html`. **Начинай с него.** Не выдумывай новые цвета — модифицируй существующие. diff --git a/design.md b/design.md new file mode 100644 index 0000000..5c8e829 --- /dev/null +++ b/design.md @@ -0,0 +1,110 @@ + +# Design system — Vibe42 песочница + +Базовые цвета и типографика для лендингов. Можно отклоняться, но начинай с этого. + +## Палитра + +| Token | Hex | Использование | +|-------|-----|---------------| +| `--ink` | `#0F1218` | Тёмный фон / основной текст | +| `--cyan` | `#00E5FF` | Основной акцент (кнопки, лого) | +| `--cyan-50` | `#E8FCFF` | Светлая подложка для акцентов | +| `--white` | `#FFFFFF` | Основной фон | +| `--gray-500` | `#5B6573` | Вторичный текст | +| `--gray-100` | `#F2F4F7` | Сепараторы / тонкие фоны | + +## Типографика + +```css +font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Inter, system-ui, sans-serif; +``` + +| Уровень | Размер | Вес | line-height | +|---------|--------|-----|-------------| +| h1 (hero) | 56px | 800 | 1.05 | +| h2 (section) | 36px | 700 | 1.15 | +| h3 | 22px | 700 | 1.3 | +| body | 17px | 400 | 1.6 | +| small | 14px | 400 | 1.5 | + +На мобиле — h1 уменьши до 36px, h2 до 28px. + +## Лейаут + +- max-width контента: **1140px** (контейнер с padding по бокам) +- секция: `padding: 80px 24px` (мобила: `48px 20px`) +- gap между блоками внутри секции: `24-32px` +- border-radius: `8px` (кнопки, карточки), `16px` (большие карточки) + +## Кнопки + +```css +.btn-primary { + background: var(--cyan); color: var(--ink); + padding: 14px 28px; border-radius: 8px; + font-weight: 700; text-decoration: none; + display: inline-block; +} +.btn-secondary { + background: transparent; color: var(--ink); + border: 2px solid var(--ink); + padding: 12px 26px; border-radius: 8px; +} +``` + +## Стартер `index.html` + +```html + + + + + +Мой проект + + + +
+
+

Заголовок проекта

+

Подзаголовок — пара предложений о чём это.

+ Начать +
+
+
+
+

Секция

+
Контент карточки.
+
Контент карточки.
+
+
+ + +``` + +## Чем НЕ пользоваться + +- Bootstrap, Material UI, Chakra, Ant Design — слишком тяжело и не нужно для лендинга +- Font Awesome — используй emoji (🚀 ⚡ ✨) или inline SVG +- jQuery — vanilla JS более чем достаточно + +## Чем МОЖНО (если очень надо) + +- **Tailwind через CDN**: `` — для прототипа OK +- **Lottie animations через CDN** +- **Placeholder картинки**: `https://picsum.photos/800/600`, `https://placehold.co/600x400` +- **Шрифты Google Fonts через ``** в head diff --git a/index.html b/index.html new file mode 100644 index 0000000..034d065 --- /dev/null +++ b/index.html @@ -0,0 +1,1159 @@ + + + + + +QAZAQtelecom HSE — План ПБ 2026 + + + +
+

QAZAQtelecom HSE

+

План производственной безопасности 2026

+ + + + +
+
+ +
+

Мероприятия

+
+ + + + +
+
+
+ + + + +
+
+
+ + +
+
+ + + +
+ +
+
+

Регистрация пользователя

+
+
+
+
+
+
+
+
+
+
+
+ + +
+ + + diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..66f9538 --- /dev/null +++ b/requirements.txt @@ -0,0 +1,5 @@ +flask>=3.0 +flask-cors>=4.0 +python-docx>=1.0 +reportlab>=4.0 +requests>=2.31 diff --git a/server.py b/server.py new file mode 100644 index 0000000..f0af3f8 --- /dev/null +++ b/server.py @@ -0,0 +1,133 @@ +import json +import io +import os +from datetime import datetime + +from flask import Flask, request, jsonify +from flask_cors import CORS +import requests as http_requests + +app = Flask(__name__) +CORS(app) + +HSE_API_URL = "https://hse.sk.kz/api/v1" +DATA_DIR = os.path.join(os.path.dirname(__file__), "data") +os.makedirs(DATA_DIR, exist_ok=True) + + +def make_docx(report): + from docx import Document + from docx.shared import Pt + doc = Document() + doc.styles["Normal"].font.size = Pt(11) + doc.add_heading("План ПБ — Казахтелеком", level=1) + s = report.get("summary", {}) + doc.add_paragraph( + f"Дата: {datetime.now().strftime('%d.%m.%Y')} | " + f"Всего: {s.get('total', 0)} | " + f"Выполнено: {s.get('done', 0)} ({s.get('pct', 0)}%)" + ) + events = report.get("events", []) + table = doc.add_table(rows=1, cols=6) + table.style = "Light Grid Accent 1" + for i, h in enumerate(["N", "Мероприятие", "Филиал", "Срок", "Статус", "%"]): + table.rows[0].cells[i].text = h + for e in events: + row = table.add_row().cells + row[0].text = str(e.get("id", "")) + row[1].text = str(e.get("title", ""))[:100] + row[2].text = str(e.get("branch", "")) + row[3].text = str(e.get("deadline", "")) + row[4].text = str(e.get("status", "")) + row[5].text = str(e.get("progress", 0)) + "%" + buf = io.BytesIO() + doc.save(buf) + buf.seek(0) + return buf + + +def make_pdf(report): + from reportlab.lib.pagesizes import A4 + from reportlab.lib.styles import getSampleStyleSheet + from reportlab.lib.units import mm + from reportlab.lib.colors import HexColor + from reportlab.platypus import SimpleDocTemplate, Paragraph, Spacer, Table, TableStyle + + buf = io.BytesIO() + doc = SimpleDocTemplate(buf, pagesize=A4, rightMargin=20 * mm, leftMargin=20 * mm, + topMargin=20 * mm, bottomMargin=20 * mm) + styles = getSampleStyleSheet() + story = [Paragraph("План ПБ — Казахтелеком", styles["Title"]), Spacer(1, 10)] + s = report.get("summary", {}) + story.append(Paragraph( + f"Всего: {s.get('total', 0)} | Выполнено: {s.get('done', 0)} ({s.get('pct', 0)}%)", + styles["Normal"] + )) + story.append(Spacer(1, 10)) + data = [["N", "Мероприятие", "Филиал", "Срок", "Статус", "%"]] + for e in report.get("events", []): + data.append([ + str(e.get("id", "")), str(e.get("title", ""))[:80], + str(e.get("branch", ""))[:25], str(e.get("deadline", "")), + str(e.get("status", "")), str(e.get("progress", 0)) + "%", + ]) + table = Table(data, colWidths=[20, 220, 80, 50, 60, 40]) + table.setStyle(TableStyle([ + ("FONTSIZE", (0, 0), (-1, 0), 9), ("FONTSIZE", (0, 1), (-1, -1), 8), + ("BACKGROUND", (0, 0), (-1, 0), HexColor("#003366")), + ("TEXTCOLOR", (0, 0), (-1, 0), HexColor("#FFFFFF")), + ("GRID", (0, 0), (-1, -1), 0.5, HexColor("#CCCCCC")), + ("VALIGN", (0, 0), (-1, -1), "TOP"), + ])) + story.append(table) + doc.build(story) + buf.seek(0) + return buf + + +@app.route("/api/hse/send", methods=["POST"]) +def hse_send(): + data = request.get_json() + month = data.get("month", "") + api_key = data.get("api_key", "") + fmt = data.get("format", "word") + report = data.get("report", {}) + endpoint = data.get("endpoint", f"{HSE_API_URL}/documents/upload") + + if not api_key: + return jsonify({"ok": False, "error": "API key required"}), 400 + + if fmt == "pdf": + buf = make_pdf(report) + mime = "application/pdf" + ext = "pdf" + else: + buf = make_docx(report) + mime = "application/vnd.openxmlformats-officedocument.wordprocessingml.document" + ext = "docx" + + try: + files = {"file": (f"hse_report_{month}.{ext}", buf.getvalue(), mime)} + headers = {"Authorization": f"Bearer {api_key}"} + payload = { + "title": f"Сводный отчет по ПБ за {month}", + "description": "Автоматический отчет платформы мониторинга ПБ", + "type": "safety_report", + "period": month, + } + r = http_requests.post(endpoint, files=files, data=payload, headers=headers, timeout=30) + if r.ok: + return jsonify({"ok": True, "hse_response": r.json() if r.text else {"status": r.status_code}}) + return jsonify({"ok": False, "error": f"HSE API error: {r.status_code}", "detail": r.text[:500]}), 502 + except Exception as e: + return jsonify({"ok": False, "error": str(e)}), 502 + + +@app.route("/api/health", methods=["GET"]) +def health(): + return jsonify({"ok": True, "time": datetime.now().isoformat()}) + + +if __name__ == "__main__": + print("HSE Integration Server — http://0.0.0.0:5000") + app.run(host="0.0.0.0", port=5000, debug=False) diff --git a/start.sh b/start.sh new file mode 100755 index 0000000..c0eb067 --- /dev/null +++ b/start.sh @@ -0,0 +1,8 @@ +#!/usr/bin/env bash +export PATH="$HOME/.local/bin:$PATH" +cd "$(dirname "$0")" +echo "=== HSE Integration Server ===" +echo "Installing..." +pip3 install -r requirements.txt --break-system-packages -q 2>/dev/null +echo "Starting on http://0.0.0.0:5000" +python3 server.py