Compare commits
124 Commits
2022-archi
...
v0.3.1
| Author | SHA1 | Date | |
|---|---|---|---|
| e15017f3a6 | |||
| 958f11f398 | |||
| 3ab38b4885 | |||
| 4b038302c3 | |||
| 3c10b490b3 | |||
| 2d7e0813d6 | |||
| 5678624608 | |||
| cc739c5e07 | |||
| 54ed78185c | |||
| b376fdcaf9 | |||
| a3bc67613f | |||
| cb9dab9e56 | |||
| 9575e0e0d9 | |||
| 2396387883 | |||
| 8fe641c8f4 | |||
| 58e17d7d3b | |||
| 2dee8b77cb | |||
| 03ed9d24f9 | |||
| 71059bdae6 | |||
| 50b5ee4bdf | |||
| 98912808a1 | |||
| 968c4219aa | |||
| 7f81a79862 | |||
| ec82d4d36f | |||
| 468d8e8fc4 | |||
| 71d0115046 | |||
| 1e7c18a507 | |||
| 394f92ebce | |||
| ff250b10a5 | |||
| cca42ac39c | |||
| 369bf9aca8 | |||
| 74411a7add | |||
| b1a9b2649f | |||
| 09f9e7ec6a | |||
| a00c526c87 | |||
| 98b0b79c87 | |||
| 12ba0fe92d | |||
| ce63d7d42b | |||
| 6c59c4e0c1 | |||
| 370a5ed359 | |||
| 7c830530ef | |||
| c96d85da44 | |||
| b27d6f0546 | |||
| 998e6caf8f | |||
| 81680b75ab | |||
| d5c1024124 | |||
| bcebc20943 | |||
| bfd785d78e | |||
| 7747155595 | |||
| ae39f9462a | |||
| 5cd2acc84c | |||
| d028f85356 | |||
| a5736e19c7 | |||
| bb08213153 | |||
| c4fb0a29b7 | |||
| 0f30dca006 | |||
| 0efc12fe7a | |||
| c62fbceaaf | |||
| 37f0535ddf | |||
| db6cd46915 | |||
| b719b58c7b | |||
| 9f4b86a418 | |||
| bcc8c56970 | |||
| dc379fa8da | |||
| 1b2666f3d7 | |||
| b615d2caa4 | |||
| 71c8bbc9ea | |||
| ece99a398e | |||
| b74e70e4f7 | |||
| 8a485b756b | |||
| 4abd8a43f1 | |||
| 5a507f3e17 | |||
| 8da90ad0fd | |||
| 065c51681d | |||
| 6eb8d15ba8 | |||
| 4d49e6775a | |||
| 9b8d71ee71 | |||
| 096300406a | |||
| 9b4cf9de51 | |||
| 90af1f2a9e | |||
| 90ecfef39d | |||
| ef7e9b465f | |||
| fc5eae03c2 | |||
| fda5afa6e4 | |||
| f45165822b | |||
| 004d7392ad | |||
| 0104db39bd | |||
| b1bbde2d59 | |||
| 359664796d | |||
| 45ad5481aa | |||
| e260ded477 | |||
| 4420bcd3d1 | |||
| 5e30d3f8a4 | |||
| 9fb4843f8f | |||
| df74175504 | |||
| 1f03219372 | |||
| 002ed0f82a | |||
| d654b73d65 | |||
| 80b9abfaef | |||
| 21656e63de | |||
| ac68214297 | |||
| cf24fab53a | |||
| 91b4d635da | |||
| 4181f34bcb | |||
| 43f63f4461 | |||
| 1afe77fde0 | |||
| 6016247718 | |||
| 026a2a74ab | |||
| 59d8ad1e4c | |||
| 2778ccf45a | |||
| 5933fa46de | |||
| de2d14332e | |||
| 25e148a21e | |||
| 7f07f82b11 | |||
| 45f243b38b | |||
| 15abbe7367 | |||
| 096aa2d78b | |||
| 2d7b6f86a4 | |||
| 37fe09588b | |||
| 2867ec485c | |||
| 1f24b12754 | |||
| e779b78707 | |||
| 63cf85515b | |||
| f3a16788b0 |
128
.dockerignore
Normal file
128
.dockerignore
Normal file
@@ -0,0 +1,128 @@
|
||||
# Git и версионирование
|
||||
.git
|
||||
.gitignore
|
||||
.gitattributes
|
||||
|
||||
# Документация и инструкции
|
||||
*.md
|
||||
README.md
|
||||
SETUP.md
|
||||
AGENTS.md
|
||||
MANAGEMENT_RUNBOOK.md
|
||||
|
||||
# Python и виртуальное окружение
|
||||
__pycache__
|
||||
*.py[cod]
|
||||
*$py.class
|
||||
*.so
|
||||
.Python
|
||||
env/
|
||||
venv/
|
||||
ENV/
|
||||
build/
|
||||
develop-eggs/
|
||||
dist/
|
||||
downloads/
|
||||
eggs/
|
||||
.eggs/
|
||||
lib/
|
||||
lib64/
|
||||
parts/
|
||||
sdist/
|
||||
var/
|
||||
wheels/
|
||||
*.egg-info/
|
||||
.installed.cfg
|
||||
*.egg
|
||||
pip-log.txt
|
||||
pip-delete-this-directory.txt
|
||||
|
||||
# IDE и редакторы
|
||||
.vscode
|
||||
.idea
|
||||
*.swp
|
||||
*.swo
|
||||
*~
|
||||
.DS_Store
|
||||
.project
|
||||
.pydevproject
|
||||
.settings/
|
||||
*.sublime-workspace
|
||||
*.sublime-project
|
||||
|
||||
# Тестирование и линтинг
|
||||
.pytest_cache/
|
||||
.coverage
|
||||
htmlcov/
|
||||
.tox/
|
||||
.mypy_cache/
|
||||
.hypothesis/
|
||||
|
||||
# Локальные данные и конфиги
|
||||
.env
|
||||
.env.local
|
||||
local_settings.py
|
||||
database/*.sqlite3
|
||||
database/*.db
|
||||
logs/
|
||||
*.log
|
||||
|
||||
# Статика для разработки и bootstrap-стили
|
||||
public/static/bootstrap-3.3.7-dist.zip
|
||||
public/static/css/*.map
|
||||
public/static/js/*.map
|
||||
public/static/**/*.map
|
||||
|
||||
# Временные файлы
|
||||
*.tmp
|
||||
*.temp
|
||||
*.bak
|
||||
|
||||
# Шрифты
|
||||
*.ttf
|
||||
*.otf
|
||||
|
||||
# Системные файлы macOS
|
||||
.AppleDouble
|
||||
.LSOverride
|
||||
.Spotlight-V100
|
||||
.Trashes
|
||||
.VolumeIcon.icns
|
||||
|
||||
# Системные файлы Linux
|
||||
.Xauthority
|
||||
.Xdefaults
|
||||
|
||||
# Docker и CI/CD
|
||||
docker-compose.override.yml
|
||||
.dockerignore
|
||||
Dockerfile*
|
||||
|
||||
# Node modules (если понадобятся в будущем)
|
||||
node_modules/
|
||||
npm-debug.log
|
||||
yarn-error.log
|
||||
|
||||
# Конфиги оставляем как пример!
|
||||
# config/ -> НЕ ИСКЛЮЧАЕМ (нужен как образец)
|
||||
# *.ini, *.conf -> НЕ ИСКЛЮЧАЕМ (нужны как примеры)
|
||||
|
||||
# Django автогенерируемые файлы (создаются при старте контейнера, не нужны в образе)
|
||||
# public/static_collected/ # собранная статика (создаётся при collectstatic)
|
||||
# public/media/ # медиа-файлы (могут быть большие, создаются динамически)
|
||||
# oknardia/templates/seria_info/prepared/ # пре-рендер шаблоны серий (создаются управ.командой)
|
||||
# public/media/_serv_sitemap/ # sitemap'ы (создаются управ.командой)
|
||||
|
||||
# Примечание: Мы не исключаем эти директории потому что:
|
||||
# 1. public/media/ может содержать важные медиа для разработки
|
||||
# 2. пре-рендер шаблоны кэшируются для production оптимизации
|
||||
# При сборке production образа они создаются автоматически в docker-compose через manage.py команды.
|
||||
|
||||
# Базы данных и кеши
|
||||
*.sqlite3
|
||||
*.db
|
||||
database/oknadria_backup*.sqlite3
|
||||
|
||||
# Миграции можем оставить, они нужны для контейнера
|
||||
# oknardia/oknardia/migrations/ -> НЕ ИСКЛЮЧАЕМ
|
||||
|
||||
72
.env.sample
72
.env.sample
@@ -1,5 +1,6 @@
|
||||
# Django Configuration Example
|
||||
# Копируйте этот файл в .env.local и заполните реальные значения
|
||||
# Все настройки читаются из переменных окружения (DEV/PROD без распознавания хоста)
|
||||
# Для локальной разработки можно скопировать файл в .env.local и экспортировать его.
|
||||
|
||||
# ============================================================================
|
||||
# DJANGO
|
||||
@@ -15,19 +16,39 @@ DEBUG=False
|
||||
# Допустимые хосты (разделены запятой без пробелов)
|
||||
ALLOWED_HOSTS=localhost,127.0.0.1,yourdomain.com
|
||||
|
||||
# Базовый публичный URL сайта (используется для абсолютных URL в sitemap.xml)
|
||||
SITE_BASE_URL=https://yourdomain.com
|
||||
|
||||
# Админы для email-оповещений Django (формат: Имя:email,Имя2:email2)
|
||||
ADMINS=Admin:admin@example.com
|
||||
|
||||
# URL для доступа к админке Django (можно сменить для безопасности, чтобы боты не могли её найти)
|
||||
ADMIN_URL=admin/
|
||||
|
||||
# CSRF Trusted Origins (для корректной работы CSRF при доступе к админке и другим формам с разных доменов/портов)
|
||||
DJANGO_CSRF_TRUSTED_ORIGINS=http://127.0.0.1:8000,http://localhost:8000,https://yourdomain.com
|
||||
|
||||
# ============================================================================
|
||||
# DATABASE
|
||||
# ============================================================================
|
||||
|
||||
# Database backend (по умолчанию mysql)
|
||||
DATABASE_ENGINE=django.db.backends.mysql
|
||||
# Database backend (по умолчанию SQLite)
|
||||
DATABASE_ENGINE=django.db.backends.sqlite3
|
||||
|
||||
# Database connection
|
||||
DATABASE_HOST=localhost
|
||||
DATABASE_PORT=3306
|
||||
DATABASE_NAME=django_oknardia
|
||||
DATABASE_USER=web
|
||||
DATABASE_PASSWORD=your-db-password-here
|
||||
# Имя/путь базы данных:
|
||||
# - для SQLite: только имя файла (полный путь соберется в settings.py через PROJECT_ROOT/database)
|
||||
# - для MySQL/MariaDB: имя базы
|
||||
DATABASE_NAME=oknardia.sqlite3
|
||||
|
||||
# Для MySQL/MariaDB (используются, если DATABASE_ENGINE=django.db.backends.mysql)
|
||||
# DATABASE_HOST=localhost
|
||||
# DATABASE_PORT=3306
|
||||
# DATABASE_USER=name-for-db-user
|
||||
# DATABASE_PASSWORD=your-db-password-here
|
||||
|
||||
|
||||
# Подкаталог в MEDIA_ROOT, где хранится кеш sitemap-файлов
|
||||
SITEMAP_SUBDIR=_serv_sitemap
|
||||
|
||||
# ============================================================================
|
||||
# EMAIL
|
||||
@@ -40,6 +61,7 @@ EMAIL_BACKEND=django.core.mail.backends.smtp.EmailBackend
|
||||
EMAIL_HOST=smtp.example.com
|
||||
EMAIL_PORT=587
|
||||
EMAIL_USE_TLS=True
|
||||
EMAIL_USE_SSL=False
|
||||
EMAIL_HOST_USER=your-email@example.com
|
||||
EMAIL_HOST_PASSWORD=your-email-password
|
||||
|
||||
@@ -95,12 +117,22 @@ LOG_LEVEL=INFO
|
||||
# CELERY_BROKER_URL=redis://localhost:6379/0
|
||||
# CELERY_RESULT_BACKEND=redis://localhost:6379/0
|
||||
|
||||
# ============================================================================
|
||||
# DOCKER: LOCAL PRODUCTION TESTING
|
||||
# ============================================================================
|
||||
|
||||
# Разрешить Django обслуживать медиа-файлы через Python (только для локального тестирования)
|
||||
# ВАЖНО: В настоящем production медиа и статику обслуживает Nginx, а не Django!
|
||||
# Используется ТОЛЬКО в docker-compose.local-prod.yml для локального тестирования production конфигурации.
|
||||
# На production сервере НЕ устанавливайте это значение в True!
|
||||
ALLOW_MEDIA_SERVE=False
|
||||
|
||||
# ============================================================================
|
||||
# ИНСТРУКЦИЯ ПО ИСПОЛЬЗОВАНИЮ
|
||||
# ============================================================================
|
||||
|
||||
# 1. Скопируйте этот файл:
|
||||
# cp .env.example .env.local
|
||||
# cp .env.sample .env.local
|
||||
#
|
||||
# 2. Отредактируйте значения в .env.local:
|
||||
# nano .env.local
|
||||
@@ -108,9 +140,8 @@ LOG_LEVEL=INFO
|
||||
# 3. Убедитесь, что .env.local в .gitignore:
|
||||
# grep ".env" .gitignore
|
||||
#
|
||||
# 4. Используйте python-dotenv для загрузки переменных в settings.py:
|
||||
# from dotenv import load_dotenv
|
||||
# load_dotenv()
|
||||
# 4. Экспортируйте переменные перед запуском Django:
|
||||
# set -a; source .env.local; set +a
|
||||
#
|
||||
# ВАЖНО:
|
||||
# - НИКОГДА не коммитьте .env.local или файлы с реальными значениями в git!
|
||||
@@ -121,3 +152,18 @@ LOG_LEVEL=INFO
|
||||
# - AWS Systems Manager Parameter Store
|
||||
# - HashiCorp Vault
|
||||
|
||||
|
||||
# ****************************************************************************************************************
|
||||
# Системные пути на хосте (ТОЛЬКО ДЛЯ ПРОДАКШЕНА)
|
||||
# Используется скриптом для генерации корректного Nginx конфига (alias к медиа-файлам).
|
||||
# На локальной машине разработчика (Dev) эта переменная игнорируется.
|
||||
# В ПРОДАКШЕНЕ: Укажите полный путь к папке проекта на сервере
|
||||
HOST_PROJECT_PATH=/home/username/projects
|
||||
|
||||
# ****************************************************************************************************************
|
||||
# Настройки доступа к пакетам в репозитории, чтобы wathtower мог проверять их свежесть и скачивать для обновления.
|
||||
# Получить эти данные можно в настройках вашего репозитория, например:
|
||||
# для GitHub: в разделе "Developer settings" -> "Personal access tokens";
|
||||
# для Gitea: в разделе "Settings / Настройки" -> "Actions / Действия" -> "Secrets / Секреты".
|
||||
REPO_USER=[login]
|
||||
REPO_PASS=[token]
|
||||
|
||||
72
.gitea/workflows/docker-publish.yaml
Normal file
72
.gitea/workflows/docker-publish.yaml
Normal file
@@ -0,0 +1,72 @@
|
||||
name: Build and Push Docker Image
|
||||
run-name: Build and Push Docker Image ${{ github.ref_name }}
|
||||
|
||||
on:
|
||||
push:
|
||||
# Запускать сборку только при создании тега, начинающегося с 'v' (например, v1.0.0, v2.3.1)
|
||||
tags:
|
||||
- 'v*'
|
||||
|
||||
env:
|
||||
REGISTRY: git.cube2.ru
|
||||
IMAGE_NAME: ${{ github.repository }}
|
||||
|
||||
jobs:
|
||||
build-and-push:
|
||||
runs-on: ubuntu-latest # Или метка вашего раннера, если он специфичный (например, macos или self-hosted)
|
||||
container:
|
||||
image: catthehacker/ubuntu:act-latest
|
||||
|
||||
permissions:
|
||||
contents: read
|
||||
packages: write
|
||||
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v3
|
||||
|
||||
# Настройка QEMU для мультиплатформенной сборки (если нужно собирать под разные архитектуры)
|
||||
- name: Set up QEMU
|
||||
uses: docker/setup-qemu-action@v2
|
||||
|
||||
# Настройка Docker Buildx (обязательно для build-push-action)
|
||||
- name: Set up Docker Buildx
|
||||
uses: docker/setup-buildx-action@v2
|
||||
|
||||
# Логин в реестр Gitea
|
||||
- name: Log in to the Container registry
|
||||
uses: docker/login-action@v2
|
||||
with:
|
||||
registry: ${{ env.REGISTRY }}
|
||||
username: ${{ github.actor }}
|
||||
password: ${{ secrets.REGISTRY_PASSWORD }}
|
||||
|
||||
# Извлечение метаданных (тегов и лейблов) для Docker
|
||||
- name: Extract metadata (tags, labels) for Docker
|
||||
id: meta
|
||||
uses: docker/metadata-action@v4
|
||||
with:
|
||||
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
|
||||
tags: |
|
||||
type=ref,event=tag
|
||||
type=raw,value=latest,enable=${{ github.ref_type == 'tag' }}
|
||||
|
||||
# Сборка и отправка образа
|
||||
- name: Build and push Docker image
|
||||
uses: docker/build-push-action@v4
|
||||
with:
|
||||
context: .
|
||||
file: Dockerfile
|
||||
push: true
|
||||
# Собираем под текущую архитектуру (linux/amd64).
|
||||
# Если сервер и MacMini на разных архитектурах (x86 vs ARM), добавьте нужные, например: linux/amd64,linux/arm64
|
||||
# platforms: linux/amd64,linux/arm64
|
||||
# ---
|
||||
# Собираем только под linux/amd64 (для скорости + все равно сервер на x86).
|
||||
platforms: linux/amd64
|
||||
tags: ${{ steps.meta.outputs.tags }}
|
||||
labels: ${{ steps.meta.outputs.labels }}
|
||||
# ДОБАВЛЕНО для медленного интернета и оптимизации сборки:
|
||||
cache-from: type=gha
|
||||
cache-to: type=gha,mode=max
|
||||
timeout: 1800 # Увеличено до 30 минут на всю сборку
|
||||
35
.gitignore
vendored
35
.gitignore
vendored
@@ -133,3 +133,38 @@ dmypy.json
|
||||
|
||||
# Pyre type checker
|
||||
.pyre/
|
||||
|
||||
# MacOS specific files
|
||||
.DS_Store
|
||||
|
||||
# Database dumps and backups (CRITICAL - NEVER commit production data!)
|
||||
SQL/
|
||||
*.sql
|
||||
*.dump
|
||||
*.backup
|
||||
*.sql.gz
|
||||
db.json
|
||||
db.json.zip
|
||||
|
||||
# API keys, certificates, and credentials
|
||||
*.key
|
||||
*.pem
|
||||
*.p12
|
||||
*.p8
|
||||
*.crt
|
||||
*.cert
|
||||
|
||||
# IDE and editor specific
|
||||
.vscode/settings.json
|
||||
.idea/vcs.xml
|
||||
.idea/inspectionProfiles/
|
||||
|
||||
# Project specific ignore patterns
|
||||
.github/
|
||||
.log/
|
||||
.logs/
|
||||
sitemap*.xml
|
||||
|
||||
# Django static files (собранная статика, пересоздается при деплое)
|
||||
public/static_collected/
|
||||
|
||||
|
||||
@@ -213,6 +213,7 @@ python manage.py collectstatic # собрать статику для
|
||||
5. **Foreign Key ON_DELETE**: используется в основном `DO_NOTHING` и `SET_NULL`, будь осторожен при удалении
|
||||
6. **Двойной хост**: убедись, что используешь правильные переменные из `my_secret.py` для текущей машины
|
||||
7. **Индексирование БД**: большинство полей для поиска уже имеют `db_index=True`, но проверь при добавлении фильтров
|
||||
8. **SEO-даты и свежесть контента**: при переделке вьюх/шаблонов отдельно проверяй, нужны ли ещё `last_update`, `PUB_DAT`, `Date4Meta` и `Last4Meta`; если дата не участвует в смысловой логике страницы, лучше оставить базовые `{% now %}` из `base.html`, а не тащить лишний контекст во вьюху и не нагружать бекенд.
|
||||
|
||||
## Реферальные ссылки (для более глубокого изучения)
|
||||
|
||||
|
||||
361
CACHE_PRERENDER_SYSTEM.md
Normal file
361
CACHE_PRERENDER_SYSTEM.md
Normal file
@@ -0,0 +1,361 @@
|
||||
# Система двухуровневого кеширования страниц серий
|
||||
|
||||
## 📖 Описание
|
||||
|
||||
Система страниц серий домов использует **двухуровневое кеширование**:
|
||||
|
||||
1. **Статический кеш** — дорогостоящие данные (карты, графики, схемы), генерируются один раз и сохраняются на диск
|
||||
2. **Динамические данные** — верхняя статья (редактируется через админку) и таблица оконных проёмов (показывает свежие предложения), пересчитываются при каждом запросе
|
||||
|
||||
---
|
||||
|
||||
## 🏗️ Архитектура кеша
|
||||
|
||||
Для каждой серии создаются **3 отдельных кешируемых файла**:
|
||||
|
||||
| Файл | Содержимое | Размер | Где появляется |
|
||||
|------|-----------|--------|---|
|
||||
| `{seria_id}_id_static_flaps.html` | Схемы открывания и типовые размеры окон | 4-7KB | В разделе "Дома серии: типовые размеры" |
|
||||
| `{seria_id}_id_static_graph.html` | Google Charts: график ввода в эксплуатацию | 2-3KB | В контейнере height:300px |
|
||||
| `{seria_id}_id_static_map_stats.html` | Yandex Maps + блок статистики | 6-46KB | Карта (col-md-7) + статистика (col-md-4) |
|
||||
|
||||
### Верхняя статья — ДИНАМИЧЕСКИЕ ДАННЫЕ, не кешируется!
|
||||
|
||||
**Верхняя статья про серию** (`THIS_SERIA_DESCRIPTION`) — это динамические данные, которые:
|
||||
- Хранятся в БД (поле `sDescription` модели `Seria_Info`)
|
||||
- **Редактируются через админку** → нужны изменения **без перезагрузки контейнера**
|
||||
- Рендерятся всегда из БД, не сохраняются в кеш-файлы
|
||||
|
||||
Если бы мы кешировали верхнюю статью:
|
||||
- Админ редактирует статью → кеш-файл не обновляется автоматически
|
||||
- Нужно перезагрузить контейнер или вручную удалять файл кеша
|
||||
- При регенерации кеша может произойти перезапись старыми данными
|
||||
|
||||
**Решение**: верхняя статья **всегда рендерится из БД**, как и таблица окон.
|
||||
|
||||
---
|
||||
|
||||
## 🔄 Логика работы
|
||||
|
||||
### При первом запросе seriesId (cache miss):
|
||||
|
||||
1. View `catalog_seria_info()` обнаруживает отсутствие кеш-файлов
|
||||
2. Вычисляются дорогостоящие данные:
|
||||
- Геокоординаты всех зданий серии
|
||||
- Год ввода в эксплуатацию (для графика)
|
||||
- Схемы открывания окон
|
||||
3. Генерируются **3 отдельных файла** в `oknardia/templates/seria_info/prepared/`
|
||||
4. Верхняя статья и таблица окон **рендерятся из БД** при каждом запросе
|
||||
5. Контекст получает пути к 3 кеш-файлам + свежие динамические данные
|
||||
6. Main-шаблон включает 3 кеш-файла + 2 динамических блока
|
||||
7. Ответ отправляется пользователю
|
||||
|
||||
### При последующих запросах (cache hit):
|
||||
|
||||
1. View обнаруживает, что все 3 файла существуют
|
||||
2. **Верхняя статья пересчитывается заново** из БД (от админа)
|
||||
3. **Таблица окон пересчитывается заново** (новые предложения видны сразу)
|
||||
4. Main-шаблон включает 3 статических файла + 2 динамических блока
|
||||
5. Ответ отправляется быстро (схемы, график, геоданные из кеша)
|
||||
|
||||
### Ключевой мюмент: Всё свежее!
|
||||
|
||||
- Кеширование **не трогает** верхнюю статью и таблицу окон → они всегда свежие
|
||||
- Новые цены от поставщиков видны пользователям **сразу**
|
||||
- Админ может редактировать текст про серию → видно **без перезагрузки** контейнера
|
||||
- Таблица пересчитывается в запросе через Q-фильтры к БД (есть индексы на БД)
|
||||
|
||||
---
|
||||
|
||||
## 📁 Структура файлов
|
||||
|
||||
### Templates (шаблоны)
|
||||
|
||||
```
|
||||
oknardia/templates/seria_info/
|
||||
├── all_seria_info_pre_light.html # Main-шаблон (включает 3 static-файла + динамику)
|
||||
├── all_seria_info_pre_light_static_flaps.html # ШАГ 1: Схемы открывания (кешируется)
|
||||
├── all_seria_info_pre_light_static_graph.html # ШАГ 2: График ввода (кешируется)
|
||||
├── all_seria_info_pre_light_static_map_stats.html # ШАГ 3: Карта + статистика (кешируется)
|
||||
└── prepared/ # Директория с кеш-файлами
|
||||
├── 210_id_static_flaps.html
|
||||
├── 210_id_static_graph.html
|
||||
├── 210_id_static_map_stats.html
|
||||
├── 100_id_static_flaps.html
|
||||
└── ... (93 файла: 31 серия × 3 типа)
|
||||
```
|
||||
|
||||
### Генерация кеша (Web-логика)
|
||||
|
||||
```
|
||||
oknardia/web/
|
||||
├── catalog_series.py # catalog_seria_info() — генерирует 3 файла + динамику
|
||||
└── management/commands/
|
||||
└── regenerate_seria_prerender.py # Batch-команда для регенерации всех серий
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 🚀 Управление кешем
|
||||
|
||||
### Разработка (DEV-режим)
|
||||
|
||||
```bash
|
||||
python manage.py runserver
|
||||
```
|
||||
|
||||
В DEV-режиме (`DEBUG = True`):
|
||||
- Кеш **не используется** (всегда `PRE_RENDERED_STATIC_*_PATH = ""`)
|
||||
- Main-шаблон рендерит данные напрямую
|
||||
- Удобно для разработки (вижу изменения сразу)
|
||||
|
||||
### Production (PROD-режим)
|
||||
|
||||
#### Первоначальная генерация (все 31 серия):
|
||||
|
||||
```bash
|
||||
python manage.py regenerate_seria_prerender
|
||||
```
|
||||
|
||||
Вывод:
|
||||
```
|
||||
OK seria 100: 3 кеш-файла созданы
|
||||
OK seria 12: 3 кеш-файла созданы
|
||||
...
|
||||
Готово. Обработано: 31. Создано/пересоздано: 31 × 3 файла. Пропущено: 0.
|
||||
```
|
||||
|
||||
#### Регенерация конкретной серии:
|
||||
|
||||
```bash
|
||||
python manage.py regenerate_seria_prerender --seria-id 210
|
||||
```
|
||||
|
||||
#### Dry-run (без создания файлов):
|
||||
|
||||
```bash
|
||||
python manage.py regenerate_seria_prerender --dry-run
|
||||
```
|
||||
|
||||
#### Force-переписать даже если есть кеш:
|
||||
|
||||
```bash
|
||||
python manage.py regenerate_seria_prerender --force
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 📊 Когда нужна регенерация кеша?
|
||||
|
||||
### ✅ Регенерируйте, если:
|
||||
|
||||
- Изменены координаты зданий (geo-данные)
|
||||
- Добавлены новые здания в серию
|
||||
- Обновлены годы ввода в эксплуатацию (для графика)
|
||||
- Изменены схемы открывания окон
|
||||
- Обновлена верхняя статья про серию
|
||||
|
||||
```bash
|
||||
# Все изменилось → перестроить все
|
||||
python manage.py regenerate_seria_prerender --force
|
||||
|
||||
# Изменилась одна серия → перестроить одну
|
||||
python manage.py regenerate_seria_prerender --seria-id 210
|
||||
```
|
||||
|
||||
### ❌ НЕ нужна регенерация, если:
|
||||
|
||||
- Добавлены новые **предложения** (цены от поставщиков)
|
||||
- Обновлены **наличие/status** существующих предложений
|
||||
- Система просто должна **показать свежие цены**
|
||||
|
||||
Таблица окон пересчитывается при каждом запросе автоматически! 🎉
|
||||
|
||||
---
|
||||
|
||||
## 🔌 Интеграция с Docker
|
||||
|
||||
В контейнере (PROD-режим):
|
||||
|
||||
```dockerfile
|
||||
# Генерируем кеш пр<D0BF><D180> запуске
|
||||
RUN python manage.py regenerate_seria_prerender
|
||||
|
||||
# Запускаем сервер
|
||||
CMD ["gunicorn", "oknardia.wsgi:application", "--bind", "0.0.0.0:8000"]
|
||||
```
|
||||
|
||||
Кеш-файлы сохраняются на диск в томе, поэтому переживают перезагрузку контейнера.
|
||||
|
||||
---
|
||||
|
||||
## 📝 Контекстные переменные
|
||||
|
||||
### При первом запросе (происходит генерация):
|
||||
|
||||
View `catalog_seria_info()` отправляет в шаблон:
|
||||
|
||||
```python
|
||||
to_template = {
|
||||
"THIS_SERIA_ID": seria_id, # ID серии (210)
|
||||
"THIS_SERIA_NAME": q_seria.sName, # Название ("1-335")
|
||||
"THIS_SERIA_DESCRIPTION": html_description, # Верхняя статья
|
||||
"FLAP_DIM": flap_dimensions, # Массив схем открывания
|
||||
"DATA4GRAPH": graph_data, # Годы и кол-во домов
|
||||
"DATA4GEO": geo_data, # Координаты зданий
|
||||
"ACCOUNTS": buildings_count, # Кол-во квартир
|
||||
"APARTMENTS": families_count, # Кол-во семей
|
||||
"RESIDENTIAL_M2": total_area, # Площадь жилая
|
||||
# ... и другие
|
||||
}
|
||||
```
|
||||
|
||||
### Для шаблонов генерации кеша:
|
||||
|
||||
Каждый template файл получает **все** эти переменные и рендерится отдельно.
|
||||
|
||||
### Для main-шаблона:
|
||||
|
||||
Main-шаблон получает пути только к 3 кеш-файлам:
|
||||
|
||||
```python
|
||||
to_template.update({
|
||||
"PRE_RENDERED_STATIC_FLAPS_PATH": "seria_info/prepared/210_id_static_flaps.html",
|
||||
"PRE_RENDERED_STATIC_GRAPH_PATH": "seria_info/prepared/210_id_static_graph.html",
|
||||
"PRE_RENDERED_STATIC_MAP_STATS_PATH": "seria_info/prepared/210_id_static_map_stats.html",
|
||||
})
|
||||
|
||||
# Верхняя статья всегда передается как THIS_SERIA_DESCRIPTION и рендерится динамически
|
||||
to_template.update({
|
||||
"THIS_SERIA_DESCRIPTION": html_description, # Из БД, не кешируется
|
||||
})
|
||||
```
|
||||
|
||||
Main-шаблон использует `{% include %}` для 3 кеш-файлов, а верхнюю статью рендерит напрямую: `{{ THIS_SERIA_DESCRIPTION|safe }}`.
|
||||
|
||||
---
|
||||
|
||||
## 🎯 Примеры использования
|
||||
|
||||
### Добавили новое здание в серию 210
|
||||
|
||||
```bash
|
||||
# Обновили БД через Django ORM/admin
|
||||
python manage.py shell
|
||||
>>> from oknardia.models import Building_Info
|
||||
>>> Building_Info.objects.create(...)
|
||||
|
||||
# Регенерируем кеш только этой серии
|
||||
python manage.py regenerate_seria_prerender --seria-id 210
|
||||
|
||||
# Пользователи видят новое здание на карте и в таблице
|
||||
```
|
||||
|
||||
### Обновили года ввода в эксплуатацию
|
||||
|
||||
```bash
|
||||
# Изменили данные
|
||||
python manage.py shell
|
||||
>>> from oknardia.models import Building_Info
|
||||
>>> Building_Info.objects.filter(...).update(...)
|
||||
|
||||
# Кеш станет невалидным — нужна регенерация
|
||||
python manage.py regenerate_seria_prerender --force
|
||||
|
||||
# Пользователи видят обновленный график
|
||||
```
|
||||
|
||||
### Добавили новое предложение (цену)
|
||||
|
||||
```bash
|
||||
# PriceOffer.objects.create(...) → система добавляет новое предложение
|
||||
# НЕ нужна регенерация!
|
||||
|
||||
# Таблица окон обновилась сама на следующем запросе
|
||||
# Пользователи видят новое предложение сразу
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 🐛 Отладка
|
||||
|
||||
### Проверить, какие кеш-файлы существуют:
|
||||
|
||||
```bash
|
||||
ls -lah oknardia/templates/seria_info/prepared/
|
||||
```
|
||||
|
||||
### Вручную удалить кеш (для тестирования):
|
||||
|
||||
```bash
|
||||
# Удалить кеш одной серии
|
||||
rm oknardia/templates/seria_info/prepared/210*.html
|
||||
|
||||
# Удалить ВСЕ кеш-файлы
|
||||
rm oknardia/templates/seria_info/prepared/*_id_static_*.html
|
||||
```
|
||||
|
||||
### Проверить содержимое кеш-файла:
|
||||
|
||||
```bash
|
||||
cat oknardia/templates/seria_info/prepared/210_id_static_graph.html | head -20
|
||||
cat oknardia/templates/seria_info/prepared/210_id_static_flaps.html | head -30
|
||||
cat oknardia/templates/seria_info/prepared/210_id_static_map_stats.html | head -50
|
||||
```
|
||||
|
||||
### Логирование:
|
||||
|
||||
В `catalog_series.py` логируем создание файлов:
|
||||
|
||||
```python
|
||||
logger.info(f"Cache created: {file_flaps}")
|
||||
logger.info(f"Cache created: {file_graph}")
|
||||
logger.info(f"Cache created: {file_map_stats}")
|
||||
# file_upper НЕ создается — верхняя статья рендерится динамически
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 📈 Производительность
|
||||
|
||||
### Без кеша (DEV-режим):
|
||||
|
||||
- ~2-3 сек на запрос (вычисляются geo, graph, flaps)
|
||||
- Каждый запрос трогает БД
|
||||
- Удобно для разработки
|
||||
|
||||
### С кешем (PROD-режим):
|
||||
|
||||
- ~100-300 мс на первый запрос (генерируется кеш)
|
||||
- ~50-100 мс на последующие (включаются файлы + динамическая таблица)
|
||||
- Статические данные из кеша (очень быстро)
|
||||
- Таблица окон кешируется на уровне DB-запроса (индексы работают)
|
||||
|
||||
---
|
||||
|
||||
## ✅ Чек-лист для администратора
|
||||
|
||||
При развертывании в production:
|
||||
|
||||
- [ ] Запустить `python manage.py regenerate_seria_prerender` (сгенерировать все 93 файла)
|
||||
- [ ] Проверить размеры файлов в `prepared/` (~100-200 KB всего)
|
||||
- [ ] Протестировать на локалхосте `DEBUG = False`
|
||||
- [ ] Проверить, что таблица окон обновляется при добавлении новых предложений
|
||||
- [ ] Настроить логирование создания кеша в продакшене
|
||||
|
||||
---
|
||||
|
||||
## 🔗 Близкие компоненты
|
||||
|
||||
- **Main-шаблон**: `oknardia/templates/seria_info/all_seria_info_pre_light.html`
|
||||
- **Динамическая таблица**: `oknardia/templates/seria_info/all_seria_info_pre_light_dynamic_include.html`
|
||||
- **View**: `oknardia/web/catalog_series.py::catalog_seria_info()`
|
||||
- **Management-команда**: `oknardia/web/management/commands/regenerate_seria_prerender.py`
|
||||
- **Тесты**: `oknardia/web/test_prices.py` (проверяет свежесть таблицы)
|
||||
|
||||
---
|
||||
|
||||
**Версия:** 2.0 (Двухуровневое кеширование)
|
||||
**Последнее обновление:** 2026-05-19
|
||||
**Статус:** ✅ Production-ready
|
||||
|
||||
88
Dockerfile
Normal file
88
Dockerfile
Normal file
@@ -0,0 +1,88 @@
|
||||
# =================================================
|
||||
# STAGE 1: Builder - Установка зависимостей
|
||||
# =================================================
|
||||
FROM python:3.12-slim AS builder
|
||||
|
||||
# Устанавливаем переменные окружения
|
||||
ENV PYTHONDONTWRITEBYTECODE=1
|
||||
ENV PYTHONUNBUFFERED=1
|
||||
ENV PIP_DEFAULT_TIMEOUT=100
|
||||
|
||||
# Устанавливаем Poetry
|
||||
RUN pip install --no-cache-dir --default-timeout=100 --retries 10 poetry poetry-plugin-export
|
||||
|
||||
# Создаем рабочую директорию
|
||||
WORKDIR /app
|
||||
|
||||
# Копируем только файлы зависимостей для кэширования этого слоя
|
||||
COPY pyproject.toml poetry.lock /app/
|
||||
|
||||
# Экспортируем lock-файл в requirements.txt и ставим зависимости через pip.
|
||||
# Это обычно быстрее и проще для Docker, чем полноценная установка через Poetry.
|
||||
RUN poetry export --format requirements.txt --without-hashes --with dev --output /tmp/requirements.txt \
|
||||
&& pip install --no-cache-dir --default-timeout=100 --retries 10 -r /tmp/requirements.txt
|
||||
|
||||
|
||||
# =================================================
|
||||
# STAGE 2: Final - Создание чистого и безопасного образа
|
||||
# =================================================
|
||||
FROM python:3.12-slim AS stage-final
|
||||
|
||||
# Устанавливаем переменные окружения
|
||||
ENV PYTHONDONTWRITEBYTECODE=1
|
||||
ENV PYTHONUNBUFFERED=1
|
||||
ENV DJANGO_SETTINGS_MODULE=oknardia.settings
|
||||
|
||||
|
||||
# Удалить: Для DEV окружения отключены создание непривилегированного пользователя
|
||||
# и все связанные с ним операции chown — контейнер запускается от root,
|
||||
# что избегает проблем с доступом к смонтированным томам (база, статика).
|
||||
# RUN addgroup --system app && adduser --system --ingroup app app
|
||||
|
||||
# Создаем рабочую директорию (где находится manage.py)
|
||||
WORKDIR /home/app/oknardia
|
||||
|
||||
# Копируем установленные Python-пакеты из builder-стадии
|
||||
COPY --from=builder /usr/local/lib/python3.12/site-packages /usr/local/lib/python3.12/site-packages
|
||||
|
||||
# Удалить для dev: Копируем исходный код проекта. В DEV хозяин files не критичен (root).
|
||||
# COPY --chown=1000:1000 . .
|
||||
# Копируем весь проект в /home/app (корень проекта)
|
||||
COPY . /home/app/
|
||||
|
||||
# Удалить для dev: Создание директорий и установка прав
|
||||
# RUN mkdir -p /nginx_configs_host/nginx && chown -R 1000:1000 /nginx_configs_host
|
||||
# RUN mkdir -p /home/app/web/public/staticfiles && chown -R 1000:1000 /home/app/web/public
|
||||
# RUN mkdir -p /home/app/web/public/media/_error && chown -R 1000:1000 /home/app/web/public/media
|
||||
# RUN mkdir -p /home/app/web/database && chown -R 1000:1000 /home/app/web/database
|
||||
|
||||
# Удалить для dev: USER 1000 — для DEV запускаем от root
|
||||
# USER 1000
|
||||
|
||||
|
||||
# Собираем статику
|
||||
# Используем dummy ключ, так как .env файла нет на этапе сборки
|
||||
# ВАЖНО: В DEV режиме collectstatic запускается в docker-compose command, а не при сборке,
|
||||
# чтобы избежать ошибок с недоступными файлами.
|
||||
# RUN SECRET_KEY=dummy python oknardia/manage.py collectstatic --noinput --clear
|
||||
|
||||
# Открываем порт
|
||||
EXPOSE 8000
|
||||
|
||||
# Проверка здоровья контейнера
|
||||
# Docker будет периодически проверять, жив ли контейнер, отправляя GET запрос к главной странице.
|
||||
# Параметры:
|
||||
# --interval=30s - проверка каждые 30 секунд
|
||||
# --timeout=3s - ожидаем ответ максимум 3 секунды
|
||||
# --start-period=10s - даем контейнеру 10 секунд на запуск перед первой проверкой
|
||||
# --retries=3 - объявляем контейнер unhealthy после 3 неудачных попыток
|
||||
HEALTHCHECK --interval=30s --timeout=3s --start-period=10s --retries=3 \
|
||||
CMD python -c "import urllib.request; urllib.request.urlopen('http://localhost:8000/').read()" || exit 1
|
||||
|
||||
# Переходим в директорию с manage.py для корректного запуска Django
|
||||
WORKDIR /home/app/oknardia
|
||||
|
||||
# Команда запуска по умолчанию (для продакшена).
|
||||
# В DEV режиме используется runserver через docker-compose.local.yml,
|
||||
# который автоматически отдаёт статику и имеет auto-reload.
|
||||
CMD ["python", "-m", "gunicorn", "--workers", "2", "--bind", "0.0.0.0:8000", "oknardia.wsgi:application"]
|
||||
939
MANAGEMENT_RUNBOOK.md
Normal file
939
MANAGEMENT_RUNBOOK.md
Normal file
@@ -0,0 +1,939 @@
|
||||
# MANAGEMENT_RUNBOOK.md
|
||||
|
||||
Единый runbook по management-командам проекта.
|
||||
|
||||
Документ отвечает на 3 вопроса:
|
||||
- что запускать;
|
||||
- когда запускать;
|
||||
- как безопасно откатываться/повторять запуск.
|
||||
|
||||
## Каталог команд
|
||||
|
||||
1. `regenerate_seria_roots` — пересчет корневых серий (иерархия и консолидация).
|
||||
2. `generate_map_js` — генерация JavaScript для карт с геоданными зданий.
|
||||
3. `generate_sitemaps` — оффлайн генерация sitemap-файлов.
|
||||
4. `regenerate_seria_prerender` — оффлайн пересборка pre-render шаблонов для `catalog_seria_info`.
|
||||
5. `populate_seo_fields` — автозаполнение SEO-полей блога из существующих данных.
|
||||
6. `make_rating` — пересчёт рейтингов профилей и стеклопакетов методом Манна-Уитни.
|
||||
|
||||
## Общие правила запуска
|
||||
|
||||
- Запускать команды из корня репозитория.
|
||||
- Для локального/CI запуска использовать `poetry`.
|
||||
- Не запускать тяжелые операции через HTTP-эндпоинты `/service/*`.
|
||||
- Перезапуск веб-сервера (`gunicorn`/`uWSGI`) делать отдельным шагом оркестрации, а не из кода Django.
|
||||
|
||||
Базовый шаблон запуска:
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py <command> [args]
|
||||
```
|
||||
|
||||
## 1) Команда `regenerate_seria_roots`
|
||||
|
||||
Назначение:
|
||||
- пересчитать корневые серии (root) для всей иерархии серий домов.
|
||||
- консолидировать различные написания одной и той же серии в одну "корневую" серию.
|
||||
|
||||
### Контекст
|
||||
|
||||
На разных источниках данные о типовых сериях домов были записаны с разными орфографическими вариантами.
|
||||
Например серия **II-57** могла быть обозначена как:
|
||||
- `2-57` (цифра вместо кириллицы)
|
||||
- `И-57` (кириллица И вместо латинской II)
|
||||
- `П-57` (опечатка)
|
||||
- и т.п.
|
||||
|
||||
При парсинге данных эти варианты могли оказаться в БД как отдельные серии, хотя на самом деле это одна и та же серия.
|
||||
Функция `regenerate_seria_roots` связывает все эти варианты (алиасы) с одной **корневой** серией.
|
||||
|
||||
### Когда это нужно
|
||||
|
||||
При добавлении новых адресов и серий типового строительства, при ручном редактировании иерархии серий в админке,
|
||||
при загрузке новых данных с разными орфографическими вариантами серий. Кроме того, если для уже существующих в базе
|
||||
серий будут получены данные о типовых размерах оконных проёмов и типов квартирах (сейчас в базе около 4500 серий, из
|
||||
них всего 31 корневых серий и 1957 серий с найденным корнем... ещё 2502 неописанных серии без корня, т.е. больше
|
||||
половины, могут пополнить каталог Окнардии.
|
||||
|
||||
### Как это работает
|
||||
|
||||
1. **Этап 1**: Находит "корневые" серии (root series) — те, что реально используются в таблице `Apartment_Type`
|
||||
(т.е. у которых есть квартиры). Для каждой такой серии устанавливает `kRoot_id = own_id`.
|
||||
|
||||
2. **Этап 2**: Для всех остальных серий:
|
||||
- Движется вверх по дереву иерархии (`kParent_id`)
|
||||
- Ищет корневую серию (ту, которая либо не имеет родителя, либо она в списке корневых)
|
||||
- Устанавливает найденную корневую серию в поле `kRoot_id`
|
||||
- Если не находит корневую серию → `kRoot_id = None`
|
||||
|
||||
### Пример структуры после обработки
|
||||
|
||||
```
|
||||
Таблица: Seria_Info
|
||||
|
||||
id | sSeriaName | kParent_id | kRoot_id | Комментарий
|
||||
----|------------|------------|-----------|-------------------
|
||||
123 | II-57 | NULL | 123 | Корневая серия
|
||||
124 | 2-57 | 123 | 123 | Алиас (через родителя)
|
||||
125 | И-57 | 123 | 123 | Алиас (через родителя)
|
||||
126 | П-57 | 123 | 123 | Алиас (через родителя)
|
||||
```
|
||||
|
||||
### Базовый запуск
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py regenerate_seria_roots
|
||||
```
|
||||
|
||||
### Параметры запуска
|
||||
|
||||
**`--verbosity 0`** — только ошибки (минимум информации):
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py regenerate_seria_roots --verbosity 0
|
||||
```
|
||||
|
||||
**`--verbosity 1`** — точки/плюсы (стандартный режим):
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py regenerate_seria_roots
|
||||
# или явно
|
||||
poetry run python oknardia/manage.py regenerate_seria_roots --verbosity 1
|
||||
```
|
||||
|
||||
**`--verbosity 2`** — подробный вывод (названия серий):
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py regenerate_seria_roots --verbosity 2
|
||||
```
|
||||
|
||||
**`--verbosity 3`** — очень подробный вывод в виде таблицы:
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py regenerate_seria_roots --verbosity 3
|
||||
```
|
||||
|
||||
### Примеры вывода
|
||||
|
||||
**Verbosity 0 (только ошибки - чистый вывод):**
|
||||
```
|
||||
✅ Пересчет завершен! Время: 2.18с
|
||||
```
|
||||
|
||||
**Verbosity 1 (магический режим - точки и символы):**
|
||||
```
|
||||
=== ПЕРЕСЧЕТ КОРНЕВЫХ СЕРИЙ ===
|
||||
Этап 1: Ищем корневые серии в таблице квартир...
|
||||
✓ Найдено корневых серий: 241
|
||||
...............................
|
||||
|
||||
Этап 2: Главная магия - обрабатываем все серии в иерархии...
|
||||
-----++..--.++--...--.+.++++.-+++++++-.-.++.-+--++--++++-+++++++-++.+--.+++-+-..++++++-++++++++++--.-++++++-++++
|
||||
+--+-+++++++++-++++++++++-++++--+++.+++--+++++++++++++++++++---++.+-+-+++++-++++++++++-+++-+----+.-+++-+--++++++
|
||||
+++----+--+-+++++-+--+++--+++-.+++++-++++++++-+---++-+++++---+++------++----++-+--++----+--++--++++--+++++++++++
|
||||
+++++++-------++++---+++-[... очень много символов ...]
|
||||
|
||||
=== РЕЗУЛЬТАТЫ ===
|
||||
✓ Корневых серий (обработаны на этапе 1): 31
|
||||
✓ Серий с найденным корнем: 1957
|
||||
⚠ Серий без корня: 2502
|
||||
|
||||
✅ Пересчет завершен! Время: 2.18с
|
||||
```
|
||||
|
||||
**Легенда магического режима:**
|
||||
- `.` = корневая серия (обработана на этапе 1)
|
||||
- `+` = серия с найденным корнем
|
||||
- `-` = серия без найденного корня
|
||||
- `E` = ошибка при обработке
|
||||
|
||||
**Verbosity 2 (подробный):**
|
||||
```
|
||||
=== ПЕРЕСЧЕТ КОРНЕВЫХ СЕРИЙ ===
|
||||
Этап 1: Ищем корневые серии в таблице квартир...
|
||||
✓ Найдено корневых серий: 241
|
||||
✓ 0008 П-44
|
||||
✓ 0009 П-3
|
||||
✓ 0012 II-49
|
||||
✓ 0017 КОПЭ
|
||||
...
|
||||
|
||||
Этап 2: Главная магия - обрабатываем все серии в иерархии...
|
||||
0001: Нет корня
|
||||
0002: Нет корня
|
||||
0006: корень → 12
|
||||
0007: корень → 9
|
||||
0008: корневая
|
||||
...
|
||||
```
|
||||
|
||||
**Verbosity 3 (очень подробный - таблица):**
|
||||
```
|
||||
=== ПЕРЕСЧЕТ КОРНЕВЫХ СЕРИЙ ===
|
||||
Этап 1: Ищем корневые серии в таблице квартир...
|
||||
✓ Найдено корневых серий: 241
|
||||
✓ 0008 | П-44 | корневая
|
||||
✓ 0009 | П-3 | корневая
|
||||
✓ 0012 | II-49 | корневая
|
||||
...
|
||||
|
||||
Этап 2: Главная магия - обрабатываем все серии в иерархии...
|
||||
--------------------------------------------------------------------------------------------------------------
|
||||
ID | Название | Родитель | Путь | Результат
|
||||
--------------------------------------------------------------------------------------------------------------
|
||||
...
|
||||
...
|
||||
2565 | Г-ЗИ | 9 | 9 | ✓ Корень #9
|
||||
2566 | I-528КП-809/69 | - | (нет) | ✗ Нет корня
|
||||
2567 | 464Д-0154 | 3339 | 3339 → 963 → 375 | ✓ Корень #375
|
||||
2568 | УЛГ-507-4/64 | 2105 | 2105 | ✓ Корень #2105
|
||||
2569 | ЛГ-507-4 | 2105 | 2105 | ✓ Корень #2105
|
||||
2570 | 1ЛГ-600-И-1 | - | (нет) | ✗ Нет корня
|
||||
2571 | 464Д-0154 Новополоцкого ДСК | 3339 | 3339 → 963 → 375 | ✓ Корень #375
|
||||
2572 | 121-0142,13,87 | - | (нет) | ✗ Нет корня
|
||||
...
|
||||
...
|
||||
--------------------------------------------------------------------------------------------------------------
|
||||
|
||||
=== РЕЗУЛЬТАТЫ ===
|
||||
✓ Корневых серий (обработаны на этапе 1): 31
|
||||
✓ Серий с найденным корнем: 1957
|
||||
⚠ Серий без корня: 2502
|
||||
|
||||
✅ Пересчет завершен! Время: 2.18с
|
||||
```
|
||||
|
||||
### Когда запускать
|
||||
|
||||
- **После первого развертывания** — консолидировать иерархию серий.
|
||||
- **После ручного редактирования иерархии** (добавления родитель-потомков в админку).
|
||||
- **После загрузки новых данных** с разными орфографическими вариантами серий.
|
||||
- **По расписанию** (опционально, например раз в месяц):
|
||||
```bash
|
||||
0 2 * * 1 cd /home/user/app-path/2022-oknardia && poetry run python oknardia/manage.py regenerate_seria_roots >> /var/log/oknardia-seria-roots.log 2>&1
|
||||
```
|
||||
|
||||
### Откат и безопасность
|
||||
|
||||
- **Безопасна для повторного запуска** — просто пересчитывает все `kRoot_id`.
|
||||
- **Откат через SQL** — если нужно очистить поле (перед запуском рекомендуется бэкап):
|
||||
```sql
|
||||
UPDATE oknardia_seria_info SET kRoot_id = NULL;
|
||||
```
|
||||
- **Проверка результатов** — после запуска можно проверить:
|
||||
```bash
|
||||
poetry run python oknardia/manage.py shell -c "
|
||||
from oknardia.models import Seria_Info
|
||||
count_null = Seria_Info.objects.filter(kRoot_id__isnull=True).count()
|
||||
count_with_root = Seria_Info.objects.filter(kRoot_id__isnull=False).count()
|
||||
print(f'Серий без корня: {count_null}')
|
||||
print(f'Серий с корнем: {count_with_root}')
|
||||
"
|
||||
```
|
||||
|
||||
## 2) Команда `generate_map_js`
|
||||
|
||||
Назначение:
|
||||
- сгенерировать JavaScript-файл для отрисовки карты всех зданий типовых серий в Яндекс.Картах.
|
||||
- файл содержит геоданные (latitude/longitude), ID адресов, привязку к сериям и информацию для balloon-окон на картах.
|
||||
|
||||
### Что происходит
|
||||
|
||||
1. **Сбор геоданных** — для всех корневых серий (где `id = kRoot_id`)
|
||||
- Запрашиваются здания из таблицы `Building_Info` с non-zero координатами
|
||||
- Для каждого здания собирается: широта, долгота, ID адреса, адрес в латинице, ID серии
|
||||
|
||||
2. **Генерация JavaScript** — на основе шаблона `service/JavaScript4AllSeriaMap.js.html`
|
||||
- Генерируется массив цветов для каждой серии
|
||||
- Объявляются переменные с ID и названиями серий
|
||||
- Инициализируется Yandex.Maps с PlaceMarks для каждого здания
|
||||
|
||||
3. **Минификация через Terser** — уменьшение размера JavaScript
|
||||
- Удаляются ненужные пробелы и переносы строк
|
||||
- Сокращаются имена переменных (mangling)
|
||||
- Удаляются console.log и debugger
|
||||
|
||||
4. **Запись в файлы**:
|
||||
- `public/static/js/4maps/_ALL_seria_on_map.js` — исходный форматированный файл (715 KB)
|
||||
- `public/static/js/4maps/_ALL_seria_on_map.mini.js` — минифицированный файл (639 KB)
|
||||
|
||||
### Оптимизация размера
|
||||
|
||||
Файл был оптимизирован в три этапа:
|
||||
|
||||
| Этап | Размер | Сжатие |
|
||||
|------|--------|--------|
|
||||
| Исходный (2016 год) | 2.5 MB | — |
|
||||
| **Уровень 1**: функция-фабрика `m()` | 715 KB | **71%** |
|
||||
| **Уровень 2**: Terser минификация | 639 KB | +10.6% |
|
||||
| **Уровень 3**: Gzip в браузере | 188 KB | +29.4% |
|
||||
| **Итого сжатие** | **188 KB** | **92.5%** |
|
||||
|
||||
> **Примечание**: Gzip применяется автоматически браузером и веб-сервером при наличии в заголовках `Content-Encoding: gzip`
|
||||
|
||||
Содержимое:
|
||||
- **Маркеры на карте**: 18,228 зданий
|
||||
- **Серии с цветами**: 31
|
||||
- **Корневые серии**: 31
|
||||
|
||||
### Базовый запуск
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py generate_map_js
|
||||
```
|
||||
|
||||
### Параметры запуска
|
||||
|
||||
**`--force`** — пересгенерировать файл (перезаписать если существует):
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py generate_map_js --force
|
||||
```
|
||||
|
||||
**`--verbosity 2`** — подробный вывод со статистикой:
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py generate_map_js --verbosity 2
|
||||
```
|
||||
|
||||
### Когда запускать
|
||||
|
||||
- **После первого развертывания** — создать файл карты один раз.
|
||||
- **После добавления новых зданий** в БД (через парсеры или импорт).
|
||||
- **По расписанию** (опционально, если здания редко добавляются):
|
||||
```bash
|
||||
0 3 * * 0 cd /home/user/app-path/2022-oknardia && poetry run python oknardia/manage.py generate_map_js >> /var/log/oknardia-map-js.log 2>&1
|
||||
```
|
||||
|
||||
### Пример вывода
|
||||
|
||||
```
|
||||
=== ГЕНЕРАЦИЯ JAVASCRIPT ДЛЯ КАРТ ===
|
||||
|
||||
Этап 1: Сбор информации о корневых сериях...
|
||||
✓ Найдено корневых серий: 31
|
||||
|
||||
Этап 2: Генерация единого JS-файла для ВСЕ серий...
|
||||
✓ Написан исходный файл: _ALL_seria_on_map.js
|
||||
Размер: 734.0 KB
|
||||
|
||||
Этап 3: Минификация JavaScript (rjsmin)...
|
||||
[*] Минификация успешна!
|
||||
Исходный файл: 734.015 KB
|
||||
Минифицированный: 732.952 KB
|
||||
Сжатие: 0.14%
|
||||
Время: 0.0017с
|
||||
[i] Полная статистика по сериям:
|
||||
- Жилых м²: 125,749,341
|
||||
- Муниципальных м²: 11,302,860
|
||||
- Жильцов: 6,342,742
|
||||
- Квартир: 2,769,800
|
||||
|
||||
=== РЕЗУЛЬТАТЫ ===
|
||||
✓ Серий обработано: 31
|
||||
✓ Зданий на карте: 18228
|
||||
✓ JS-файлов создано: 2 (исходный + минифицированный)
|
||||
✓ Исходный файл: _ALL_seria_on_map.js
|
||||
✓ Минифицированный: _ALL_seria_on_map.mini.js
|
||||
✓ Обфускация: Base64 кодирование координат
|
||||
|
||||
[OK] Генерация завершена! Время: 1.10с
|
||||
```
|
||||
|
||||
## 3) Команда `generate_sitemaps`
|
||||
|
||||
Назначение:
|
||||
- пересобрать `sitemap.xml` и chunk-файлы в `MEDIA_ROOT/_serv_sitemap`.
|
||||
|
||||
Базовый запуск:
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py generate_sitemaps
|
||||
```
|
||||
|
||||
Запуск с параметрами:
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py generate_sitemaps \
|
||||
--compare-min-depth 2 \
|
||||
--compare-max-depth 4 \
|
||||
--max-items 40000 \
|
||||
--max-file-size 5242880 \
|
||||
--max-files-qty 998
|
||||
```
|
||||
|
||||
Когда запускать:
|
||||
- после деплоя;
|
||||
- по расписанию (cron/systemd timer);
|
||||
- после крупных изменений данных каталога/блога.
|
||||
|
||||
### Важные замечания
|
||||
|
||||
Чтобы `sitemap.xml` отдавал прокси-nginx напрямую из файловой системы, нужно, чтобы он физически лежал
|
||||
в `MEDIA_ROOT/_serv_sitemap/sitemap.xml`.
|
||||
|
||||
Допустимо, что файл доступен по двум URL (корневой и media), но в `robots.txt` должен быть указан один
|
||||
канонический вариант `sitemap.xml`
|
||||
|
||||
#### NGINX snippet (alias для корневого sitemap)
|
||||
|
||||
```nginx
|
||||
# Корневой sitemap.xml (для привычного для поисковиков URL)
|
||||
location = /sitemap.xml {
|
||||
alias /<путь-к-каталогку-с-докер-приложением>/media/_serv_sitemap/sitemap.xml;
|
||||
default_type application/xml;
|
||||
add_header Cache-Control "public, max-age=300";
|
||||
}
|
||||
```
|
||||
|
||||
## 4) Команда `regenerate_seria_prerender`
|
||||
|
||||
Назначение:
|
||||
- пересобрать pre-render шаблоны для **статических данных** страниц серий (`catalog_seria_info`) в каталоге `seria_info/prepared/`.
|
||||
- генерирует **3 типа файлов** для каждой серии (верхняя статья НЕ кешируется, она рендерится динамически из БД):
|
||||
- `{seria_id}_id_static_flaps.html` — схемы открывания
|
||||
- `{seria_id}_id_static_graph.html` — график ввода в эксплуатацию
|
||||
- `{seria_id}_id_static_map_stats.html` — карта + статистика
|
||||
|
||||
### Контекст: двухуровневое кеширование
|
||||
|
||||
Система использует двухуровневое кеширование:
|
||||
- **Статический кеш** — дорогостоящие операции (геокоординаты, графики, схемы). Генерируются один раз и сохраняются на диск.
|
||||
- **Динамические данные** — верхняя статья про серию (может редактироваться через админку, видно БЕЗ перезагрузки контейнера) и таблица оконных проёмов (показывает актуальные предложения).
|
||||
|
||||
Подробнее: см. [`CACHE_PRERENDER_SYSTEM.md`](CACHE_PRERENDER_SYSTEM.md)
|
||||
|
||||
### Проверка без записи файлов (dry-run):
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py regenerate_seria_prerender --dry-run
|
||||
```
|
||||
|
||||
Пересборка только отсутствующих файлов (стандартный запуск):
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py regenerate_seria_prerender
|
||||
```
|
||||
|
||||
Вывод:
|
||||
```
|
||||
OK seria 210: 3 кеш-файла созданы
|
||||
OK seria 100: 3 кеш-файла созданы
|
||||
...
|
||||
Готово. Обработано: 31. Создано/пересоздано: 31 × 3 файла. Пропущено: 0.
|
||||
```
|
||||
|
||||
Принудительная пересборка всех root-серий (даже если кеш существует):
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py regenerate_seria_prerender --force
|
||||
```
|
||||
|
||||
Выборочная пересборка (конкретные серии):
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py regenerate_seria_prerender --seria-id 843 --seria-id 2100 --force
|
||||
```
|
||||
|
||||
### Когда запускать
|
||||
|
||||
- **После первого развертывания** — сгенерировать кеш всех 31 серии один раз.
|
||||
- **После обновления логики `catalog_seria_info`** — изменились параметры графика или карты.
|
||||
- **После изменения координат зданий** — geo-данные обновлены.
|
||||
- **После добавления новых зданий в серию** — карта и список зда<D0B4><D0B0>ий изменились.
|
||||
- **По расписанию** (опционально, если данные по геокоординатам обновляются):
|
||||
```bash
|
||||
0 3 * * 0 cd /Users/e-serg/PRJ/2022-oknardia && poetry run python oknardia/manage.py regenerate_seria_prerender >> /var/log/oknardia-prerender.log 2>&1
|
||||
```
|
||||
|
||||
### Когда НЕ нужна регенерация
|
||||
|
||||
- **При добавлении новых предложений/цен** — таблица окон обновляется при каждом запросе автоматически.
|
||||
- **При редактировании верхней статьи через админку** — она рендерится динамически, кешируется НЕ нужно.
|
||||
- **При изменении наличия/статуса профилей** — рейтинги пересчитываются запросом через `make_rating`.
|
||||
|
||||
## 5) Команда `populate_seo_fields`
|
||||
|
||||
Назначение:
|
||||
- автозаполнить SEO-поля (`sSlug`, `sMetaDescription`, `sMetaKeywords`) для всех существующих записей блога.
|
||||
|
||||
Используется:
|
||||
- при первом развертывании новой версии с автогенерацией SEO-полей;
|
||||
- при восстановлении из бэкапа где SEO-поля пусты;
|
||||
- при изменении логики автогенерации (с флагом `--force`).
|
||||
|
||||
### Базовый запуск
|
||||
|
||||
Заполнить только пустые SEO-поля (стандартный вариант):
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py populate_seo_fields
|
||||
```
|
||||
|
||||
### Параметры запуска
|
||||
|
||||
**`--dry-run`** — только показать что будет сделано (без сохранения в БД):
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py populate_seo_fields --dry-run
|
||||
```
|
||||
|
||||
**`--force`** — переполнить ВСЕ SEO-поля, даже уже заполненные:
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py populate_seo_fields --force
|
||||
```
|
||||
|
||||
**`--clean`** — очистить все SEO-поля перед заполнением (для переделки):
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py populate_seo_fields --clean
|
||||
```
|
||||
|
||||
**Комбинация флагов** — сухой прогон переполнения всех полей:
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py populate_seo_fields --dry-run --force
|
||||
```
|
||||
|
||||
### Что заполняется
|
||||
|
||||
| Поле | Источник | Результат |
|
||||
|------|----------|-----------|
|
||||
| `sSlug` | `sPostHeader` | URL-безопасный слаг (max 200 символов) |
|
||||
| `sMetaDescription` | `sPostContent` | Первые 160 символов (исключая теги `<cut>`) |
|
||||
| `sMetaKeywords` | `sPostHeader` | Заголовок + префикс "oknardia, окнардия, блог, публикация" (max 256 символов) |
|
||||
|
||||
Пример результата:
|
||||
|
||||
```python
|
||||
sPostHeader = "Профиль Brusbox Super Aero"
|
||||
↓
|
||||
sSlug = "profil-brusbox-super-aero"
|
||||
sMetaDescription = "brusbox-super-aero-pyatikamernaya-profil-sistema..."
|
||||
sMetaKeywords = "oknardia, окнардия, блог, публикация, Профиль Brusbox Super Aero"
|
||||
```
|
||||
|
||||
### Когда запускать
|
||||
|
||||
- **После первого развертывания** — заполнить SEO-поля всех 29 существующих постов одной командой.
|
||||
- **Один раз** — команда идемпотентна (при повторном запуске не будет ничего менять, т.к. пустые поля остатся).
|
||||
- **При изменении логики** — использовать `--clean --force` для полной переделки всех SEO-полей.
|
||||
|
||||
### Пример полного сценария
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
|
||||
# Шаг 1: Проверить что будет заполнено
|
||||
poetry run python oknardia/manage.py populate_seo_fields --dry-run
|
||||
|
||||
# Шаг 2: Если результат устраивает — запустить реально
|
||||
poetry run python oknardia/manage.py populate_seo_fields
|
||||
|
||||
# Шаг 3: Проверить что заполнилось
|
||||
poetry run python oknardia/manage.py shell -c "
|
||||
from oknardia.models import BlogPosts
|
||||
posts = BlogPosts.objects.all()
|
||||
print(f'Пусто sSlug: {posts.filter(sSlug=\"\").count()}')
|
||||
print(f'Пусто sMetaDescription: {posts.filter(sMetaDescription=\"\").count()}')
|
||||
print(f'Пусто sMetaKeywords: {posts.filter(sMetaKeywords=\"\").count()}')
|
||||
"
|
||||
```
|
||||
|
||||
### Возвращаемая информация
|
||||
|
||||
```
|
||||
======================================================================
|
||||
ИТОГОВЫЙ ОТЧЕТ
|
||||
======================================================================
|
||||
|
||||
✓ sSlug заполнено: 28 раз
|
||||
✓ sMetaDescription заполнено: 28 раз
|
||||
✓ sMetaKeywords заполнено: 28 раз
|
||||
✓ Записей обновлено в БД: 28
|
||||
✗ Ошибок при обработке: 0
|
||||
|
||||
Обновлено 28 записей успешно!
|
||||
```
|
||||
|
||||
### Откат и безопасность
|
||||
|
||||
- **Безопасна для повторного запуска** — пустые поля не изменяются при повторной работе.
|
||||
- **Откат через SQL** — если нужно очистить, используй: `UPDATE oknardia_blogposts SET sSlug='', sMetaDescription='', sMetaKeywords='';`
|
||||
- **Всегда используй `--dry-run`** перед первым запуском для проверки.
|
||||
|
||||
## 6) Команда `make_rating`
|
||||
|
||||
Назначение:
|
||||
- пересчитать рейтинги оконных профилей, стеклопакетов и наборов услуг используя адаптированный метод Манна-Уитни (Mann-Whitney U Step Rank).
|
||||
- сохранить результаты в поля `fProfileRating`, `fGlazingRating`, `fSetRating` (0.0 … 5.0 звёзд).
|
||||
- заполнить JSON-состав рейтинга (детальный разбор по каждому параметру) в поля `sProfileDescription`, `sGlazingDescription`, `sSetDescription`.
|
||||
- алгоритм рассчитывает три этапа ранжирования: профили → стеклопакеты → наборы (которые зависят от профилей и стеклопакетов).
|
||||
|
||||
### Базовый запуск
|
||||
|
||||
Пересчитать рейтинги всех профилей и стеклопакетов (стандартный режим):
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py make_rating
|
||||
```
|
||||
|
||||
### Параметры запуска
|
||||
|
||||
**`--verbosity 0`** — минимум информации (только ошибки):
|
||||
**`--verbosity 1`** — стандартная информация (по умолчанию):
|
||||
**`--verbosity 3`** — очень подробный вывод (для отладки, для каждого профиля/стеклопакета таблица):
|
||||
|
||||
Пример использования с параметром `--verbosity`:
|
||||
|
||||
```bash
|
||||
poetry run python oknardia/manage.py make_rating --verbosity 3 | head -500
|
||||
```
|
||||
|
||||
### АЛГОРИТМ: Метод Манна-Уитни (Mann-Whitney U Step Rank)
|
||||
|
||||
Команда использует адаптированный вариант критерия Манна-Уитни для ранжирования параметров качества оконных
|
||||
предложений и комопнентов (профилей, стеклопакетов, наборов услуг) на основе их технических характеристик
|
||||
и популярности у поставщиков.
|
||||
|
||||
#### Как это работает:
|
||||
|
||||
1. **Сортировка объектов** по одному параметру (например, по теплопередаче):
|
||||
- Профиль A: 0.60 Ro → ранг = 0.0
|
||||
- Профиль B: 0.60 Ro → ранг = 0.0 (то же значение, ранг не меняется)
|
||||
- Профиль C: 0.80 Ro → ранг = 1.0 (новое значение, добавляем вес параметра)
|
||||
- Профиль D: 0.95 Ro → ранг = 2.0 (ещё новое значение)
|
||||
|
||||
2. **Направление ранжирования** определяется флагом `revers`:
|
||||
- `revers=False` — **БОЛЬШЕ = ЛУЧШЕ** (например, теплопередача, звукоизоляция)
|
||||
- `revers=True` — **МЕНЬШЕ = ЛУЧШЕ** (например, высота в проёме для прочности)
|
||||
|
||||
3. **Нормализация рангов** к диапазону 0.0 … 1.0:
|
||||
- Профиль A: 0.0 / 2.0 = 0.0
|
||||
- Профиль B: 0.0 / 2.0 = 0.0
|
||||
- Профиль C: 1.0 / 2.0 = 0.5
|
||||
- Профиль D: 2.0 / 2.0 = 1.0
|
||||
|
||||
4. **Суммирование рангов** по всем параметрам:
|
||||
- TmpRating = Σ(ранг_параметра × вес_параметра)
|
||||
|
||||
5. **Преобразование в звёзды** (0.0 … 5.0):
|
||||
- ТmpRating нормализуется к 0..1
|
||||
- Умножается на 5.0 для получения финального рейтинга
|
||||
|
||||
#### Пример итогового рейтинга профиля:
|
||||
|
||||
```
|
||||
Профиль "Brusbox Super Aero"
|
||||
Теплопередача: 0.60 Ro (ранг 0.9, вес 1.0)
|
||||
Звукоизоляция: 33 дБ (ранг 0.8, вес 1.0)
|
||||
Высота в проёме: 112 мм (ранг 0.6, вес 0.3)
|
||||
Количество камер: 6 шт (ранг 0.7, вес 0.1)
|
||||
|
||||
Итого: (0.9×1.0 + 0.8×1.0 + 0.6×0.3 + 0.7×0.1) / 2.3 ≈ 3.8 звёзд ⭐⭐⭐⭐
|
||||
```
|
||||
|
||||
### ПРОФИЛИ: какие параметры учитываются
|
||||
|
||||
| № | Параметр | Поле БД | ЛУЧШЕ | Вес | Описание |
|
||||
|---|----------------------|----------------------------|--------------------|-----|-----------------------------------------------------|
|
||||
| 1 | Звукоизоляция | `fProfileSoundproofing` | БОЛЬШЕ дБ | 1.0 | Сопротивление шуму (дБ) |
|
||||
| 2 | Теплопередача | `fProfileHeatTransf` | БОЛЬШЕ Ro | 1.0 | Сопротивление теплопередаче (м²×°C/Вт) |
|
||||
| 3 | Высота в проёме | `iProfileHeight` | МЕНЬШЕ мм | 0.3 | Видимая высота в световом проёме (экономия) |
|
||||
| 4 | Высота фальца | `iProfileRabbet` | БОЛЬШЕ мм | 0.2 | Глубина фальца для герметизации |
|
||||
| 5 | Толщина стеклопакета | `iProfileGlazingThickness` | БОЛЬШЕ мм | 0.2 | Максимальная толщина стеклопакета |
|
||||
| 6 | Толщина профиля | `iProfileThickness` | БОЛЬШЕ мм | 0.2 | Монтажная (боковая) ширина профиля |
|
||||
| 7 | Контуры уплотнения | `fProfileSeals` | БОЛЬШЕ контуров | 1.2 | Количество контуров уплотнения |
|
||||
| 8 | Количество камер | `iProfileCameras` | БОЛЬШЕ шт | 0.1 | Число камер в профиле (из рамки + створки) |
|
||||
| 9 | Популярность | `NumOffer` | БОЛЬШЕ предложений | 0.1 | Используется ли профиль в коммерческих предложениях |
|
||||
|
||||
**Примеры интерпретации:**
|
||||
- Профиль с рейтингом **5.0 ⭐⭐⭐⭐⭐**: отличная теплопередача + звукоизоляция + много камер + многоконтурные
|
||||
уплотнения.
|
||||
- Профиль с рейтингом **2.0 ⭐⭐**: среднее качество, слабые характеристики.
|
||||
- Профиль с рейтингом **0.5 ⭐**: слабые характеристики или производить не предоставил данных и их нет в отрытых источниках.
|
||||
|
||||
### СТЕКЛОПАКЕТЫ: какие параметры учитываются
|
||||
|
||||
| № | Параметр | Поле БД | ЛУЧШЕ | Вес | Описание |
|
||||
|---|-------------------|-----------------------------|--------------|------|----------------------------------------------------------------------------------------|
|
||||
| 1 | Звукоизоляция | `fGlazingSoundproofing` | БОЛЬШЕ дБ | 1.0 | Звукоизоляционный коэффициент (дБ) |
|
||||
| 2 | Теплопередача | `fGlazingHeatTransfer` | БОЛЬШЕ Ro | 1.0 | Сопротивление теплопередаче (м²×°C/Вт) |
|
||||
| 3 | Светопропускание | `fGlazingLightTransmission` | БОЛЬШЕ % | 0.25 | Коэффициент пропускания видимого света (%), отражение света снаружи |
|
||||
| 4 | Солнцепропускание | `fGlazingPassingSun` | **МЕНЬШЕ %** | 0.15 | Коэффициент солнечного излучения (SHGC) — В России меньше = лучше для охлаждения летом |
|
||||
| 5 | Толщина | `iGlazingThickness` | БОЛЬШЕ мм | 0.1 | Общая толщина стеклопакета |
|
||||
| 6 | Количество камер | `iGlazingCamerasN` | БОЛЬШЕ шт | 0.1 | Число воздушных/аргоновых камер |
|
||||
|
||||
**Особенности стеклопакетов:**
|
||||
- **Светопропускание** = как много естественного света проходит в помещение (больше = лучше)
|
||||
- **Солнцепропускание** = как много солнечного тепла/излучения проходит (в России: меньше = лучше, потому что внутри есть отражающее напыление)
|
||||
- Двухкамерный (с аргоном) почти всегда лучше однокамерного
|
||||
- Трёхкамерные = премиум для холодного климата
|
||||
|
||||
**Примеры интерпретации:**
|
||||
- **5.0 ⭐⭐⭐⭐⭐**: трёхкамерный с хорошей теплопередачей, звукоизоляцией (обычно с аргоном и напылением).
|
||||
- **3.0 ⭐⭐⭐**: двухкамерный, среднее качество
|
||||
- **1.0 ⭐**: однокамерный старого образца или с плохими характеристиками
|
||||
|
||||
### НАБОРЫ: какие параметры учитываются
|
||||
|
||||
| № | Параметр | Поле БД | ЛУЧШЕ | Вес | Описание |
|
||||
|---|-------------------|--------------------------|------------------|-----|---------------------------------------------------|
|
||||
| 1 | Актуальность | `dModify` | МЕНЬШЕ (свежее) | 0.3 | Дата последнего обновления (timestamp) |
|
||||
| 2 | Доставка | `bSetDelivery` | ДА (1) | 0.8 | Включена ли доставка в стоимость |
|
||||
| 3 | Монтаж/демонтаж | `bSetUninstallInstall` | ДА (1) | 1.0 | Включены ли услуги монтажа и демонтажа |
|
||||
| 4 | Подоконник | `sSetSill` | ДА (1) | 0.5 | Включен ли подоконник |
|
||||
| 5 | Водоотлив | `sSetPanes` | ДА (1) | 0.8 | Включен ли водоотлив/козырёк |
|
||||
| 6 | Откос | `sSetSlope` | ДА (1) | 0.5 | Включены ли откосы |
|
||||
| 7 | Климат-контроль | `sSetClimateControl` | ДА (1) | 0.3 | Включено ли управление микроклиматом |
|
||||
| 8 | Число предложений | `NumOffer` | БОЛЬШЕ | 0.2 | Популярность набора (кол-во активных предложений) |
|
||||
| 9 | Гибкость скидок | `iDiscountVariantsCount` | БОЛЬШЕ вариантов | 0.5 | Кол-во вариантов скидок из формулы офиса |
|
||||
| 10| Размер скидок | `fDiscountMax` | БОЛЬШЕ % | 1.0 | Максимальная скидка из всех вариантов |
|
||||
|
||||
**ВАЖНО: Итоговый рейтинг набора состоит из трёх компонентов:**
|
||||
- Рейтинг параметров услуг (Актуальность, Доставка, Монтаж, Подоконник и т.д.)
|
||||
- Рейтинг входящего стеклопакета (ранжируется отдельно)
|
||||
- Рейтинг входящего профиля (ранжируется отдельно)
|
||||
|
||||
Формула итогового рейтинга набора (fSetRating):
|
||||
```
|
||||
k1 = нормализованный TmpRating (услуги) * вес услуг
|
||||
k2 = нормализованный рейтинг стеклопакета * RARING_WEIGHT_GLAZING_IN_SET (обычно 1.5)
|
||||
k3 = нормализованный рейтинг профиля * RARING_WEIGHT_PVC_PROFILE_IN_SET (обычно 1.5)
|
||||
|
||||
fSetRating = k1 + k2 + k3 (итого от 0.0 до 5.0 звёзд)
|
||||
```
|
||||
|
||||
**Примеры интерпретации:**
|
||||
- **5.0 ⭐⭐⭐⭐⭐**: набор с премиум компонентами (хороший профиль и стеклопакет) + полный пакет услуг (доставка, монтаж, подоконник, откос, климат-контроль) + значительные скидки.
|
||||
- **3.5 ⭐⭐⭐⭐**: хороший профиль/стеклопакет + базовые услуги (доставка, монтаж) + скромные скидки.
|
||||
- **2.0 ⭐⭐**: эконом компоненты или слабые услуги (нет доставки, нет откосов).
|
||||
- **1.0 ⭐**: минимальный пакет или устаревшие предложения (давно не обновлялись).
|
||||
|
||||
### Когда запускать
|
||||
|
||||
- **После первого развертывания** — заполнить рейтинги всех профилей, стеклопакетов и наборов.
|
||||
- **После изменения каталога** (добавление нового профиля/стеклопакета/набора).
|
||||
- **После уточнения характеристик** (например, поставщик предоставил новые данные).
|
||||
```bash
|
||||
poetry run python oknardia/manage.py make_rating
|
||||
```
|
||||
|
||||
- **По расписанию** (например, ежемесячно, чтобы пересчитать популярность):
|
||||
```bash
|
||||
30 2 * * 1 cd /home/user/app-path/2022-oknardia && poetry run python oknardia/manage.py make_rating >> /var/log/oknardia-rating.log 2>&1
|
||||
```
|
||||
- **После обновления весов** в `settings.py` (константы `RANK_PVCP_*`, `RANK_GLAZ_*`).
|
||||
|
||||
### Откат и безопасность
|
||||
|
||||
- **Безопасна для повторного запуска** — пересчитывает все рейтинги заново.
|
||||
- **Всегда обновляет только рейтинги** — другие данные в таблицах не меняются.
|
||||
- **Откат через SQL** — если нужно установить нулевые значения (перед запуском рекомендуется бэкап базы):
|
||||
```sql
|
||||
-- Очистить рейтинги профилей
|
||||
UPDATE oknardia_pvcprofiles SET fProfileRating = 0.0, sProfileDescription = '{}';
|
||||
|
||||
-- Очистить рейтинги стеклопакетов
|
||||
UPDATE oknardia_glazing SET fGlazingRating = 0.0, sGlazingDescription = '{}';
|
||||
|
||||
-- Очистить рейтинги наборов
|
||||
UPDATE oknardia_setkit SET fSetRating = 0.0, sSetDescription = '{}';
|
||||
```
|
||||
|
||||
### Примеры из реальных данных
|
||||
|
||||
Пример вывода `--verbosity 1`:
|
||||
|
||||
```
|
||||
=== НАЧАЛИ ПЕРЕСЧЁТ РЕЙТИНГОВ ===
|
||||
|
||||
========================================
|
||||
[ЭТАП 1]: Пересчёт рейтингов ПРОФИЛЕЙ...
|
||||
========================================
|
||||
✓ Обнулены рейтинги у 94 профилей
|
||||
✓ Найдено 94 профилей для ранжирования
|
||||
✓ Сохранено 94 профилей с финальными рейтингами
|
||||
|
||||
=============================================
|
||||
[ЭТАП 2]: Пересчёт рейтингов СТЕКЛОПАКЕТОВ...
|
||||
=============================================
|
||||
✓ Обнулены рейтинги у 97 стеклопакетов
|
||||
✓ Найдено 97 стеклопакетов для ранжирования
|
||||
✓ Сохранено 97 стеклопакетов с финальными рейтингами
|
||||
|
||||
================================================
|
||||
[ЭТАП 3]: Пересчёт рейтингов НАБОРОВ (SetKit)...
|
||||
================================================
|
||||
✓ Обнулены рейтинги у 27 наборов
|
||||
✓ Найдено 27 наборов для ранжирования
|
||||
✓ Сохранено 27 наборов с финальными рейтингами
|
||||
|
||||
[OK!] ПЕРЕСЧЁТ РЕЙТИНГОВ ЗАВЕРШЁН УСПЕШНО!
|
||||
• Обновлено профилей: 94
|
||||
• Обновлено стеклопакетов: 97
|
||||
• Обновлено наборов: 27
|
||||
```
|
||||
|
||||
Пример вывода `--verbosity 3` (наиболее подробный):
|
||||
|
||||
```
|
||||
=== НАЧАЛИ ПЕРЕСЧЁТ РЕЙТИНГОВ ===
|
||||
|
||||
========================================
|
||||
[ЭТАП 1]: Пересчёт рейтингов ПРОФИЛЕЙ...
|
||||
========================================
|
||||
|
||||
✓ Обнулены рейтинги у 94 профилей
|
||||
✓ Найдено 94 профилей для ранжирования
|
||||
...
|
||||
...
|
||||
====================================================================================================
|
||||
ПРОФИЛЬ: politech W80 (ID: 78)
|
||||
====================================================================================================
|
||||
Характеристика Значение Ранг (0..1) Вклад
|
||||
----------------------------------------------------------------------------------------------------
|
||||
Высота в проёме 120 мм 0.368 *
|
||||
Популярность 0 предл. 0.000
|
||||
Теплопередача 0.91 Ro 0.657 ***
|
||||
Толщина профиля 80 мм 0.588 **
|
||||
Толщина стеклопакета 42 мм 0.409 **
|
||||
Уплотнители 3 контуров 1.000 *****
|
||||
Фальц 14 мм 0.150
|
||||
Число камер 12 шт 0.714 ***
|
||||
Шумоизоляция 44.00 дБ 0.909 ****
|
||||
----------------------------------------------------------------------------------------------------
|
||||
ИТОГО: Рейтинг = 4.94/5.0 ****
|
||||
...
|
||||
...
|
||||
✓ Сохранено 94 профилей с финальными рейтингами
|
||||
|
||||
=============================================
|
||||
[ЭТАП 2]: Пересчёт рейтингов СТЕКЛОПАКЕТОВ...
|
||||
=============================================
|
||||
|
||||
✓ Обнулены рейтинги у 97 стеклопакетов
|
||||
✓ Найдено 97 стеклопакетов для ранжирования
|
||||
...
|
||||
...
|
||||
====================================================================================================
|
||||
СТЕКЛОПАКЕТ: Однокамерный 5-4, 25 мм (И+аргон) (ID: 60) | Марка:СПО 5М1-Ar16-И4
|
||||
====================================================================================================
|
||||
Характеристика Значение Ранг (0..1) Вклад
|
||||
----------------------------------------------------------------------------------------------------
|
||||
Камеры — 0.000
|
||||
Светопропускание 74.00% 0.824 ****
|
||||
Солнцепропускание 58.00% 0.450 **
|
||||
Теплопередача 0.91 Ro 0.936 ****
|
||||
Толщина 25 мм 0.400 **
|
||||
Шумоизоляция — 0.429 **
|
||||
----------------------------------------------------------------------------------------------------
|
||||
ИТОГО: Рейтинг = 4.87/5.0 ****
|
||||
...
|
||||
...
|
||||
✓ Сохранено 97 стеклопакетов с финальными рейтингами
|
||||
|
||||
================================================
|
||||
[ЭТАП 3]: Пересчёт рейтингов НАБОРОВ (SetKit)...
|
||||
================================================
|
||||
✓ Обнулены рейтинги у 27 наборов
|
||||
✓ Найдено 27 наборов для ранжирования
|
||||
...
|
||||
...
|
||||
========================================================================================================================
|
||||
НАБОР: Элит (ID: 3)
|
||||
========================================================================================================================
|
||||
Параметр Значение Ранг (0..1) Вклад
|
||||
------------------------------------------------------------------------------------------------------------------------
|
||||
Актуальность свежий 0.375 *
|
||||
Водоотлив ✓ Да 1.000 *****
|
||||
Гибкость скидок 0 вариантов 0.500 **
|
||||
Доставка ✓ Да 1.000 *****
|
||||
Климат-контроль ✓ Да 1.000 *****
|
||||
Монтаж ✓ Да 1.000 *****
|
||||
Откос ✓ Да 1.000 *****
|
||||
Подоконник ✓ Да 1.000 *****
|
||||
Размер скидок 0.0% 0.500 **
|
||||
Число предложений 46 шт 0.250 *
|
||||
------------------------------------------------------------------------------------------------------------------------
|
||||
ИТОГО: Рейтинг = 4.16/5.0 ****
|
||||
...
|
||||
...
|
||||
|
||||
[OK!] ПЕРЕСЧЁТ РЕЙТИНГОВ ЗАВЕРШЁН УСПЕШНО!
|
||||
• Обновлено профилей: 94
|
||||
• Обновлено стеклопакетов: 97
|
||||
• Обновлено наборов: 27
|
||||
```
|
||||
|
||||
## Оркестрация и reload веб-сервера
|
||||
|
||||
Важно:
|
||||
- reload веб-сервера не встроен в management-команды;
|
||||
- это отдельная операция окружения.
|
||||
|
||||
Пример для systemd + gunicorn:
|
||||
|
||||
```bash
|
||||
sudo systemctl reload gunicorn
|
||||
```
|
||||
|
||||
Рекомендуемый batch-сценарий:
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py regenerate_seria_prerender --force
|
||||
poetry run python oknardia/manage.py generate_sitemaps
|
||||
sudo systemctl reload gunicorn
|
||||
```
|
||||
|
||||
## Cron/systemd timer (пример)
|
||||
|
||||
Пример cron (раз в сутки в 03:20):
|
||||
|
||||
```bash
|
||||
20 3 * * * cd /Users/e-serg/PRJ/2022-oknardia && poetry run python oknardia/manage.py regenerate_seria_prerender --force && poetry run python oknardia/manage.py generate_sitemaps >> /var/log/oknardia-maintenance.log 2>&1
|
||||
```
|
||||
|
||||
Если нужен reload после batch, добавляй отдельной строкой/шагом оркестратора.
|
||||
|
||||
## Диагностика
|
||||
|
||||
Быстрая проверка конфигурации:
|
||||
|
||||
```bash
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
poetry run python oknardia/manage.py check
|
||||
```
|
||||
|
||||
Типовые причины проблем:
|
||||
- нет прав записи в директории `templates/seria_info/prepared` или `MEDIA_ROOT/_serv_sitemap`;
|
||||
- устаревшее виртуальное окружение / неустановленные зависимости;
|
||||
- запуск не из того каталога.
|
||||
|
||||
## План миграции `/service/*` -> management commands
|
||||
|
||||
Текущее направление:
|
||||
- все тяжелые и административные операции переносить из HTTP в management-команды;
|
||||
- `/service/*` оставлять только как thin UI/мониторинг или убрать полностью.
|
||||
|
||||
|
||||
|
||||
---
|
||||
|
||||
См. также:
|
||||
- `SETUP.md`
|
||||
- `README.md`
|
||||
|
||||
242
PRODUCTION_DEPLOY.md
Normal file
242
PRODUCTION_DEPLOY.md
Normal file
@@ -0,0 +1,242 @@
|
||||
# Production Deployment Guide для Oknardia
|
||||
|
||||
## Структура на хосте
|
||||
|
||||
Структура директорий на production сервере должна быть:
|
||||
|
||||
```
|
||||
~/docker-app/oknardia-site/
|
||||
├── .env # Переменные окружения Django
|
||||
├── config/ # Конфиги
|
||||
│ └── nginx/
|
||||
│ └── oknardia-app--external-nginx.conf # Конфиг Nginx (шаблон)
|
||||
├── database/ # БД SQLite (создается автоматически)
|
||||
│ └── oknardia.sqlite3
|
||||
├── media/ # Медиа файлы, uploads, кеши
|
||||
│ ├── img_avatar/
|
||||
│ ├── img_seria/
|
||||
│ ├── _serv_sitemap/
|
||||
│ └── _error/ # Error pages (генерируются контейнером)
|
||||
└── docker-compose.prod.yml # Конфиг Docker
|
||||
```
|
||||
|
||||
## Установка на Production сервере
|
||||
|
||||
### 1. Создать директории и скопировать файлы
|
||||
|
||||
```bash
|
||||
# На хосте
|
||||
mkdir -p ~/docker-app/oknardia-site/{config/nginx,database,media}
|
||||
|
||||
# Скопировать docker-compose.prod.yml и .env из проекта
|
||||
cp docker-compose.prod.yml ~/docker-app/oknardia-site/
|
||||
cp .env ~/docker-app/oknardia-site/
|
||||
|
||||
# Скопировать конфиг nginx
|
||||
cp config/nginx/oknardia-app--external-nginx.conf ~/docker-app/oknardia-site/config/nginx/
|
||||
```
|
||||
|
||||
### 2. Настроить .env на сервере
|
||||
|
||||
Отредактировать `~/docker-app/oknardia-site/.env`:
|
||||
|
||||
```bash
|
||||
# ОБЯЗАТЕЛЬНО заполнить эти переменные!
|
||||
|
||||
# Путь к проекту на хосте (используется для sed-замены в конфиге nginx)
|
||||
HOST_PROJECT_PATH=/home/default_user/projects/oknardia-site
|
||||
|
||||
# Credentials для скачивания образа из реестра Gitea
|
||||
REPO_USER=имя_пользователя_gitea
|
||||
REPO_PASS=токен_из_личного_кабинета_gitea
|
||||
|
||||
# Остальное скопировать из локального .env
|
||||
DEBUG=False
|
||||
DJANGO_SECRET_KEY=...
|
||||
```
|
||||
|
||||
**Важно:** `HOST_PROJECT_PATH` должен совпадать с корневой папкой проекта на хосте, куда ты скопировал docker-compose.prod.yml!
|
||||
|
||||
### 3. Запустить контейнеры
|
||||
|
||||
```bash
|
||||
cd ~/docker-app/oknardia-site/
|
||||
|
||||
# Первый запуск (холодный старт)
|
||||
docker compose -f docker-compose.prod.yml up -d
|
||||
|
||||
# Проверить логи
|
||||
docker compose -f docker-compose.prod.yml logs -f web
|
||||
|
||||
# Проверить статус контейнеров
|
||||
docker compose -f docker-compose.prod.yml ps
|
||||
```
|
||||
|
||||
### 4. Проверить что nginx конфиг был сгенерирован
|
||||
|
||||
После первого запуска контейнер должен:
|
||||
- Создать `/config/nginx/oknardia-app--external-nginx.conf` (боевой конфиг)
|
||||
- Создать `/config/nginx/oknardia-app--external-nginx.conf.example` (exemplo)
|
||||
|
||||
```bash
|
||||
ls -la ~/docker-app/oknardia-site/config/nginx/
|
||||
```
|
||||
|
||||
### 5. Настроить Nginx на хосте
|
||||
|
||||
На хосте установить Nginx и скопировать/ссылаться на конфиг из контейнера:
|
||||
|
||||
```bash
|
||||
# На хосте (например, Ubuntu 20.04)
|
||||
sudo systemctl install nginx
|
||||
|
||||
# Включить сгенерированный конфиг в основной конфиг Nginx
|
||||
# /etc/nginx/sites-available/oknardia или /etc/nginx/conf.d/oknardia.conf
|
||||
|
||||
# Пример:
|
||||
sudo ln -s /home/default_user/projects/oknardia-site/config/nginx/oknardia-app--external-nginx.conf \
|
||||
/etc/nginx/sites-enabled/oknardia
|
||||
|
||||
# Проверить и перезагрузить Nginx
|
||||
sudo nginx -t
|
||||
sudo systemctl reload nginx
|
||||
```
|
||||
|
||||
## Команды для управления
|
||||
|
||||
```bash
|
||||
cd ~/docker-app/oknardia-site/
|
||||
|
||||
# Запустить контейнеры (если они были остановлены)
|
||||
docker compose -f docker-compose.prod.yml up -d
|
||||
|
||||
# Остановить контейнеры
|
||||
docker compose -f docker-compose.prod.yml down
|
||||
|
||||
# Перезагрузить контейнеры (удалить и создать заново)
|
||||
docker compose -f docker-compose.prod.yml down
|
||||
docker compose -f docker-compose.prod.yml up -d
|
||||
|
||||
# Посмотреть логи
|
||||
docker compose -f docker-compose.prod.yml logs -f web
|
||||
docker compose -f docker-compose.prod.yml logs -f watchtower
|
||||
|
||||
# Выполнить команду внутри контейнера Django
|
||||
docker compose -f docker-compose.prod.yml exec web python manage.py shell
|
||||
|
||||
# Пересоздать пре-рендер кеши вручную
|
||||
docker compose -f docker-compose.prod.yml exec web python manage.py regenerate_seria_prerender
|
||||
```
|
||||
|
||||
## Что делает docker-compose.prod.yml
|
||||
|
||||
### Сервис `web` (Django + Gunicorn)
|
||||
|
||||
При старте контейнера:
|
||||
1. ✅ Применяет миграции БД (`migrate --noinput`)
|
||||
2. ✅ Собирает статику (`collectstatic --noinput`)
|
||||
3. ✅ Генерирует sitemap'ы (`generate_sitemaps`)
|
||||
4. ✅ Пересоздает пре-рендер кеши для серий (`regenerate_seria_prerender`)
|
||||
5. ✅ Пересчитывает рейтинги (`make_rating`)
|
||||
6. ✅ Копирует конфиг nginx с авто-заменой путей (через `sed`)
|
||||
7. ✅ Запускает Gunicorn на `127.0.0.1:8000`
|
||||
|
||||
### Сервис `watchtower` (авто-обновление)
|
||||
|
||||
- 👁️ Следит за реестром Gitea (проверка каждые 30 минут)
|
||||
- 🔄 Автоматически скачивает новый образ, если он есть
|
||||
- ♻️ По-graceful перезагружает контейнер web
|
||||
- 🗑️ Удаляет старые образы (WATCHTOWER_CLEANUP=true)
|
||||
|
||||
Watchtower использует метку на контейнере `web`:
|
||||
```yaml
|
||||
labels:
|
||||
- "com.centurylinklabs.watchtower.scope=oknardia-scope"
|
||||
```
|
||||
|
||||
## Файлы, которые монтируются с хоста
|
||||
|
||||
| На хосте | В контейнере | Назначение |
|
||||
|----------|----------|-----------|
|
||||
| `./database/` | `/home/app/database/` | БД SQLite (КРИТИЧНО!) |
|
||||
| `./media/` | `/home/app/public/media/` | Медиа файлы, uploads |
|
||||
| `./config/` | `/nginx_configs_host/` | Конифиги (читаются и пишущутся контейнером) |
|
||||
|
||||
## Переменные окружения в docker-compose.prod.yml
|
||||
|
||||
```yaml
|
||||
environment:
|
||||
- DEBUG=False # Production: без debug
|
||||
- DJANGO_LOG_LEVEL=INFO # Минимум логов
|
||||
- ALLOW_MEDIA_SERVE=False # Nginx обслуживает медиа
|
||||
- HOST_PROJECT_PATH=... # Путь для sed-замены в конфиге nginx
|
||||
- REPO_USER=${REPO_USER} # Из .env (для Watchtower)
|
||||
- REPO_PASS=${REPO_PASS} # Из .env (для Watchtower)
|
||||
```
|
||||
|
||||
## Health Check
|
||||
|
||||
Контейнер имеет healthcheck, который проверяет доступность Django:
|
||||
- Проверка каждые 3 минуты
|
||||
- Таймаут: 12 секунд
|
||||
- Unhealthy после 3 неудачных попыток
|
||||
- Это критично для Watchtower (он ждет что контейнер станет healthy перед finalization)
|
||||
|
||||
## Troubleshooting
|
||||
|
||||
### ContainerError при запуске
|
||||
|
||||
Проверить логи:
|
||||
```bash
|
||||
docker compose -f docker-compose.prod.yml logs web
|
||||
```
|
||||
|
||||
### БД не синхронизируется между перезагрузками
|
||||
|
||||
Убедиться что `./database/` правильно монтирует:
|
||||
```bash
|
||||
docker compose -f docker-compose.prod.yml exec web ls -la /home/app/database/
|
||||
```
|
||||
|
||||
### Watchtower не обновляет контейнер
|
||||
|
||||
Проверить:
|
||||
- Есть ли интернет на сервере
|
||||
- Правильны ли REPO_USER и REPO_PASS в .env
|
||||
- Существует ли образ `oknardia:latest` в реестре
|
||||
|
||||
```bash
|
||||
docker compose -f docker-compose.prod.yml logs watchtower
|
||||
```
|
||||
|
||||
### Nginx конфиг не обновился
|
||||
|
||||
Конфиг копируется при старте контейнера. Если нужно пересоздать:
|
||||
```bash
|
||||
docker compose -f docker-compose.prod.yml down
|
||||
docker compose -f docker-compose.prod.yml up -d
|
||||
```
|
||||
|
||||
Потом проверить:
|
||||
```bash
|
||||
cat ~/docker-app/oknardia-site/config/nginx/oknardia-app--external-nginx.conf
|
||||
```
|
||||
|
||||
## Резервные копии
|
||||
|
||||
Важно регулярно бэкапить:
|
||||
- `./database/oknardia.sqlite3` - БД
|
||||
- `./media/` - Загруженные файлы
|
||||
|
||||
Пример cronjob для ежедневного бэкапа:
|
||||
```bash
|
||||
# /etc/cron.daily/backup-oknardia
|
||||
#!/bin/bash
|
||||
BACKUP_DIR="/backup/oknardia"
|
||||
APP_DIR="/home/default_user/projects/oknardia-site"
|
||||
|
||||
mkdir -p $BACKUP_DIR
|
||||
tar -czf $BACKUP_DIR/oknardia-$(date +%Y%m%d).tar.gz $APP_DIR
|
||||
find $BACKUP_DIR -name "oknardia-*.tar.gz" -mtime +30 -delete # Удалять старше 30 дней
|
||||
```
|
||||
|
||||
119
README.md
119
README.md
@@ -1,6 +1,55 @@
|
||||
# Оконный агрегатор «Окнардия»
|
||||
### Переделка под Python 3.8 и Django 4.1
|
||||
|
||||
**Окнардия** — веб-сервис для сравнения цен на установку оконных конструкций в типовых многоквартирных домах России.
|
||||
|
||||
* **Пользователь, желающий заменить окна**, вводит адрес дома → система распознаёт серию строения → выдаёт типовые размеры оконных проёмов → показывает предложения от поставщиков на установку (замену) окон, с ценами, характеристиками компонентов и условиями.
|
||||
Алгоритмические рейтинги защищают потенциального покупателя от возможных накруток отзывов и позволяют сравнивать предложения по объективным характеристикам.
|
||||
|
||||
* Для **производителей и поставщиков окон** платформа — это канал прямого доступа к целевой аудитории в конкретном районе города.
|
||||
Они размещают свои предложения (компоненты, наборы, цены) и конкурируют на равных условиях. Масштабируемый каталог позволяет охватить
|
||||
множество адресов типовой застройки, а система алгоритмического рейтинга (нет отзывов клиентов, а значит накрутка отзывов невозможна) ранжирует предложения исключительно по характеристикам и условиям предложений.
|
||||
|
||||
**Стек**: Python 3.12+ · Django 5.2+ · SQLite/MariaDB · Bootstrap 3.3 · jQuery · Yandex Maps API
|
||||
|
||||
---
|
||||
|
||||
### Переделка под Python 3.12 и Django 5.2.13 (апрель-май 2026)
|
||||
|
||||
Сделано:
|
||||
* Переход проекта под Python 3.12 и Django 5.2.13, удаление устаревших зависимостей, унификация функций и хелперов.
|
||||
* Перехода на SQLite (возможно, после нагрузочного тестирования переход обратно на mariaDB или PostgreSQL).
|
||||
* Переделаны все raw SQL-запросы на ORM для лучшей поддержки разных СУБД в будущем.
|
||||
* Все сервисные функции из `service/` вынесены в management-команды.
|
||||
* Переработаны все шаблоны с целью SEO- и LLM-оптимизацим: более корректные meta-теги, разметка schema.org
|
||||
через `JSON-LD`, оптимизирована структура "хлебных крошек" и изменение роутинга.
|
||||
* Облегчение шаблона `base.html`: блок логин-логаут подгружается через AJAX только по клику, модуль авторизации
|
||||
вынесен в отдельный JS-файл (`/static/js/auth.js`), счетчики посещений перенесены в подгружаемый JS.
|
||||
* Шаблоны `report/report_last_user_visit.html` больше не требуют серверного рендеринга, а формируются
|
||||
на стороне клиента из кук.
|
||||
* Добавлены SEO-поля блогов.
|
||||
|
||||
### Планы, задачи, маркеры на будущее:
|
||||
|
||||
* Оптимизация кеширования pre-render шаблонов: настроить cronjob для ежедневной/еженедельной очистки `seria_info/prepared/`.
|
||||
* Улучшение администрирования в блогах (Codemirror 6, Типографф).
|
||||
* Упаковать всё в контейнеры: бакенд Django + Gunicorn + WhiteNoise...
|
||||
* CI/CD через gitea + Watchtower для автоматического деплоя при пуше тега `v*.*.*` в репозиторий.
|
||||
* Фронтенд: перейти на новый Bootstrap 5, добавить интерактивные элементы через HTMX + Alpine, сделать адаптивность для мобильных устройств. Убрать jQuery и старые плагины, заменить на современные аналоги.
|
||||
* Оптимизация для мобильных устройств: адаптивный дизайн, оптимизация изображений, улучшение производительности.
|
||||
* Переход проекта под Python 3.14 и Django 6.x.
|
||||
* Нагрузочное тестирование (рпи необходимости переход с SQLite на PostgreSQL в продакшене).
|
||||
|
||||
# См. также:
|
||||
|
||||
* [`MANAGEMENT_RUNBOOK.md`](MANAGEMENT_RUNBOOK.md) – единый runbook по management-командам и кастом-операциям (регенерация кеша, рейтингов, sitemap и т.д.).
|
||||
* [`CACHE_PRERENDER_SYSTEM.md`](CACHE_PRERENDER_SYSTEM.md) – двухуровневая система кеширования страниц серий, структура статик-шаблонов, управление кешем.
|
||||
* [`AGENTS.md`](AGENTS.md) – контекст проекта для AI-ассистентов (архитектура, конвенции, рабочие сценарии).
|
||||
* [`SETUP.md`](SETUP.md) – пошаговая настройка окружения, запуск проекта и базовые команды разработки.
|
||||
|
||||
|
||||
---
|
||||
Легаси-материалы старого README, которые могут быть полезны для понимания устройства проекта и его
|
||||
администрирования, а также для будущей реорганизации документации.
|
||||
|
||||
### Немного о механике кеширования:
|
||||
|
||||
@@ -13,63 +62,29 @@
|
||||
Эти картинки создаются автоматически. Можно не удалять. Даже если какая-то схема открывания или размер проёма станет
|
||||
неактуальным, лишняя картинка просто будет лежать в папке (вдруг такой проём появится снова).
|
||||
|
||||
#### Кеширование шаблонов
|
||||
#### Кеширование pre-render шаблонов серий домов
|
||||
|
||||
В папке `oknardia/oknardia/templates/seria_info/prepared` создаются пре-рендер шаблоны с информацией о сериях домов.
|
||||
В папке `oknardia/templates/seria_info/prepared/` создаются пре-рендер HTML-шаблоны с информацией о сериях домов.
|
||||
|
||||
Эти шаблоны надо периодически удалять. Они нужны для скорости. Но если меняются данные по серии, размерам окон, появляются
|
||||
новые коммерческие предложения -- их надо удалять и тогда построятся новые. Вообще на быстрых серверах скорость может
|
||||
не быть проблемой, так что возможно стоит просто настроить через crone ежедневное или еженедельное удаление этих
|
||||
пре-рендер шаблонов. При обращении к соответсвующий страницам эти шаблоны будут пересозданы автоматически.
|
||||
**Архитектура (май 2026)**: Для каждой серии создаются **3 отдельных кешируемых файла** (верхняя статья НЕ кешируется):
|
||||
* `{seria_id}_id_static_flaps.html` — схемы открывания окон
|
||||
* `{seria_id}_id_static_graph.html` — график ввода в эксплуатацию
|
||||
* `{seria_id}_id_static_map_stats.html` — карта Яндекса и статистика
|
||||
|
||||
**Верхняя статья рендерится динамически** из БД, поэтому изменения через админку видны без перезагрузки контейнера.
|
||||
|
||||
### Некоторые заметки относительно разработки (DEV) на macOS:
|
||||
Таблица оконных проёмов **не кешируется** — пересчитывается при каждом запросе, поэтому новые предложения видны пользователям сразу.
|
||||
|
||||
Т.к. MariaDB "сидит" в контейнере Dockers могут возникнуть трудности при установке коннектора к базам данных MySQL/MariaDB. Примерно такие:
|
||||
```txt
|
||||
Collecting mysqlclient
|
||||
Using cached mysqlclient-2.1.1.tar.gz (88 kB)
|
||||
Preparing metadata (setup.py) ... error
|
||||
error: subprocess-exited-with-error
|
||||
|
||||
× python setup.py egg_info did not run successfully.
|
||||
│ exit code: 1
|
||||
╰─> [16 lines of output]
|
||||
/bin/sh: mysql_config: command not found
|
||||
/bin/sh: mariadb_config: command not found
|
||||
/bin/sh: mysql_config: command not found
|
||||
Traceback (most recent call last):
|
||||
File "<string>", line 2, in <module>
|
||||
File "<pip-setuptools-caller>", line 34, in <module>
|
||||
File "/private/var/folders/jh/gbhf3vk11svg9w4mvhntlb7c0000gn/T/pip-install-nu5ar2g2/mysqlclient_a07e3d9dbe514c7793dc71f1183dda19/setup.py", line 15, in <module>
|
||||
metadata, options = get_config()
|
||||
File "/private/var/folders/jh/gbhf3vk11svg9w4mvhntlb7c0000gn/T/pip-install-nu5ar2g2/mysqlclient_a07e3d9dbe514c7793dc71f1183dda19/setup_posix.py", line 70, in get_config
|
||||
libs = mysql_config("libs")
|
||||
File "/private/var/folders/jh/gbhf3vk11svg9w4mvhntlb7c0000gn/T/pip-install-nu5ar2g2/mysqlclient_a07e3d9dbe514c7793dc71f1183dda19/setup_posix.py", line 31, in mysql_config
|
||||
raise OSError("{} not found".format(_mysql_config_path))
|
||||
OSError: mysql_config not found
|
||||
mysql_config --version
|
||||
mariadb_config --version
|
||||
mysql_config --libs
|
||||
[end of output]
|
||||
|
||||
note: This error originates from a subprocess, and is likely not a problem with pip.
|
||||
error: metadata-generation-failed
|
||||
|
||||
× Encountered error while generating package metadata.
|
||||
╰─> See above for output.
|
||||
|
||||
note: This is an issue with the package mentioned above, not pip.
|
||||
hint: See above for details.
|
||||
**Регенерация кеша**:
|
||||
```bash
|
||||
python manage.py regenerate_seria_prerender # все сер<D0B5><D180>и
|
||||
python manage.py regenerate_seria_prerender --seria-id 210 # конкретная серия
|
||||
```
|
||||
|
||||
Починить проблему можно воспользовавшись ([рецептом со StackOverflow](https://stackoverflow.com/a/44268445/1504067)):
|
||||
```shell
|
||||
brew install mariadb-connector-c
|
||||
# sudo ln -s /usr/local/opt/mariadb-connector-c/bin/mariadb_config /usr/local/bin/mysql_config
|
||||
⏱️ **Когда регенерировать**: Изменены координаты зданий, добавлены новые здания, обновлены годы ввода в эксплуатацию.
|
||||
|
||||
❌ **Когда НЕ нужна регенерация**: Добавлены новые предложения/цены (таблица обновляется автоматически), изменены статьи через админку (рендерятся динамически).
|
||||
|
||||
**Подробности**: см. [`CACHE_PRERENDER_SYSTEM.md`](CACHE_PRERENDER_SYSTEM.md)
|
||||
|
||||
pip install mysqlclient
|
||||
|
||||
# rm /usr/local/bin/mysql_config
|
||||
brew unlink mariadb-connector-c
|
||||
```
|
||||
388
SETUP.md
388
SETUP.md
@@ -1,311 +1,187 @@
|
||||
# 🚀 SETUP.md — Первичная настройка Окнардии
|
||||
|
||||
**Версия**: 0.2.0 | **Дата**: 16.04.2026
|
||||
**Версия**: 0.2.0 | **Дата**: 18.05.2026 | **Docker**: ✅ Поддерживается
|
||||
|
||||
Этот документ описывает пошаговую настройку проекта для разработки и деплоя.
|
||||
|
||||
## 📋 Предварительные требования
|
||||
---
|
||||
|
||||
- **Python**: 3.12+
|
||||
- **Django**: 6.0+
|
||||
- **MariaDB/MySQL**: 5.7+ или 8.0+
|
||||
- **Redis** (опционально, для кеширования): 6.0+
|
||||
- **Poetry** (для управления зависимостями)
|
||||
## 🐳 Быстрый старт: Docker Dev Environment
|
||||
|
||||
### На macOS:
|
||||
### 1️⃣ Запуск контейнера
|
||||
|
||||
```bash
|
||||
# Установка зависимостей (если не установлены)
|
||||
brew install mariadb-connector-c
|
||||
brew install redis # опционально
|
||||
cd /Users/e-serg/PRJ/2022-oknardia
|
||||
docker compose -f docker-compose.local.yml up --build
|
||||
```
|
||||
|
||||
## 🔑 Шаг 1: Конфигурация секретов
|
||||
Сайт будет доступен на **http://localhost:8060**
|
||||
|
||||
### 1.1 Создайте файл `my_secret.py`
|
||||
### 2️⃣ Основные команды
|
||||
|
||||
```bash
|
||||
cd oknardia/oknardia
|
||||
cp my_secret.py.template my_secret.py
|
||||
nano my_secret.py # отредактируйте значения
|
||||
# Просмотр логов в реальном времени
|
||||
docker compose -f docker-compose.local.yml logs web -f
|
||||
|
||||
# Зайти в контейнер (bash)
|
||||
docker compose -f docker-compose.local.yml exec web bash
|
||||
|
||||
# Перезагрузить контейнер
|
||||
docker compose -f docker-compose.local.yml restart web
|
||||
|
||||
# Остановить контейнер
|
||||
docker compose -f docker-compose.local.yml down
|
||||
```
|
||||
|
||||
**Что нужно заполнить:**
|
||||
- IP адреса и хосты (MY_HOST_HOME2, MY_DATABASE_HOST_DEV2)
|
||||
- Пароль БД (MY_DATABASE_PASSWORD_DEV)
|
||||
- Email credentials (MY_EMAIL_HOST_USER_DEV, MY_EMAIL_HOST_PASSWORD_DEV)
|
||||
- Пути к файлам (MY_MEDIA_ROOT_DEV2, MY_STATIC_ROOT_DEV2)
|
||||
- SECRET_KEY (сгенерируйте новый!)
|
||||
**✨ Особенности:**
|
||||
- ✅ **Live reload** — при изменении кода автоматически перезагружается
|
||||
- ✅ **Синхронизция файлов** — база, медиа, статика синхронизированы с хостом
|
||||
- ✅ **Миграции автоматические** — применяются при каждом старте
|
||||
- ✅ **DEBUG режим** — подробные ошибки и админка
|
||||
|
||||
### 1.2 (Опционально) Создайте файл `.env.local`
|
||||
👁️ **Подробнее про Docker разработку** → см. раздел **"🐳 Docker Development"** ниже.
|
||||
|
||||
```bash
|
||||
cd /path/to/project
|
||||
cp .env.example .env.local
|
||||
nano .env.local # отредактируйте значения
|
||||
---
|
||||
|
||||
## 🐳 Docker Development
|
||||
|
||||
### Структура контейнера
|
||||
|
||||
```
|
||||
/home/app/ # PROJECT_ROOT
|
||||
├── oknardia/ # основная папка Django
|
||||
│ ├── manage.py # точка входа
|
||||
│ ├── oknardia/ # конфиг Django
|
||||
│ ├── web/ # приложение
|
||||
│ └── templates/ # шаблоны
|
||||
├── database/ # SQLite БД (синхронизирована)
|
||||
├── public/
|
||||
│ ├── static/ # исходная статика
|
||||
│ ├── static_collected/ # собранная статика
|
||||
│ └── media/ # загруженные файлы
|
||||
└── ...
|
||||
```
|
||||
|
||||
**Примечание**: либо используйте `my_secret.py`, либо `.env.local`, выбирайте удобный способ.
|
||||
### Volume mounts (синхронизация)
|
||||
|
||||
## 🗄️ Шаг 2: Настройка БД
|
||||
|
||||
### 2.1 Создайте БД и пользователя
|
||||
|
||||
```bash
|
||||
# Подключитесь к MySQL
|
||||
mysql -u root -p
|
||||
|
||||
# В MySQL консоли:
|
||||
CREATE DATABASE django_oknardia_dev;
|
||||
CREATE USER 'web'@'localhost' IDENTIFIED BY 'your-password';
|
||||
GRANT ALL PRIVILEGES ON django_oknardia_dev.* TO 'web'@'localhost';
|
||||
FLUSH PRIVILEGES;
|
||||
EXIT;
|
||||
```yaml
|
||||
volumes:
|
||||
- .:/home/app
|
||||
```
|
||||
|
||||
### 2.2 Выполните миграции
|
||||
Это монтирует весь проект (`/Users/e-serg/PRJ/2022-oknardia`) в `/home/app` контейнера.
|
||||
|
||||
**Синхронизация:**
|
||||
- Изменения на хосте сразу видны в контейнере
|
||||
- Разные между хостом и контейнером сохраняются на диск
|
||||
- БД и медиа-файлы персистент (не теряются при рестарте контейнера)
|
||||
|
||||
### Миграции в Docker
|
||||
|
||||
```bash
|
||||
cd /path/to/project/oknardia
|
||||
# Автоматические при запуске (через docker-compose command):
|
||||
python manage.py migrate --noinput
|
||||
|
||||
# Или вручную внутри контейнера:
|
||||
docker compose -f docker-compose.local.yml exec web bash
|
||||
python manage.py makemigrations
|
||||
python manage.py migrate
|
||||
|
||||
# Или непосредственно внутри контейнера:
|
||||
docker compose -f docker-compose.local.yml exec web python manage.py migrate
|
||||
```
|
||||
|
||||
### 2.3 Создайте суперпользователя
|
||||
### Установка новых пакетов
|
||||
|
||||
```bash
|
||||
python manage.py createsuperuser
|
||||
# 1. На хосте добавьте в pyproject.toml:
|
||||
poetry add some_package
|
||||
|
||||
# 2. Пересоберите контейнер:
|
||||
docker compose -f docker-compose.local.yml up --build
|
||||
|
||||
# 3. Зависимости переустановят при старте контейнера
|
||||
```
|
||||
|
||||
## 📦 Шаг 3: Установка зависимостей
|
||||
|
||||
### Вариант 1: Poetry (рекомендуется)
|
||||
### Создание суперюзера (админ)
|
||||
|
||||
```bash
|
||||
# Установите poetry (если не установлен)
|
||||
curl -sSL https://install.python-poetry.org | python3 -
|
||||
|
||||
# Установите зависимости
|
||||
poetry install
|
||||
|
||||
# Активируйте виртуальное окружение
|
||||
poetry shell
|
||||
docker compose -f docker-compose.local.yml exec web python manage.py createsuperuser
|
||||
```
|
||||
|
||||
### Вариант 2: pip (классический способ)
|
||||
### Очистка кеша и статики
|
||||
|
||||
```bash
|
||||
python -m venv venv
|
||||
source venv/bin/activate # На Windows: venv\Scripts\activate
|
||||
pip install -r requirements.txt
|
||||
# Пересоберите статику:
|
||||
docker compose -f docker-compose.local.yml exec web python manage.py collectstatic --clear --noinput
|
||||
|
||||
# Или удалите и пересоздайте:
|
||||
rm -rf ./public/static_collected/*
|
||||
docker compose -f docker-compose.local.yml restart web
|
||||
```
|
||||
|
||||
## 🏃 Шаг 4: Запуск разработки
|
||||
|
||||
### 4.1 Запустите локальный сервер
|
||||
### DEBUG и логирование
|
||||
|
||||
```bash
|
||||
cd oknardia
|
||||
python manage.py runserver
|
||||
# DEBUG=True включен по умолчанию
|
||||
# Смотрите подробные логи:
|
||||
docker compose -f docker-compose.local.yml logs web -f --tail=50
|
||||
|
||||
# Или с фильтром (только ошибки):
|
||||
docker compose -f docker-compose.local.yml logs web --tail=100 | grep -i error
|
||||
```
|
||||
|
||||
Откройте браузер: **http://127.0.0.1:8000**
|
||||
### Типичные проблемы Docker
|
||||
|
||||
### 4.2 Запустите задачи Celery (опционально)
|
||||
**"unable to open database file"**
|
||||
```bash
|
||||
# Проверьте что БД существует:
|
||||
ls -la ./database/oknardia.sqlite3
|
||||
|
||||
# Если нет:
|
||||
cp ./database/oknadria_backup-2026-05-12.sqlite3 ./database/oknardia.sqlite3
|
||||
docker compose -f docker-compose.local.yml restart web
|
||||
```
|
||||
|
||||
**"404 Not Found" для статики**
|
||||
```bash
|
||||
# Пересоберите статику:
|
||||
docker compose -f docker-compose.local.yml exec web python manage.py collectstatic --noinput
|
||||
docker compose -f docker-compose.local.yml restart web
|
||||
```
|
||||
|
||||
**Контейнер падает**
|
||||
```bash
|
||||
# Смотрите полные логи:
|
||||
docker compose -f docker-compose.local.yml logs web --tail=200
|
||||
|
||||
# Часто ошибка в settings.py или import'ах
|
||||
```
|
||||
|
||||
**"Connection refused" при обращении в БД**
|
||||
```bash
|
||||
# Проверьте что контейнер запущен:
|
||||
docker compose -f docker-compose.local.yml ps
|
||||
|
||||
# Перезагрузитесь:
|
||||
docker compose -f docker-compose.local.yml down && docker compose -f docker-compose.local.yml up
|
||||
```
|
||||
|
||||
### Очистка Docker
|
||||
|
||||
```bash
|
||||
celery -A oknardia worker -l info
|
||||
# Удалить контейнер и образ:
|
||||
docker compose -f docker-compose.local.yml down -v
|
||||
|
||||
# Удалить весь Docker мусор:
|
||||
docker system prune -a --volumes
|
||||
```
|
||||
|
||||
## 📁 Шаг 5: Создание необходимых директорий
|
||||
|
||||
```bash
|
||||
# Статика и медиа файлы
|
||||
mkdir -p public/media
|
||||
mkdir -p public/static
|
||||
mkdir -p public/static/img/_flap.cfg
|
||||
mkdir -p public/static/img/_miniflap.cfg
|
||||
|
||||
# Логи
|
||||
mkdir -p logs
|
||||
|
||||
# Сгенерируйте статику
|
||||
python manage.py collectstatic --noinput
|
||||
```
|
||||
|
||||
## 🧪 Шаг 6: Тестирование
|
||||
|
||||
```bash
|
||||
# Запустите тесты
|
||||
python manage.py test
|
||||
|
||||
# С покрытием (если установлен coverage)
|
||||
coverage run --source='.' manage.py test
|
||||
coverage report
|
||||
```
|
||||
|
||||
## 🔐 Шаг 7: Проверка безопасности
|
||||
|
||||
### 7.1 Django встроенная проверка
|
||||
|
||||
```bash
|
||||
python manage.py check --deploy
|
||||
```
|
||||
|
||||
### 7.2 Проверка на утечки секретов
|
||||
|
||||
```bash
|
||||
# Установите инструмент
|
||||
pip install truffleHog
|
||||
|
||||
# Проверьте репозиторий
|
||||
truffleHog filesystem . --json
|
||||
```
|
||||
|
||||
## ✅ Проверка готовности
|
||||
|
||||
Убедитесь, что все работает:
|
||||
|
||||
```bash
|
||||
# 1. Статус БД
|
||||
python manage.py dbshell < /dev/null && echo "✓ Database OK"
|
||||
|
||||
# 2. Статус приложений Django
|
||||
python manage.py check && echo "✓ Django OK"
|
||||
|
||||
# 3. Статус файлов
|
||||
test -d public/media && test -d public/static && echo "✓ Directories OK"
|
||||
|
||||
# 4. Тесты
|
||||
python manage.py test 2>&1 | tail -5
|
||||
```
|
||||
|
||||
## 🚀 Развертывание на продакшене
|
||||
|
||||
### Для разных хостов
|
||||
|
||||
**Masterhost VDS:**
|
||||
```bash
|
||||
# Установка окружения
|
||||
export DJANGO_SECRET_KEY="your-production-key"
|
||||
export DATABASE_PASSWORD="production-db-password"
|
||||
export DATABASE_HOST="localhost"
|
||||
export DEBUG="False"
|
||||
|
||||
# Запуск через uWSGI + Nginx
|
||||
uwsgi --ini config/oknardia.ini
|
||||
```
|
||||
|
||||
**Docker (рекомендуется):**
|
||||
```bash
|
||||
docker build -t oknardia:latest .
|
||||
docker run -d \
|
||||
-e DJANGO_SECRET_KEY="..." \
|
||||
-e DATABASE_PASSWORD="..." \
|
||||
-p 8000:8000 \
|
||||
oknardia:latest
|
||||
```
|
||||
|
||||
## 🛠️ Полезные команды
|
||||
|
||||
```bash
|
||||
# Управление миграциями
|
||||
python manage.py makemigrations # Создать миграцию
|
||||
python manage.py migrate # Применить миграции
|
||||
python manage.py migrate --fake-initial # Подделать первую миграцию
|
||||
|
||||
# Управление данными
|
||||
python manage.py shell # Интерпретатор Python с контекстом Django
|
||||
python manage.py dumpdata > backup.json # Резервная копия данных
|
||||
python manage.py loaddata backup.json # Восстановление данных
|
||||
|
||||
# Статика и медиа
|
||||
python manage.py collectstatic # Собрать статику для продакшена
|
||||
python manage.py findstatic # Найти файлы статики
|
||||
|
||||
# Администрирование
|
||||
python manage.py createsuperuser # Создать администратора
|
||||
python manage.py changepassword username # Изменить пароль
|
||||
|
||||
# Очистка
|
||||
python manage.py clearsessions # Удалить старые сессии
|
||||
python manage.py remove_stale_contenttypes # Удалить устаревшие типы контента
|
||||
|
||||
# Служебные
|
||||
python manage.py check # Проверить конфигурацию
|
||||
python manage.py check --deploy # Проверка для продакшена
|
||||
```
|
||||
|
||||
## 📚 Дополнительные ресурсы
|
||||
---
|
||||
## Дополнительные ресурсы
|
||||
|
||||
- [Django документация](https://docs.djangoproject.com/en/stable/)
|
||||
- [AGENTS.md](./AGENTS.md) — архитектура и конвенции проекта
|
||||
- [README.md](./README.md) — основная информация о проекте
|
||||
- [SECURITY_AUDIT_REPORT.md](./SECURITY_AUDIT_REPORT.md) — отчёт безопасности
|
||||
|
||||
## ❓ Решение проблем
|
||||
|
||||
### Проблема: `mysqlclient` не устанавливается на macOS
|
||||
|
||||
**Решение:**
|
||||
```bash
|
||||
brew install mariadb-connector-c
|
||||
pip install mysqlclient
|
||||
# или
|
||||
brew unlink mariadb-connector-c # после установки
|
||||
```
|
||||
|
||||
### Проблема: `ModuleNotFoundError: No module named 'oknardia'`
|
||||
|
||||
**Решение:**
|
||||
```bash
|
||||
# Убедитесь, что находитесь в правильной директории
|
||||
cd /path/to/project/oknardia
|
||||
python manage.py runserver
|
||||
```
|
||||
|
||||
### Проблема: `OperationalError: (2002, "Can't connect to local MySQL server")`
|
||||
|
||||
**Решение:**
|
||||
```bash
|
||||
# Проверьте, что MySQL запущен
|
||||
# macOS:
|
||||
brew services start mariadb
|
||||
|
||||
# Linux:
|
||||
sudo systemctl start mysql
|
||||
|
||||
# Проверьте credentials в my_secret.py или .env
|
||||
```
|
||||
|
||||
### Проблема: миграции не применяются
|
||||
|
||||
**Решение:**
|
||||
```bash
|
||||
# Проверьте статус миграций
|
||||
python manage.py showmigrations
|
||||
|
||||
# Примените все миграции
|
||||
python manage.py migrate --run-syncdb
|
||||
|
||||
# Если проблема в конкретной миграции
|
||||
python manage.py migrate app_name 0001 --fake
|
||||
python manage.py migrate app_name
|
||||
```
|
||||
|
||||
## 🤝 Общие вопросы
|
||||
|
||||
**Q: Где хранятся секреты?**
|
||||
A: В `my_secret.py` (в .gitignore) или переменных окружения (.env)
|
||||
|
||||
**Q: Как запустить проект без интернета?**
|
||||
A: Установите все зависимости заранее, используйте локальное хранилище медиа
|
||||
|
||||
**Q: Как работает система рейтинга?**
|
||||
A: Смотрите [AGENTS.md](./AGENTS.md), раздел "Система рейтинга и ранжирования"
|
||||
|
||||
---
|
||||
|
||||
**Версия документа**: 1.0
|
||||
**Последнее обновление**: 16.04.2026
|
||||
**Автор**: GitHub Copilot
|
||||
|
||||
|
||||
147
config/nginx/oknardia-app--external-nginx.conf
Normal file
147
config/nginx/oknardia-app--external-nginx.conf
Normal file
@@ -0,0 +1,147 @@
|
||||
# config/nginx/oknardia-app--external-nginx.conf
|
||||
# ==============================================================================
|
||||
# КОНФИГУРАЦИОННЫЙ ФАЙЛ NGINX (Reverse Proxy для Docker + Production)
|
||||
# ==============================================================================
|
||||
#
|
||||
# ИНФОРМАЦИЯ:
|
||||
# 1. Этот файл используется как шаблон при деплое
|
||||
# 2. При первом деплое пути `/home/user/path-to-oknardia-app` заменяются на реальный путь через sed
|
||||
# 3. Сгенерированный конфиг скопируется в `/etc/nginx/sites-available/oknardia`
|
||||
# 4. Последующие деплои ОБНОВЛЯЮТ этот файл автоматически (sed + копирование)
|
||||
#
|
||||
# АРХИТЕКТУРА:
|
||||
# - Nginx (порты 80/443) <-> Gunicorn контейнер (localhost:8000)
|
||||
# - Медиа файлы отдаются из `/home/user/path-to-oknardia-app/media/` напрямую
|
||||
# - Ошибки 5xx берутся из `media/_error` (копируются контейнером при старте)
|
||||
# - Sitemap.xml отдается через Django/WhiteNoise (в media/_serv_sitemap/)
|
||||
# - Static файлы (/static/) тоже отдаются через Django/WhiteNoise
|
||||
|
||||
# 1. Описываем, где живет наш Django в Docker
|
||||
upstream oknardia-django {
|
||||
# Мы пробрасываем порт 8050 из контейнера наружу (в docker-compose.yml имя сервиса 'web', контейнер 'oknardia-backend')
|
||||
server 127.0.0.1:8060;
|
||||
keepalive_requests 200;
|
||||
}
|
||||
|
||||
# 2. Конфигурируем сервер
|
||||
server {
|
||||
server_name tmp.oknardia.ru; # Основное доменное имя
|
||||
|
||||
# Слушаем 80 порт (Certbot потом добавит сюда редирект на 443 и настройки SSL)
|
||||
listen 80;
|
||||
listen [::]:80;
|
||||
|
||||
charset utf-8;
|
||||
client_max_body_size 10M; # Разрешаем загрузку не слишком больших картинок
|
||||
|
||||
# Логи (пути могут отличаться в зависимости от настроек сервера, здесь стандартные для Ubuntu)
|
||||
access_log /var/log/nginx/oknardia.access.log;
|
||||
error_log /var/log/nginx/oknardia.error.log;
|
||||
|
||||
# --- GZIP (Сжатие) ---
|
||||
# Очень важно для динамического HTML от Django, который Gunicorn отдает несжатым.
|
||||
gzip on;
|
||||
gzip_vary on; # Добавляет заголовок Vary: Accept-Encoding
|
||||
gzip_proxied any; # Сжимать ответы, даже если мы за прокси
|
||||
gzip_comp_level 6; # Оптимальный баланс скорость/сжатие
|
||||
gzip_min_length 1000; # Не сжимать совсем мелочь
|
||||
# Типы файлов для сжатия (HTML сжимается автоматически, его писать не нужно)
|
||||
gzip_types
|
||||
text/plain
|
||||
text/css
|
||||
text/xml
|
||||
text/javascript
|
||||
application/javascript
|
||||
application/json
|
||||
application/xml
|
||||
application/xml+rss
|
||||
image/svg+xml
|
||||
image/x-icon
|
||||
application/vnd.ms-fontobject
|
||||
font/woff
|
||||
font/woff2;
|
||||
|
||||
# --- МЕДИА ФАЙЛЫ (Загруженный контент) ---
|
||||
# Nginx отдает их напрямую с диска хоста, не дергая Docker.
|
||||
# Сюда входит: загруженные картинки, документы, свитмапы, ошибки 5xx
|
||||
# Путь должен совпадать с тем, где лежит volume на хост-машине.
|
||||
location /media/ {
|
||||
alias /home/user/path-to-oknardia-app/media/;
|
||||
expires 30d; # Кешируем картинки на месяц
|
||||
add_header Cache-Control "public, no-transform";
|
||||
}
|
||||
|
||||
# --- SITEMAP.XML ---
|
||||
# Sitemap хранится в media/_serv_sitemap/ но должен быть доступен из корня
|
||||
# для поисковиков (они ищут http://example.com/sitemap.xml)
|
||||
location = /sitemap.xml {
|
||||
alias /home/user/path-to-oknardia-app/media/_serv_sitemap/sitemap.xml;
|
||||
expires 7d; # Кешируем на неделю (редко меняется)
|
||||
add_header Cache-Control "public";
|
||||
}
|
||||
|
||||
# --- СТРАНИЦЫ ОШИБОК (Custom Error Pages) ---
|
||||
# Если Django упал (502) или сработал тайм-аут (504), Nginx должен отдать статический HTML.
|
||||
# Эти файлы копируются в `media/_error` при старте контейнера Docker.
|
||||
#
|
||||
# ВАЖНО: error_page директива перехватывает ошибки от апстрима (Gunicorn).
|
||||
error_page 500 /500.html;
|
||||
error_page 502 /502.html;
|
||||
error_page 503 /503.html;
|
||||
error_page 504 /504.html;
|
||||
|
||||
location = /500.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
location = /502.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
location = /503.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
location = /504.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
|
||||
error_page 400 /400.html;
|
||||
error_page 401 /401.html;
|
||||
error_page 403 /403.html;
|
||||
error_page 404 /404.html;
|
||||
error_page 413 /413.html;
|
||||
error_page 429 /429.html;
|
||||
|
||||
location = /400.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
location = /401.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
location = /403.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
location = /404.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
location = /413.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
location = /429.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
|
||||
error_page 405 406 407 408 409 410 411 412 414 415 416 417 418 421 422 423 424 425 426 428 431 451 /under_reconstruction.html;
|
||||
location = /under_reconstruction.html { root /home/user/path-to-oknardia-app/media/_error; internal; }
|
||||
|
||||
# --- ВСЁ ОСТАЛЬНОЕ (Django + WhiteNoise) ---
|
||||
# Статика (/static/), robots.txt, favicon.ico и сам сайт обрабатываются внутри контейнера.
|
||||
# Nginx просто прокидывает запрос внутрь.
|
||||
location / {
|
||||
proxy_pass http://oknardia-django;
|
||||
|
||||
# Передаем правильные заголовки, чтобы Django знал реальный IP и протокол
|
||||
proxy_set_header Host $host;
|
||||
proxy_set_header X-Real-IP $remote_addr;
|
||||
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||
|
||||
# Если нужно чтобы Django обрабатывал и HTTP, и HTTPS, то можно раскомментировать эту строку
|
||||
# и передавать реальный протокол от клиента
|
||||
# proxy_set_header X-Forwarded-Proto $scheme;
|
||||
|
||||
# Явно указываем https, потому что клиент всегда приходит по HTTPS к Nginx
|
||||
# Даже если внутри контейнера это HTTP на 127.0.0.1:8050, для Django это должно быть HTTPS
|
||||
proxy_set_header X-Forwarded-Proto https;
|
||||
|
||||
# Тайм-ауты (важно для долгих операций, если они есть)
|
||||
proxy_read_timeout 180s;
|
||||
proxy_connect_timeout 180s;
|
||||
}
|
||||
}
|
||||
|
||||
# 3. Редирект с www на основной домен (SEO best practice)
|
||||
server {
|
||||
server_name www.tmp.oknardia.ru;
|
||||
listen 80;
|
||||
listen [::]:80;
|
||||
return 301 $scheme://tmp.oknardia.ru$request_uri;
|
||||
}
|
||||
|
||||
@@ -1,110 +0,0 @@
|
||||
# Разработка сайта OKNARDIA.RU
|
||||
# == Конфикурационный файл nginx oknardia.conf
|
||||
|
||||
# Описываем апстрим-потоки которые должен подключить Nginx
|
||||
# Для каждого сайта надо настроить свйо поток, со своим уникальным именем.
|
||||
# Если будете настраивать несколько python (django) сайтов - измените название upstream
|
||||
|
||||
upstream oknardia-django {
|
||||
# расположение файла Unix-сокет для взаимодействие с uwsgi
|
||||
server unix:/home/web/oknardia-ru/socket/oknardia.sock;
|
||||
# server ///home/web/oknardia-ru/socket/oknardia.sock;
|
||||
# также можно использовать веб-сокет (порт) для взаимодействие с uwsgi. Но это медленнее
|
||||
# server 127.0.0.1:8001; # для взаимодействия с uwsgi через веб-порт
|
||||
keepalive_requests 200;
|
||||
}
|
||||
|
||||
# конфигурируем сервер
|
||||
server {
|
||||
server_name tmp.cube2.ru; # доменное имя сайта
|
||||
listen 80; # managed by Certbot
|
||||
## listen 443 ssl http2; # managed by Certbot
|
||||
## ssl_certificate /etc/letsencrypt/live/cadpoint.ru/fullchain.pem; # managed by Certbot
|
||||
## ssl_certificate_key /etc/letsencrypt/live/cadpoint.ru/privkey.pem; # managed by Certbot
|
||||
## include /etc/letsencrypt/options-ssl-nginx.conf; # managed by Certbot
|
||||
## ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem; # managed by Certbot
|
||||
|
||||
# server_name 90.156.203.25; # доменное имя сайта
|
||||
charset utf-8; # кодировка по умолчанию
|
||||
access_log /home/web/oknardia-ru/logs/oknardia-access.log; # логи с доступом
|
||||
error_log /home/web/oknardia-ru/logs/oknardia-error.log; # логи с ошибками
|
||||
client_max_body_size 100M; # максимальный объем файла для загрузки на сайт (max upload size)
|
||||
error_page 404 /404.html;
|
||||
error_page 500 /500.html;
|
||||
|
||||
location /media { alias /home/web/oknardia-ru/public/media; } # Расположение media-файлов Django
|
||||
location /static { alias /home/web/oknardia-ru/public/static; } # Расположение static-файлов Django
|
||||
|
||||
location /robots.txt { root /home/web/oknardia-ru/public; } # Расположение robots.txt
|
||||
location /favicon.ico { root /home/web/oknardia-ru/public; } # Расположение favicon.ico
|
||||
location /favicon.gif { root /home/web/oknardia-ru/public; } # Расположение favicon
|
||||
location /favicon.png { root /home/web/oknardia-ru/public; } # Расположение favicon
|
||||
location /favicon.svg { root /home/web/oknardia-ru/public; } # Расположение favicon
|
||||
location /author.txt { root /home/web/oknardia-ru/public; } # Расположение author.txt
|
||||
location = /404.html {
|
||||
root /home/web/oknardia-ru/oknardia/templates/404.html;
|
||||
internal;
|
||||
}
|
||||
location = /500.html {
|
||||
root /home/web/oknardia-ru/oknardia/templates/500.html;
|
||||
internal;
|
||||
}
|
||||
# location ~ \.(html|htm|ico|svg|png|gif|jpg|jpeg)$ {
|
||||
location ~ \.(xml|html|htm|ico|svg|png|gif|jpg|jpeg)$ {
|
||||
root /home/web/oknardia-ru/public; # Расположение статичных *.xml, *.html и *.txt
|
||||
}
|
||||
|
||||
location / {
|
||||
uwsgi_pass oknardia-django; # upstream обрабатывающий обращений
|
||||
include uwsgi_params; # конфигурационный файл uwsgi;
|
||||
proxy_set_header Host $host;
|
||||
|
||||
# ограничение количества запросов c одного IP-адреса с помощью модуля Limit_Req_Module
|
||||
# limit_req zone=one burst=20 nodelay;
|
||||
# one — имя зоны настроеной в /etc/nginx/nginx.conf (для всех сайтов сервера) в блоке http {…}
|
||||
# burst — максимальный всплеск активности, можно регулировать до какого значения запросов
|
||||
# в секунду может быть всплеск запросов;
|
||||
# nodelay — незамедлительно, при достижении лимита подключений, выдавать код 503
|
||||
# (Service Unavailable) для этого IP
|
||||
|
||||
fastcgi_keep_conn on;
|
||||
uwsgi_read_timeout 1800; # некоторые запросы на Raspbery pi очень долго обрабатываются. Например, переиндексация.
|
||||
uwsgi_send_timeout 200; # на всякий случай время записи в сокет
|
||||
}
|
||||
}
|
||||
|
||||
# переадресация с www на "без" www
|
||||
server {
|
||||
server_name www.tmp.cube2.ru;
|
||||
listen 80;
|
||||
return 301 http://tmp.cube2.ru$request_uri;
|
||||
|
||||
## listen 443 ssl; # managed by Certbot
|
||||
## ssl_certificate /etc/letsencrypt/live/cadpoint.ru/fullchain.pem; # managed by Certbot
|
||||
## ssl_certificate_key /etc/letsencrypt/live/cadpoint.ru/privkey.pem; # managed by Certbot
|
||||
## include /etc/letsencrypt/options-ssl-nginx.conf; # managed by Certbot
|
||||
## ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem; # managed by Certbot
|
||||
}
|
||||
|
||||
# переадресация с http на https
|
||||
##server {
|
||||
## if ($host = oknardia.ru) {
|
||||
## return 301 https://$host$request_uri;
|
||||
## } # managed by Certbot
|
||||
##
|
||||
## listen 80;
|
||||
## server_name cadpoint.ru;
|
||||
## return 404; # managed by Certbot
|
||||
##}
|
||||
|
||||
# переадресация с http на https для www
|
||||
##server {
|
||||
## if ($host = www.cadpoint.ru) {
|
||||
## return 301 https://cadpoint.ru$request_uri;
|
||||
## } # managed by Certbot
|
||||
##
|
||||
## listen 80;
|
||||
## server_name www.cadpoint.ru;
|
||||
## return 404; # managed by Certbot
|
||||
##}
|
||||
|
||||
@@ -1,63 +0,0 @@
|
||||
# === Конфигурационный файл uwsgi cadpoint.ini
|
||||
[uwsgi]
|
||||
|
||||
# НАСТРОЙКИ ДЛЯ DJANGO
|
||||
# Корневая папка проекта (полный путь)
|
||||
chdir = /home/web/oknardia-ru/oknardia
|
||||
# Django wsgi файл rsvo_new/wsgi.py записываем так:
|
||||
module = oknardia.wsgi
|
||||
# полный путь к виртуальному окружению
|
||||
home = /home/web/oknardia-ru/env
|
||||
# полный путь к файлу сокета
|
||||
socket = /home/web/oknardia-ru/socket/oknardia.sock
|
||||
# Исходящие сообщения в лог
|
||||
daemonize = /home/web/oknardia-ru/logs/oknardia_uwsgi.log
|
||||
|
||||
# ЗАГАДОЧНЫЕ НАСТРОЙКИ, ПО ИДЕЕ ОНИ НУЖНЫ, НО И БЕЗ НИХ ВСЁ РАБОТАЕТ
|
||||
# расположение wsgi.py
|
||||
wsgi-file = /home/web/oknardia-ru/oknardia/oknardia/wsgi.py
|
||||
# расположение виртуального окружения (как оно работает если этот параметр не указан, не ясно)
|
||||
virtualenv = /home/web/oknardia-ru/env
|
||||
# имя файла при изменении которого происходит авторестарт приложения
|
||||
# (когда этого параметра нет, то ничего не авторестартится, но с ним все рестартится.
|
||||
# Cтоит изменить любой Python-исходник проекта, как изменения сразу вступают в силу.
|
||||
touch-reload = /home/web/oknardia-ru/logs/touch-reload.txt
|
||||
py-autoreload = 5
|
||||
|
||||
# НАСТРОЙКИ ОБЩИЕ
|
||||
# быть master-процессом
|
||||
master = true
|
||||
# максимальное количество процессов
|
||||
processes = 1
|
||||
# если uWSGI установлен как сервис через apt-get то нужно установить еще плагин:
|
||||
# sudo apt-get install uwsgi-plugin-python
|
||||
# и добавить в этот конфиг: plugin = python
|
||||
plugin = python3
|
||||
# права доступа к файлу сокета. По умолчанию должно хватать 664. Но каких-то прав не хватает, поэтому 666.
|
||||
chmod-socket = 666
|
||||
# очищать окружение от служебных файлов uwsgi по завершению
|
||||
vacuum = true
|
||||
# количество секунд после которых подкисший процесс будет перезапущен
|
||||
# Так как некоторе скрипты требуют изрядно времени (особенно полная переиндексация) то ставим значение побольше
|
||||
harakiri = 2600
|
||||
# В общем случае, при некотых значениях harakiri логах uWSGI может вываливаться предупреждение:
|
||||
# WARNING: you have enabled harakiri without post buffering. Slow upload could be rejected on post-unbuffered webservers
|
||||
# можно оставить harakiri закомментированным, но нам нужно 900 и на него не ругается. Ругается на 30.
|
||||
|
||||
# разрешаем многопоточность
|
||||
enable-threads = true
|
||||
vacuum = true
|
||||
thunder-lock = true
|
||||
max-requests = 500
|
||||
|
||||
# пользователь и группа пользователей от имени которых запускать uWSGI
|
||||
# указываем www-data: к этой группе относится nginx, и ранее мы включили в эту группу нашего [user]
|
||||
# uid = nginx
|
||||
# gid = nginx
|
||||
# uid = root
|
||||
# gid = root
|
||||
uid = web
|
||||
gid = web
|
||||
|
||||
|
||||
print = ---------------- Запущен uWSGI для oknardia.ru ----------------
|
||||
3
database/.gitignore
vendored
Normal file
3
database/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
||||
# Это папака для хранения базы данных SQLite, не должна быть в репозитории.
|
||||
*.*
|
||||
*
|
||||
90
docker-compose.local-prod.yml
Normal file
90
docker-compose.local-prod.yml
Normal file
@@ -0,0 +1,90 @@
|
||||
# ==============================================================================
|
||||
# Docker Compose для PRODUCTION тестирования (Local Production Simulation)
|
||||
# Этот файл содержит настройки для локального тестирования production конфигурации.
|
||||
#
|
||||
# ВАЖНО: Используется для отладки production сборки ЛОКАЛЬНО перед деплоем!
|
||||
#
|
||||
# Запуск: docker compose -f docker-compose.local-prod.yml up --build
|
||||
# Остановка: docker compose -f docker-compose.local-prod.yml down
|
||||
# ==============================================================================
|
||||
|
||||
services:
|
||||
web:
|
||||
# Имя контейнера для удобства
|
||||
container_name: oknardia-backend-prod-test
|
||||
|
||||
# Сборка из текущей директории
|
||||
build: .
|
||||
|
||||
# Пробрасывание портов (тестируем на другом порту, чтобы не конфликтовать с dev)
|
||||
# Dev на 8060, Prod на 8061
|
||||
ports:
|
||||
- "8061:8000"
|
||||
|
||||
# 1. КОМАНДА ЗАПУСКА (Prod режим)
|
||||
# Используем Gunicorn как в production.
|
||||
# При старте контейнера:
|
||||
# 1. Создаём медиа директорию
|
||||
# 2. Применяем миграции
|
||||
# 3. Собираем статику
|
||||
# 4. Генерируем sitemap'ы
|
||||
# 5. Пересоздаём пре-рендер шаблоны серий
|
||||
# 6. Пересчитываем рейтинги профилей, стеклопакетов и наборов
|
||||
# (метод Манна-Уитни: может занять 30-60 сек на БД с 94+ профилями)
|
||||
# 7. Запускаем Gunicorn (нет live reload — это production режим!)
|
||||
# (timeout=120 сек для запуска, достаточно для всех процедур инициализации)
|
||||
command: >
|
||||
sh -c "mkdir -p /home/app/public/media &&
|
||||
python manage.py migrate --noinput &&
|
||||
python manage.py collectstatic --noinput &&
|
||||
python manage.py generate_sitemaps &&
|
||||
python manage.py regenerate_seria_prerender &&
|
||||
python manage.py make_rating &&
|
||||
python -m gunicorn --workers 2 --bind 0.0.0.0:8000 --timeout 120 oknardia.wsgi:application"
|
||||
|
||||
# 2. МОНТИРОВАНИЕ КОДА И ДАННЫХ
|
||||
# Подключаем весь проект целиком, чтобы Django правильно вычислил пути.
|
||||
# В production файлы будут в контейнере, но для локального тестирования
|
||||
# монтируем для удобства остановки/пересоздания.
|
||||
volumes:
|
||||
# Весь проект в /home/app
|
||||
- .:/home/app
|
||||
|
||||
# 3. ПЕРЕМЕННЫЕ ОКРУЖЕНИЯ (Production)
|
||||
env_file:
|
||||
# Основной конфиг из .env (DATABASE, EMAIL, API ключи)
|
||||
- .env
|
||||
environment:
|
||||
# ПЕРЕОПРЕДЕЛЯЕМ для production режима:
|
||||
- DEBUG=False # ВАЖНО: выключен debug для production
|
||||
- DJANGO_LOG_LEVEL=INFO # Вместо DEBUG (меньше логов)
|
||||
# Для локального тестирования: разрешить отдачу медиа через Django (только test!)
|
||||
# В production это обслуживает Nginx!
|
||||
- ALLOW_MEDIA_SERVE=True
|
||||
|
||||
# 4. РЕСУРСЫ (Для локального тестирования можно без лимитов, но в реальном production нужны)
|
||||
# deploy:
|
||||
# resources:
|
||||
# limits:
|
||||
# cpus: '1'
|
||||
# memory: 512M
|
||||
# reservations:
|
||||
# cpus: '0.5'
|
||||
# memory: 256M
|
||||
|
||||
# 5. ЛОГИРОВАНИЕ
|
||||
# Для отладки production проблем полезно видеть логи
|
||||
logging:
|
||||
driver: "json-file"
|
||||
options:
|
||||
max-size: "10m"
|
||||
max-file: "3"
|
||||
|
||||
# 6. ЗДОРОВЬЕ КОНТЕЙНЕРА
|
||||
healthcheck:
|
||||
test: ["CMD", "curl", "-f", "http://localhost:8000/"]
|
||||
interval: 30s
|
||||
timeout: 10s
|
||||
retries: 3
|
||||
start_period: 40s
|
||||
|
||||
67
docker-compose.local.yml
Normal file
67
docker-compose.local.yml
Normal file
@@ -0,0 +1,67 @@
|
||||
# ==============================================================================
|
||||
# Docker Compose для РАЗРАБОТКИ (Local Development)
|
||||
# Этот файл содержит настройки для локальной работы (live reload, debug).
|
||||
# Запуск: docker compose -f docker-compose.local up --build
|
||||
# ==============================================================================
|
||||
|
||||
services:
|
||||
web:
|
||||
# Имя контейнера для удобства
|
||||
container_name: oknardia-backend-dev
|
||||
|
||||
# Сборка из текущей директории
|
||||
build: .
|
||||
|
||||
# Пробрасывание портов (чтобы сайт был доступен на localhost:8060)
|
||||
# Занятые порты можно посмотреть через `sudo ss -tulpn`
|
||||
ports:
|
||||
- "8060:8000"
|
||||
|
||||
# 1. КОМАНДА ЗАПУСКА (Dev режим)
|
||||
# Используем Django runserver для разработки:
|
||||
# - Автоматически отдаёт статику без WhiteNoise
|
||||
# - Имеет встроенный auto-reload при изменении кода
|
||||
# - Безопаснее и проще для dev, чем Gunicorn
|
||||
# - Миграции применяются автоматически при каждом старте
|
||||
# - Пересоздаём пре-рендер шаблоны и sitemap'ы для актуальности
|
||||
command: >
|
||||
sh -c "python manage.py migrate --noinput &&
|
||||
python manage.py collectstatic --noinput &&
|
||||
python manage.py generate_sitemaps &&
|
||||
python manage.py regenerate_seria_prerender &&
|
||||
python manage.py runserver 0.0.0.0:8000"
|
||||
|
||||
# 2. МОНТИРОВАНИЕ КОДА (Live Reload)
|
||||
# Подключаем весь проект целиком в /home/app для правильного вычисления путей PROJECT_ROOT.
|
||||
# При изменении кода runserver автоматически перезагружается (auto-reload).
|
||||
#
|
||||
# Структура монтирования:
|
||||
# Host: /Users/e-serg/PRJ/2022-oknardia (.)
|
||||
# Container: /home/app
|
||||
# ├── oknardia/ <- исходный код (manage.py находится здесь)
|
||||
# ├── database/ <- SQLite БД для синхронизации между хостом и контейнером
|
||||
# ├── public/ <- статика и медиа-файлы
|
||||
# └── templates/ <- Django шаблоны
|
||||
volumes:
|
||||
- .:/home/app
|
||||
|
||||
# 3. ПЕРЕМЕННЫЕ ОКРУЖЕНИЯ
|
||||
env_file:
|
||||
# файл с переменными окружения для разработки
|
||||
- .env
|
||||
environment:
|
||||
# На всякий случай принудительно включаем DEBUG и DEBUG-уровень логов (вдруг в .env что-то не так)
|
||||
# Эти настройки более приоритетные, чем в .env (если дублируются).
|
||||
- DEBUG=True
|
||||
- DJANGO_LOG_LEVEL=DEBUG
|
||||
# В dev нам не нужно ограничивать буферизацию так строго, но не помешает.
|
||||
|
||||
# 4. РЕСУРСЫ (Без лимитов для разработки)
|
||||
# Удаляем секцию ограничений, чтобы локально использовать все доступные ресурсы хоста.
|
||||
# deploy:
|
||||
# resources:
|
||||
# limits:
|
||||
# cpus: ...
|
||||
# memory: ...
|
||||
# mem_limit: ...
|
||||
|
||||
130
docker-compose.prod.yml
Normal file
130
docker-compose.prod.yml
Normal file
@@ -0,0 +1,130 @@
|
||||
# ==============================================================================
|
||||
# Docker Compose для PRODUCTION
|
||||
# Этот файл запускается на боевом сервере.
|
||||
# Вариант 1 (если переименовали в docker-compose.yml): docker compose up -d
|
||||
# Вариант 2 (если оставили имя): docker compose -f docker-compose.prod.yml up -d
|
||||
# ==============================================================================
|
||||
|
||||
services:
|
||||
# --- ОСНОВНОЙ СЕРВИС: DJANGO + GUNICORN + WHITENOISE ---
|
||||
web:
|
||||
# Имя контейнера для удобства
|
||||
container_name: oknardia-backend
|
||||
|
||||
# В production используем готовый, собранный образ из реестра (Gitea)
|
||||
image: git.cube2.ru/erjemin/oknardia:latest
|
||||
|
||||
# ПЕРЕЗАПУСК при сбое
|
||||
restart: always
|
||||
|
||||
# Метки для Watchtower (авто-обновление)
|
||||
labels:
|
||||
- "com.centurylinklabs.watchtower.scope=oknardia-scope"
|
||||
|
||||
# КОМАНДА ЗАПУСКА (Production режим)
|
||||
# При старте контейнера:
|
||||
# 1. Применяем миграции
|
||||
# 2. Собираем статику
|
||||
# 3. Генерируем sitemap'ы
|
||||
# 4. Пересоздаём пре-рендер шаблоны серий
|
||||
# 5. Пересчитываем рейтинги
|
||||
# 6. Создаём папку конфигов nginx (если нет)
|
||||
# 7. Копируем конфиг nginx с авто-заменой путей через sed
|
||||
# (переменная HOST_PROJECT_PATH берется из .env и подставляется в контейнер)
|
||||
# 8. Инициализируем боевой конфиг (если нет, копируем из примера)
|
||||
# 9. Запускаем Gunicorn
|
||||
command: >
|
||||
sh -c "python manage.py migrate --noinput &&
|
||||
python manage.py collectstatic --noinput &&
|
||||
python manage.py generate_sitemaps &&
|
||||
python manage.py regenerate_seria_prerender &&
|
||||
python manage.py make_rating &&
|
||||
mkdir -p /nginx_configs_host/nginx &&
|
||||
sed \"s|/home/user/path-to-oknardia-app|$${HOST_PROJECT_PATH}|g\" /home/app/config/nginx/oknardia-app--external-nginx.conf > /nginx_configs_host/nginx/oknardia-app--external-nginx.conf.example &&
|
||||
if [ ! -f /nginx_configs_host/nginx/oknardia-app--external-nginx.conf ]; then
|
||||
cp /nginx_configs_host/nginx/oknardia-app--external-nginx.conf.example /nginx_configs_host/nginx/oknardia-app--external-nginx.conf;
|
||||
echo 'INIT: Created new nginx config with correct paths';
|
||||
fi &&
|
||||
ERROR_DIR=/home/app/web/public/media/_error &&
|
||||
mkdir -p "$$ERROR_DIR" &&
|
||||
for code in 400 401 403 404 413 429 500 502 503 504; do
|
||||
cp /home/user/path-to-oknardia-app/oknardia/templates/error/$${code}.html "$$ERROR_DIR/$${code}.html";
|
||||
done &&
|
||||
cp /home/user/path-to-oknardia-app/oknardia/templates/error/under_reconstruction.html "$$ERROR_DIR/under_reconstruction.html"
|
||||
python -m gunicorn --workers 2 --bind 0.0.0.0:8000 --timeout 120 oknardia.wsgi:application"
|
||||
|
||||
# Пробрасывание портов
|
||||
# Слушаем только на localhost хоста для безопасности
|
||||
ports:
|
||||
- "127.0.0.1:8060:8000"
|
||||
|
||||
# МОНТИРОВАНИЕ ТОМОВ (Volumes)
|
||||
volumes:
|
||||
# БД SQLite
|
||||
- ./database:/home/app/database
|
||||
|
||||
# Медиа файлы
|
||||
- ./media:/home/app/public/media
|
||||
|
||||
# Конфиги nginx
|
||||
- ./config:/nginx_configs_host
|
||||
|
||||
# Пользователь и права
|
||||
user: "0:0"
|
||||
|
||||
# ПЕРЕМЕННЫЕ ОКРУЖЕНИЯ (Production)
|
||||
# .env файл содержит все sensitive данные (DB, Email, API keys, REPO_USER/PASS)
|
||||
env_file:
|
||||
- .env
|
||||
environment:
|
||||
- DJANGO_SETTINGS_MODULE=oknardia.settings
|
||||
- PYTHONUNBUFFERED=1
|
||||
- DEBUG=False
|
||||
- DJANGO_LOG_LEVEL=INFO
|
||||
- ALLOW_MEDIA_SERVE=False
|
||||
|
||||
# ЗДОРОВЬЕ КОНТЕЙНЕРА (Healthcheck)
|
||||
healthcheck:
|
||||
test: ["CMD", "python", "-c", "import urllib.request; urllib.request.urlopen('http://localhost:8000/').read()"]
|
||||
interval: 3m
|
||||
timeout: 12s
|
||||
start_period: 60s
|
||||
retries: 3
|
||||
|
||||
# ЛОГИРОВАНИЕ (Ротация)
|
||||
logging:
|
||||
driver: "json-file"
|
||||
options:
|
||||
max-size: "10m"
|
||||
max-file: "5"
|
||||
|
||||
# РЕСУРСЫ
|
||||
deploy:
|
||||
resources:
|
||||
limits:
|
||||
cpus: '0.75'
|
||||
memory: 768M
|
||||
mem_limit: 768M
|
||||
|
||||
# --- WATCHTOWER: АВТО-ОБНОВЛЕНИЕ ОБРАЗОВ ---
|
||||
watchtower:
|
||||
image: containrrr/watchtower
|
||||
container_name: oknardia_watchtower
|
||||
restart: always
|
||||
volumes:
|
||||
- /var/run/docker.sock:/var/run/docker.sock
|
||||
environment:
|
||||
- REPO_USER=${REPO_USER}
|
||||
- REPO_PASS=${REPO_PASS}
|
||||
- WATCHTOWER_SCOPE=oknardia-scope
|
||||
- WATCHTOWER_CLEANUP=true
|
||||
- DOCKER_API_VERSION=1.44
|
||||
- WATCHTOWER_WAIT_ON_TIMEOUT=60
|
||||
- WATCHTOWER_LIFECYCLE_HOOKS=true
|
||||
command: --interval 1800 --cleanup
|
||||
logging:
|
||||
driver: "json-file"
|
||||
options:
|
||||
max-size: "10m"
|
||||
max-file: "3"
|
||||
|
||||
@@ -1 +0,0 @@
|
||||
1
|
||||
@@ -0,0 +1,34 @@
|
||||
# Generated by Django 5.2.13 on 2026-05-10 14:39
|
||||
|
||||
import datetime
|
||||
from django.db import migrations, models
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('oknardia', '0001_initial'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AddField(
|
||||
model_name='blogposts',
|
||||
name='sMetaDescription',
|
||||
field=models.CharField(blank=True, default='', help_text='SEO: описание для мета-тега (до 160 символов). Если пусто, будет использоваться текст тизера из контента.', max_length=160, verbose_name='Meta описание'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='blogposts',
|
||||
name='sMetaKeywords',
|
||||
field=models.CharField(blank=True, default='', help_text='SEO: ключевые слова для мета-тега (до 256 символов). Если пусто, будет использоваться заголовок.', max_length=256, verbose_name='Meta ключевые слова'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='blogposts',
|
||||
name='sSlug',
|
||||
field=models.SlugField(blank=True, help_text='SEO: URL-friendly версия заголовка (автоматически генерируется, если оставить пусто)', max_length=200, verbose_name='Slug'),
|
||||
),
|
||||
migrations.AlterField(
|
||||
model_name='blogposts',
|
||||
name='dPostDataBegin',
|
||||
field=models.DateTimeField(db_index=True, default=datetime.datetime(2026, 5, 10, 17, 39, 4, 114851), help_text='Если установить будущую дату, то в назначеное время пост появится автоматически.', verbose_name='Опубликован от'),
|
||||
),
|
||||
]
|
||||
@@ -8,7 +8,8 @@ from datetime import date, datetime
|
||||
from django.utils import timezone
|
||||
from django.contrib.auth.models import User
|
||||
from oknardia.settings import *
|
||||
|
||||
from web.add_func import sanitize_slug, safe_html_spec_symbols
|
||||
import re
|
||||
|
||||
# Таблица: Каталог профилей, стеклопакетов (добавлено 09.авг.2017)
|
||||
# create table oknardia_catalog2profile
|
||||
@@ -1033,6 +1034,28 @@ class BlogPosts(models.Model):
|
||||
db_index=False,
|
||||
verbose_name=u"Создано"
|
||||
)
|
||||
sMetaDescription = models.CharField(
|
||||
max_length=160,
|
||||
blank=True,
|
||||
default=u"",
|
||||
verbose_name=u"Meta описание",
|
||||
help_text=u"SEO: описание для мета-тега (до 160 символов). Если пусто, будет использоваться текст тизера из контента."
|
||||
)
|
||||
sMetaKeywords = models.CharField(
|
||||
max_length=256,
|
||||
blank=True,
|
||||
default=u"",
|
||||
verbose_name=u"Meta ключевые слова",
|
||||
help_text=u"SEO: ключевые слова для мета-тега (до 256 символов). Если пусто, будет использоваться заголовок."
|
||||
)
|
||||
sSlug = models.SlugField(
|
||||
max_length=200,
|
||||
db_index=True,
|
||||
blank=True,
|
||||
verbose_name=u"Slug",
|
||||
help_text=u"SEO: URL-friendly версия заголовка (автоматически генерируется, если оставить пусто)"
|
||||
)
|
||||
|
||||
|
||||
def __unicode__(self):
|
||||
# return u'%s (%s)' % (self.sPostHeader, datetime.strftime(
|
||||
@@ -1042,6 +1065,46 @@ class BlogPosts(models.Model):
|
||||
def __str__(self):
|
||||
return self.__unicode__()
|
||||
|
||||
def save(self, *args, **kwargs):
|
||||
"""Переопределённый метод save() для автоматической генерации слага и SEO-полей.
|
||||
|
||||
При сохранении записи блога:
|
||||
- Генерируется sSlug из sPostHeader если тот пуст
|
||||
- Генерируется sMetaDescription из текста контента (тизер)
|
||||
- Генерируется sMetaKeywords из заголовка
|
||||
"""
|
||||
# Шаг 1: Автоматически генерируем слаг из заголовка, если он не указан
|
||||
if not self.sSlug and self.sPostHeader:
|
||||
self.sSlug = sanitize_slug(self.sPostHeader, max_length=200)
|
||||
|
||||
# Шаг 2: Автоматически генерируем sMetaDescription из контента (тизер)
|
||||
if not self.sMetaDescription and self.sPostContent:
|
||||
# Удаляем теги <cut> из контента
|
||||
content_clean = re.sub(r'<cut[\s\S]*?>', '', self.sPostContent, flags=re.IGNORECASE)
|
||||
|
||||
# Генерируем тизер (очищенный текст без HTML)
|
||||
tizer = safe_html_spec_symbols(content_clean)
|
||||
|
||||
# Обрезаем до 160 символов для мета-description
|
||||
if len(tizer) > 160:
|
||||
# Обрезаем слово целиком (не посередине)
|
||||
tizer = tizer[:160].rsplit(' ', 1)[0] + '...' if ' ' in tizer[:160] else tizer[:160]
|
||||
|
||||
self.sMetaDescription = tizer
|
||||
|
||||
# Шаг 3: Автоматически генерируем sMetaKeywords из заголовка
|
||||
if not self.sMetaKeywords and self.sPostHeader:
|
||||
|
||||
# Берём заголовок и удаляем HTML-теги
|
||||
header_clean = safe_html_spec_symbols(self.sPostHeader)
|
||||
header_clean = header_clean.strip()
|
||||
|
||||
# Генерируем ключевые слова: фиксированные + заголовок
|
||||
fixed_keywords = u"oknardia, окнардия, блог, публикация"
|
||||
self.sMetaKeywords = f"{fixed_keywords}, {header_clean}"[:256]
|
||||
|
||||
super().save(*args, **kwargs)
|
||||
|
||||
class Meta:
|
||||
# db_table = "jtb_BlogPost"
|
||||
verbose_name = u"Запись в блоге каталоге"
|
||||
@@ -1323,6 +1386,8 @@ class Win_MountDim(models.Model):
|
||||
)
|
||||
sFlapConfig = models.CharField(
|
||||
max_length=32,
|
||||
blank=True,
|
||||
default=u"",
|
||||
verbose_name=u"Открывание",
|
||||
help_text=u"Рекомендуемая гор.архитектурой конфигурации открывания (МЕТАЯЗЫК)")
|
||||
sDescripion = models.CharField(
|
||||
|
||||
@@ -1,185 +0,0 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
ШАБЛОН для my_secret.py
|
||||
|
||||
ИНСТРУКЦИЯ: скопируйте этот файл в my_secret.py и заполните реальные значения.
|
||||
|
||||
Пример:
|
||||
cp oknardia/oknardia/my_secret.py.template oknardia/oknardia/my_secret.py
|
||||
# затем отредактируйте значения в my_secret.py
|
||||
|
||||
ВАЖНО: my_secret.py НИКОГДА не должен быть в git!
|
||||
Используйте .gitignore для исключения файла.
|
||||
"""
|
||||
|
||||
# ============================================================================
|
||||
# РАЗРАБОТКА (DEV) - Хосты и сетевые настройки
|
||||
# ============================================================================
|
||||
|
||||
# Хосты на которых может работать приложение (разработка)
|
||||
MY_ALLOWED_HOSTS = [
|
||||
'127.0.0.1',
|
||||
'localhost',
|
||||
'your-dev-hostname.local', # ИЗМЕНИТЕ на ваше имя хоста
|
||||
]
|
||||
|
||||
# Допустимые хосты для разработки
|
||||
MY_HOST_HOME1 = 'your-dev-hostname-windows' # ИЗМЕНИТЕ
|
||||
MY_HOST_HOME2 = 'your-dev-hostname-mac' # ИЗМЕНИТЕ
|
||||
MY_HOST_DEV = [MY_HOST_HOME1, MY_HOST_HOME2]
|
||||
|
||||
# Хосты для продакшена (заполнять с осторожностью)
|
||||
MY_HOST_PROD = [] # На продакшене используйте переменные окружения!
|
||||
|
||||
# ============================================================================
|
||||
# БЕЗОПАСНОСТЬ - Django SECRET_KEY
|
||||
# ============================================================================
|
||||
|
||||
# SECURITY WARNING: keep the secret key used in production secret!
|
||||
# Сгенерируйте новый ключ с помощью:
|
||||
# python -c 'from django.core.management.utils import get_random_secret_key; print(get_random_secret_key())'
|
||||
MY_SECRET_KEY = 'ЗАПОЛНИТЕ_СЛУЧАЙНОЙ_СТРОКОЙ_БОЛЬШОЙ_ДЛИНЫ'
|
||||
|
||||
# ============================================================================
|
||||
# АДМИНИСТРАТОРЫ - для оповещений об ошибках
|
||||
# ============================================================================
|
||||
|
||||
MY_ADMINS = (
|
||||
('Your Name', 'your-email@example.com'),
|
||||
('Admin Name', 'admin@example.com'),
|
||||
)
|
||||
|
||||
# ============================================================================
|
||||
# ПУТИ К ФАЙЛАМ - разработка
|
||||
# ============================================================================
|
||||
|
||||
# путь к каталогу media (статика, для web-сервера nginx или apache)
|
||||
MY_MEDIA_ROOT_DEV1 = 'M:\\path\\to\\your\\media\\' # Windows (если применимо)
|
||||
MY_MEDIA_ROOT_DEV2 = '/path/to/your/media/' # Mac/Linux - ИЗМЕНИТЕ!
|
||||
|
||||
# путь к каталогу static (статика, для web-сервера nginx или apache)
|
||||
MY_STATIC_ROOT_DEV1 = 'M:\\path\\to\\your\\static' # Windows (если применимо)
|
||||
MY_STATIC_ROOT_DEV2 = '/path/to/your/static' # Mac/Linux - ИЗМЕНИТЕ!
|
||||
|
||||
# путь для кэш-блоков шаблонов
|
||||
MY_STATIC_BASE_PATH_DEV1 = MY_STATIC_ROOT_DEV1
|
||||
MY_STATIC_BASE_PATH_DEV2 = MY_STATIC_ROOT_DEV2
|
||||
|
||||
# путь для sitemap файлов
|
||||
MY_SITEMAP_ROOT_DEV1 = 'M:\\path\\to\\your\\public\\' # Windows (если применимо)
|
||||
MY_SITEMAP_ROOT_DEV2 = '/path/to/your/public/' # Mac/Linux - ИЗМЕНИТЕ!
|
||||
|
||||
# ============================================================================
|
||||
# ПУТИ К ФАЙЛАМ - продакшен
|
||||
# ============================================================================
|
||||
|
||||
MY_MEDIA_ROOT_PROD = '/home/web/oknardia-ru/public/media/' # ЗАПОЛНИТЕ!
|
||||
MY_STATIC_ROOT_PROD = '/home/web/oknardia-ru/public/static' # ЗАПОЛНИТЕ!
|
||||
MY_STATIC_BASE_PATH_PROD = MY_STATIC_ROOT_PROD
|
||||
MY_SITEMAP_ROOT_PROD = '/home/web/oknardia-ru/public/' # ЗАПОЛНИТЕ!
|
||||
|
||||
# ============================================================================
|
||||
# EMAIL - Почтовый сервер (разработка)
|
||||
# ============================================================================
|
||||
|
||||
# Email адреса для разработки
|
||||
MY_EMAIL_DEV = 'dev-email@example.com'
|
||||
MY_EMAIL_FROM_DEV = 'dev-email@example.com'
|
||||
MY_EMAIL_HOST_USER_DEV = 'your-email@smtp.example.com' # ЗАПОЛНИТЕ!
|
||||
MY_EMAIL_HOST_PASSWORD_DEV = 'YOUR_EMAIL_PASSWORD' # ЗАПОЛНИТЕ!
|
||||
MY_EMAIL_HOST_DEV = 'smtp.example.com' # ЗАПОЛНИТЕ! (например: smtp.mail.ru)
|
||||
MY_EMAIL_PORT_DEV = 587 # ЗАПОЛНИТЕ! (обычно 587 или 2525)
|
||||
|
||||
# ============================================================================
|
||||
# EMAIL - Почтовый сервер (продакшен)
|
||||
# ============================================================================
|
||||
|
||||
MY_EMAIL_PROD = MY_EMAIL_DEV
|
||||
MY_EMAIL_FROM_PROD = MY_EMAIL_FROM_DEV
|
||||
MY_EMAIL_HOST_USER_PROD = MY_EMAIL_HOST_USER_DEV # На продакшене используйте env переменные!
|
||||
MY_EMAIL_HOST_PASSWORD_PROD = MY_EMAIL_HOST_PASSWORD_DEV # На продакшене используйте env переменные!
|
||||
MY_EMAIL_HOST_PROD = MY_EMAIL_HOST_DEV
|
||||
MY_EMAIL_PORT_PROD = MY_EMAIL_PORT_DEV
|
||||
|
||||
# ============================================================================
|
||||
# БД MySQL/MariaDB - разработка
|
||||
# ============================================================================
|
||||
|
||||
MY_DATABASE_HOST_DEV1 = 'localhost' # Офисный сервер разработки - ИЗМЕНИТЕ!
|
||||
MY_DATABASE_HOST_DEV2 = 'localhost' # Домашний сервер разработки - ИЗМЕНИТЕ!
|
||||
|
||||
MY_DATABASE_NAME_DEV = 'django_oknardia_dev' # ИЗМЕНИТЕ если нужно
|
||||
MY_DATABASE_PORT_DEV = '3306' # Стандартный порт MySQL
|
||||
|
||||
MY_DATABASE_USER_DEV = 'web' # ИЗМЕНИТЕ если нужно
|
||||
MY_DATABASE_PASSWORD_DEV = 'YOUR_DB_PASSWORD' # ЗАПОЛНИТЕ!
|
||||
|
||||
# ============================================================================
|
||||
# БД MySQL/MariaDB - продакшен
|
||||
# ============================================================================
|
||||
|
||||
MY_DATABASE_HOST_PROD = 'localhost' # ЗАПОЛНИТЕ! (на продакшене)
|
||||
MY_DATABASE_NAME_PROD = 'django_oknardia_prod' # ЗАПОЛНИТЕ!
|
||||
|
||||
MY_DATABASE_PORT_PROD = '3306'
|
||||
MY_DATABASE_USER_PROD = 'web'
|
||||
|
||||
# ВНИМАНИЕ: На продакшене используйте переменные окружения или менеджер секретов!
|
||||
MY_DATABASE_PASSWORD_PROD = '' # ОСТАВЬТЕ ПУСТО! Используйте переменные окружения!
|
||||
|
||||
# ============================================================================
|
||||
# API ключи - Google Captcha
|
||||
# ============================================================================
|
||||
|
||||
# Получите ключи на https://www.google.com/recaptcha/admin
|
||||
# ВАЖНО: Никогда не коммитьте реальные ключи в git!
|
||||
# PRIVATE ключ - это СЕКРЕТ, держите его в безопасности!
|
||||
MY_CAPTCHA_PUBLIC_KEY = 'YOUR_CAPTCHA_PUBLIC_KEY_HERE' # ЗАПОЛНИТЕ!
|
||||
MY_CAPTCHA_PRIVATE_KEY = 'YOUR_CAPTCHA_PRIVATE_KEY_HERE' # ЗАПОЛНИТЕ! (СЕКРЕТ!)
|
||||
|
||||
# ============================================================================
|
||||
# API ключи - Yandex Maps
|
||||
# ============================================================================
|
||||
|
||||
# Получите ключ на https://developer.tech.yandex.ru/
|
||||
MY_YANDEX_MAPS_API_KEY = 'YOUR_YANDEX_MAPS_API_KEY'
|
||||
|
||||
# ============================================================================
|
||||
# uWSGI - Touch-reload файл (для перезагрузки при изменении кода)
|
||||
# ============================================================================
|
||||
|
||||
MY_TOUCH_RELOAD_DEV1 = 'M:\\path\\to\\touch-reload.txt' # Windows (если применимо)
|
||||
MY_TOUCH_RELOAD_DEV2 = '/path/to/logs/touch-reload.txt' # Mac/Linux - ИЗМЕНИТЕ!
|
||||
MY_TOUCH_RELOAD_PROD = '/home/web/oknardia-ru/logs/touch-reload.txt' # ЗАПОЛНИТЕ!
|
||||
|
||||
# ============================================================================
|
||||
# ИНСТРУКЦИЯ ПО ЗАПОЛНЕНИЮ
|
||||
# ============================================================================
|
||||
|
||||
"""
|
||||
1. СКОПИРУЙТЕ этот файл:
|
||||
cp oknardia/oknardia/my_secret.py.template oknardia/oknardia/my_secret.py
|
||||
|
||||
2. ОТРЕДАКТИРУЙТЕ значения, помеченные ИЗМЕНИТЕ! или ЗАПОЛНИТЕ!
|
||||
|
||||
3. УБЕДИТЕСЬ, что мой_secret.py в .gitignore:
|
||||
grep my_secret .gitignore
|
||||
|
||||
4. НИКОГДА не коммитьте my_secret.py в git!
|
||||
|
||||
5. На ПРОДАКШЕНЕ используйте переменные окружения:
|
||||
export DJANGO_SECRET_KEY="..."
|
||||
export DATABASE_PASSWORD="..."
|
||||
и т.д.
|
||||
|
||||
СОВЕТЫ:
|
||||
- Сгенерируйте новый SECRET_KEY с помощью Python:
|
||||
python -c 'from django.core.management.utils import get_random_secret_key; print(get_random_secret_key())'
|
||||
|
||||
- Используйте менеджер паролей (LastPass, 1Password, Vault) для хранения учетных данных
|
||||
|
||||
- Регулярно меняйте пароли БД и API ключи
|
||||
|
||||
- На продакшене используйте отдельные более сильные пароли
|
||||
"""
|
||||
|
||||
@@ -1,43 +1,60 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
Django settings for oknardia project.
|
||||
|
||||
Generated by 'django-admin startproject' using Django 4.1.1.
|
||||
|
||||
For more information on this file, see
|
||||
https://docs.djangoproject.com/en/4.1/topics/settings/
|
||||
|
||||
For the full list of settings and their values, see
|
||||
https://docs.djangoproject.com/en/4.1/ref/settings/
|
||||
"""
|
||||
|
||||
from pathlib import Path
|
||||
from oknardia.my_secret import *
|
||||
import socket
|
||||
import environ
|
||||
|
||||
|
||||
def _env_admins(raw_items: list[str]) -> tuple[tuple[str, str], ...]:
|
||||
# Формат: "Имя1:email1,Имя2:email2"
|
||||
admins: list[tuple[str, str]] = []
|
||||
for item in raw_items:
|
||||
if ":" not in item:
|
||||
continue
|
||||
admin_name, admin_email = item.split(":", maxsplit=1)
|
||||
admin_name = admin_name.strip()
|
||||
admin_email = admin_email.strip()
|
||||
if admin_name and admin_email:
|
||||
admins.append((admin_name, admin_email))
|
||||
return tuple(admins)
|
||||
|
||||
def _normalize_admin_url(value: str) -> str:
|
||||
"""Приводит URL админки к виду `segment/` без ведущего слэша."""
|
||||
normalized = value.strip().lstrip('/')
|
||||
if not normalized:
|
||||
return 'admin/'
|
||||
if not normalized.endswith('/'):
|
||||
normalized += '/'
|
||||
return normalized
|
||||
|
||||
# Build paths inside the project like this: BASE_DIR / 'subdir'.
|
||||
BASE_DIR = Path(__file__).resolve().parent.parent
|
||||
PROJECT_ROOT = BASE_DIR.parent
|
||||
PUBLIC_ROOT = PROJECT_ROOT / 'public'
|
||||
STATIC_SOURCE_ROOT = PUBLIC_ROOT / 'static'
|
||||
|
||||
env = environ.Env()
|
||||
environ.Env.read_env(str(PROJECT_ROOT / '.env'))
|
||||
|
||||
CSRF_TRUSTED_ORIGINS = env.list('DJANGO_CSRF_TRUSTED_ORIGINS', default=[])
|
||||
|
||||
# Quick-start development settings - unsuitable for production
|
||||
# See https://docs.djangoproject.com/en/4.1/howto/deployment/checklist/
|
||||
|
||||
# SECURITY WARNING: keep the secret key used in production secret!
|
||||
SECRET_KEY = 'django-insecure-pd&1$j6z*1w#(j*16b+(@@#&2)+@x^^ot4)zqt-e67*1+$^qch'
|
||||
SECRET_KEY = env(
|
||||
var='DJANGO_SECRET_KEY',
|
||||
default='django-insecure-pd&1$j6z*1w#(j*16b+(@@#&2)+@x^^ot4)zqt-e67*1+$^qch',
|
||||
)
|
||||
ADMIN_URL = _normalize_admin_url(env(var='ADMIN_URL', default='admin/'))
|
||||
|
||||
# SECURITY WARNING: don't run with debug turned on in production!
|
||||
# ПРЕДУПРЕЖДЕНИЕ БЕЗОПАСНОСТИ: не работайте в режиме DEBUG в продашене!
|
||||
if socket.gethostname() in MY_HOST_DEV:
|
||||
DEBUG = TEMPLATE_DEBUG = True
|
||||
else:
|
||||
# Все остальные хосты (подразумевается продакшн)
|
||||
DEBUG = TEMPLATE_DEBUG = True
|
||||
# DEBUG = TEMPLATE_DEBUG = False
|
||||
# PREDУПРЕЖДЕНИЕ БЕЗОПАСНОСТИ: не работайте в режиме DEBUG в продашене!
|
||||
DEBUG = TEMPLATE_DEBUG = env.bool('DEBUG', default=False)
|
||||
|
||||
ALLOWED_HOSTS = MY_ALLOWED_HOSTS
|
||||
# Допустимые хосты (+ 'testserver' для management команд типа regenerate_seria_prerender)
|
||||
ALLOWED_HOSTS = env.list('ALLOWED_HOSTS', default=['127.0.0.1', 'localhost', 'testserver'])
|
||||
|
||||
# Настройки сообщений об ошибках когда все упало и т.п.
|
||||
ADMINS = MY_ADMINS
|
||||
ADMINS = _env_admins(env.list('ADMINS', default=[]))
|
||||
|
||||
|
||||
# Application definition
|
||||
@@ -51,7 +68,7 @@ INSTALLED_APPS = [
|
||||
'django.contrib.staticfiles',
|
||||
|
||||
'django.contrib.humanize',
|
||||
# 'django.contrib.sitemaps',
|
||||
'django.contrib.sitemaps',
|
||||
|
||||
'oknardia.apps.OknardiaConfig',
|
||||
'web.apps.WebConfig',
|
||||
@@ -67,6 +84,13 @@ MIDDLEWARE = [
|
||||
'django.middleware.clickjacking.XFrameOptionsMiddleware',
|
||||
]
|
||||
|
||||
# Разрешенные IP для отладки (нужно для django-debug-toolbar).
|
||||
INTERNAL_IPS = env.list('INTERNAL_IPS', default=['127.0.0.1', 'localhost'])
|
||||
|
||||
if DEBUG:
|
||||
INSTALLED_APPS += ['debug_toolbar']
|
||||
MIDDLEWARE = ['debug_toolbar.middleware.DebugToolbarMiddleware', *MIDDLEWARE]
|
||||
|
||||
ROOT_URLCONF = 'oknardia.urls'
|
||||
|
||||
TEMPLATES = [
|
||||
@@ -115,61 +139,94 @@ DATETIME_FORMAT = 'Y-m-d H:i:s'
|
||||
# Статические файлы (CSS, JavaScript, Images)
|
||||
# https://docs.djangoproject.com/en/4.1/howto/static-files/
|
||||
|
||||
STATIC_URL = 'static/'
|
||||
MEDIA_URL = 'media/'
|
||||
STATIC_URL = '/static/'
|
||||
MEDIA_URL = '/media/'
|
||||
|
||||
MEDIA_ROOT = str(PUBLIC_ROOT / 'media')
|
||||
# STATIC_ROOT отделен от исходной статики, чтобы избежать staticfiles.E002.
|
||||
STATIC_ROOT = str(PUBLIC_ROOT / 'static_collected')
|
||||
|
||||
if socket.gethostname() in MY_HOST_DEV: # DEBUG: заменяем настройки прода, на настройки девопа
|
||||
MEDIA_ROOT = MY_MEDIA_ROOT_DEV1 if socket.gethostname() == MY_HOST_HOME1 else MY_MEDIA_ROOT_DEV2
|
||||
SITEMAP_ROOT = MY_SITEMAP_ROOT_DEV1 if socket.gethostname() == MY_HOST_HOME1 else MY_SITEMAP_ROOT_DEV2
|
||||
# STATIC_ROOT = MY_STATIC_ROOT_DEV1 if socket.gethostname() == MY_HOST_HOME1 else MY_STATIC_ROOT_DEV2
|
||||
STATICFILES_DIRS = [
|
||||
MY_STATIC_ROOT_DEV1 if socket.gethostname() == MY_HOST_HOME1 else MY_STATIC_ROOT_DEV2,
|
||||
]
|
||||
# путь к каталогу static (в эту переменную использовать для указания пути где будут делаться кэш-блоки для шаблонов)
|
||||
STATIC_BASE_PATH = MY_STATIC_BASE_PATH_DEV1 if socket.gethostname() == MY_HOST_HOME1 else MY_STATIC_BASE_PATH_DEV2
|
||||
# Базовый URL сайта нужен для абсолютных URL в sitemap.xml.
|
||||
SITE_BASE_URL = env('SITE_BASE_URL', default='https://oknardia.ru').rstrip('/')
|
||||
# Файлы sitemap храним в media-volume, чтобы переживали пересоздание контейнера.
|
||||
SITEMAP_SUBDIR = env('SITEMAP_SUBDIR', default='_serv_sitemap').strip('/ ')
|
||||
SITEMAP_ROOT = str(Path(MEDIA_ROOT) / SITEMAP_SUBDIR)
|
||||
SITEMAP_URL_PREFIX = f"{MEDIA_URL.rstrip('/')}/{SITEMAP_SUBDIR}"
|
||||
SITEMAP_INDEX_URL = f"{SITE_BASE_URL}{SITEMAP_URL_PREFIX}/sitemap.xml"
|
||||
|
||||
# Каталоги, откуда Django читает исходную статику в DEBUG-режиме.
|
||||
STATICFILES_DIRS = [
|
||||
str(STATIC_SOURCE_ROOT)
|
||||
] if STATIC_SOURCE_ROOT.is_dir() else []
|
||||
|
||||
# Django 5 требует явное описание хранилищ.
|
||||
# `default` нужен для загружаемых файлов (FileField, ImageField, filer и подобное) и смотрит в `MEDIA_ROOT`.
|
||||
# `staticfiles` остаётся отдельно: в dev используется обычная статика Django, в prod — WhiteNoise.
|
||||
STORAGES = {
|
||||
'default': {
|
||||
'BACKEND': 'django.core.files.storage.FileSystemStorage',
|
||||
'OPTIONS': {
|
||||
'location': MEDIA_ROOT,
|
||||
},
|
||||
},
|
||||
'staticfiles': {
|
||||
'BACKEND': 'django.contrib.staticfiles.storage.StaticFilesStorage',
|
||||
},
|
||||
}
|
||||
|
||||
# Путь к каталогу static для генерации кэш-файлов и служебных JS.
|
||||
STATIC_BASE_PATH = str(STATIC_SOURCE_ROOT)
|
||||
|
||||
# Определяем движок БД из переменной окружения (по умолчанию SQLite)
|
||||
database_engine = env('DATABASE_ENGINE', default='django.db.backends.sqlite3')
|
||||
|
||||
if database_engine == 'django.db.backends.sqlite3':
|
||||
# Для SQLite принимаем только имя файла из env и кладем БД в PROJECT_ROOT/database.
|
||||
sqlite_db_filename = Path(env('DATABASE_NAME', default='oknardia.sqlite3')).name
|
||||
sqlite_db_path = PROJECT_ROOT / 'database' / sqlite_db_filename
|
||||
DATABASES = {
|
||||
'default': {
|
||||
'ENGINE': "django.db.backends.mysql",
|
||||
'HOST': MY_DATABASE_HOST_DEV1 if socket.gethostname() == MY_HOST_HOME1 else MY_DATABASE_HOST_DEV2,
|
||||
'PORT': MY_DATABASE_PORT_DEV, # Set to "" for default. Not used with sqlite3.
|
||||
'NAME': MY_DATABASE_NAME_DEV, # Not used with sqlite3.
|
||||
'USER': MY_DATABASE_USER_DEV, # Not used with sqlite3.
|
||||
'PASSWORD': MY_DATABASE_PASSWORD_DEV, # Not used with sqlite3.
|
||||
# 'OPTIONS': { 'autocommit': True, }
|
||||
}
|
||||
'ENGINE': 'django.db.backends.sqlite3',
|
||||
'NAME': str(sqlite_db_path),
|
||||
'OPTIONS': {
|
||||
'timeout': 20,
|
||||
},
|
||||
},
|
||||
}
|
||||
TOUCH_RELOAD = MY_TOUCH_RELOAD_DEV1 if socket.gethostname() == MY_HOST_HOME1 else MY_TOUCH_RELOAD_DEV2
|
||||
else:
|
||||
MEDIA_ROOT = MY_MEDIA_ROOT_PROD
|
||||
# STATICFILES_DIRS = [MY_STATIC_ROOT_PROD1, ]
|
||||
STATIC_ROOT = MY_STATIC_ROOT_PROD
|
||||
SITEMAP_ROOT = MY_SITEMAP_ROOT_PROD
|
||||
# путь к каталогу static (в эту переменную использовать для указания пути где будут делаться кэш-блоки для шаблонов)
|
||||
STATIC_BASE_PATH = MY_STATIC_BASE_PATH_PROD
|
||||
# База не SQLite (mariaDB, например): читаем все параметры подключения из env.
|
||||
DATABASES = {
|
||||
'default': {
|
||||
'ENGINE': "django.db.backends.mysql",
|
||||
'HOST': MY_DATABASE_HOST_PROD, # Set to "" for localhost. Not used with sqlite3.
|
||||
'PORT': MY_DATABASE_PORT_PROD, # Set to "" for default. Not used with sqlite3.
|
||||
'NAME': MY_DATABASE_NAME_PROD, # Not used with sqlite3.
|
||||
'USER': MY_DATABASE_USER_PROD, # Not used with sqlite3.
|
||||
'PASSWORD': MY_DATABASE_PASSWORD_PROD, # Not used with sqlite3.
|
||||
# 'OPTIONS': { 'autocommit': True, }
|
||||
'ENGINE': database_engine,
|
||||
'HOST': env('DATABASE_HOST', default='localhost'),
|
||||
'PORT': env('DATABASE_PORT', default='3306'),
|
||||
'NAME': env('DATABASE_NAME', default=''),
|
||||
'USER': env('DATABASE_USER', default=''),
|
||||
'PASSWORD': env('DATABASE_PASSWORD', default=''),
|
||||
}
|
||||
}
|
||||
TOUCH_RELOAD = MY_TOUCH_RELOAD_PROD
|
||||
|
||||
|
||||
#########################################
|
||||
# настройки для почтового сервера (они одинаковые для DEV и PROD)
|
||||
EMAIL_HOST = MY_EMAIL_HOST_DEV
|
||||
EMAIL_PORT = MY_EMAIL_PORT_DEV
|
||||
EMAIL_HOST_USER = MY_EMAIL_HOST_USER_DEV
|
||||
EMAIL_HOST_PASSWORD = MY_EMAIL_HOST_PASSWORD_DEV
|
||||
SERVER_EMAIL = DEFAULT_FROM_EMAIL = EMAIL_HOST_USER
|
||||
EMAIL_USE_TLS = True
|
||||
EMAIL_BACKEND = env(
|
||||
'EMAIL_BACKEND',
|
||||
default='django.core.mail.backends.smtp.EmailBackend',
|
||||
)
|
||||
EMAIL_HOST = env('EMAIL_HOST', default='localhost')
|
||||
EMAIL_PORT = env.int('EMAIL_PORT', default=25)
|
||||
EMAIL_HOST_USER = env('EMAIL_HOST_USER', default='')
|
||||
EMAIL_HOST_PASSWORD = env('EMAIL_HOST_PASSWORD', default='')
|
||||
EMAIL_USE_TLS = env.bool('EMAIL_USE_TLS', default=True)
|
||||
EMAIL_USE_SSL = env.bool('EMAIL_USE_SSL', default=False)
|
||||
DEFAULT_FROM_EMAIL = env('DEFAULT_FROM_EMAIL', default=EMAIL_HOST_USER)
|
||||
SERVER_EMAIL = env('SERVER_EMAIL', default=DEFAULT_FROM_EMAIL)
|
||||
EMAIL_SUBJECT_PREFIX = 'OKNARDIA ERR: ' # префикс для оповещений об ошибках и необработанных исключениях
|
||||
|
||||
SECURE_SSL_REDIRECT = env.bool('SECURE_SSL_REDIRECT', default=False)
|
||||
SESSION_COOKIE_SECURE = env.bool('SESSION_COOKIE_SECURE', default=False)
|
||||
CSRF_COOKIE_SECURE = env.bool('CSRF_COOKIE_SECURE', default=False)
|
||||
|
||||
|
||||
# Default primary key field type
|
||||
# https://docs.djangoproject.com/en/4.1/ref/settings/#default-auto-field
|
||||
@@ -177,10 +234,17 @@ EMAIL_SUBJECT_PREFIX = 'OKNARDIA ERR: ' # префикс для оповещ
|
||||
DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'
|
||||
|
||||
# ключи для Google Captha
|
||||
CAPTCHA_PUBLIC_KEY = MY_CAPTCHA_PUBLIC_KEY
|
||||
CAPTCHA_PRIVATE_KEY = MY_CAPTCHA_PRIVATE_KEY
|
||||
CAPTCHA_PUBLIC_KEY = env('CAPTCHA_PUBLIC_KEY', default='')
|
||||
CAPTCHA_PRIVATE_KEY = env('CAPTCHA_PRIVATE_KEY', default='')
|
||||
|
||||
# количество коммерческих предложений во фреме отчета
|
||||
# МАГИЧЕСКИЕ ЧИСЛА
|
||||
# если непонятно какая серия выбрана через каталог (finger fix) выбираем серию типового строения:
|
||||
DEFAULT_SERIA_ID_FOR_CATALOG = 843 # СЕРИЯ 1-515/9 -- дом в котором я живу
|
||||
DEFAULT_WIN_WIDTH_MM = 670 # Ширина типового окна для ID=16 (если не выбрано)
|
||||
DEFAULT_WIN_HEIGHT_MM = 2160 # Высота типового окна для ID=16 (если не выбрано)
|
||||
DEFAULT_WIN_ID = 16 # ID типового окна (если не выбрано)
|
||||
|
||||
# количество коммерческих предложений во фрейме отчета
|
||||
OFFER_PER_FRAME = 5
|
||||
OFFER_PER_FRAME_FOR_ONE_FLAP = 10
|
||||
# папка для хранения изображений
|
||||
@@ -272,4 +336,63 @@ CATALOG_SORTER_MAGIC_NUMBER_TIZER = 1
|
||||
|
||||
MAX_LEN_RING_LOG_BUFFER = 250 # МАКСИМАЛЬНЫЙ РАЗМЕР КОЛЬЦЕВОГО БУФЕРА
|
||||
|
||||
YANDEX_MAPS_API_KEY = MY_YANDEX_MAPS_API_KEY
|
||||
YANDEX_MAPS_API_KEY = env('YANDEX_MAPS_API_KEY', default='')
|
||||
|
||||
# ============================================================================
|
||||
# Конфигурация в зависимости от режима разработки (DEBUG) vs. production
|
||||
# ============================================================================
|
||||
|
||||
if DEBUG:
|
||||
# В dev: стандартная отдача статики Django (без WhiteNoise/кэширования).
|
||||
# Медиа-файлы отдаются через Django.
|
||||
pass
|
||||
|
||||
else:
|
||||
# В prod: WhiteNoise + CompressedStaticFilesStorage для оптимизации.
|
||||
# Статика собирается с хешем в имени и кэшируется.
|
||||
#
|
||||
# ВАЖНО: Для production нужна полная настройка Nginx:
|
||||
# - Nginx обслуживает статику (/static/) прямо из /home/app/public/static_collected/
|
||||
# - Nginx обслуживает медиа (/media/) прямо из /home/app/public/media/
|
||||
# - Nginx проксирует остальное на Gunicorn (:8000)
|
||||
#
|
||||
# Для локального тестирования production конфига этот файл симулирует Nginx.
|
||||
|
||||
# 1. Добавляем WhiteNoise в начало MIDDLEWARE (после SecurityMiddleware) для отдачи статики
|
||||
MIDDLEWARE.insert(1, 'whitenoise.middleware.WhiteNoiseMiddleware')
|
||||
|
||||
# 2. Переводим staticfiles на WhiteNoise со сжатием
|
||||
# ВАЖНО: используем CompressedStaticFilesStorage вместо CompressedManifestStaticFilesStorage,
|
||||
# потому что Manifest не справляется с relative paths в CSS (например, jQuery UI).
|
||||
# CompressedStaticFilesStorage сжимает файлы без manifest, что работает надежнее.
|
||||
STORAGES['staticfiles'] = {
|
||||
'BACKEND': 'whitenoise.storage.CompressedStaticFilesStorage', # noqa: F821
|
||||
}
|
||||
|
||||
# 3. WhiteNoise конфиг: обслуживание корневых файлов из public/ (robots.txt, favicon.*, sitemap.xml и т.д.)
|
||||
# Параметр WHITENOISE_LOCATION указывает WhiteNoise, где искать файлы помимо STATIC_ROOT
|
||||
WHITENOISE_LOCATION = str(PUBLIC_ROOT)
|
||||
|
||||
# 4. MIME-типы для шрифтов (иначе браузер может не загрузить)
|
||||
WHITENOISE_MIMETYPES = {
|
||||
'.woff': 'font/woff',
|
||||
'.woff2': 'font/woff2',
|
||||
}
|
||||
# 5. Конфигурация WhiteNoise для обслуживания статических файлов и файлов из /public (например,
|
||||
# robots.txt, favicon.ico и т.п.)
|
||||
# WHITENOISE_ROOT = PUBLIC_ROOT
|
||||
|
||||
# 6. Кэширование неизменяемых файлов (с хешем в имени) на 1 год в браузере
|
||||
# ВАЖНО: лямбда должна принимает ДВА аргумента: path и url (как требует WhiteNoise)
|
||||
WHITENOISE_IMMUTABLE_FILE_TEST = lambda path, url: 'CACHE' in path
|
||||
|
||||
# 7. ЛОКАЛЬНЫЙ ТЕСТ: для отдачи медиа в docker-compose.local-prod.yml
|
||||
# В реальном production это обслуживает Nginx! Никогда не используй в production!
|
||||
# Добавляем StaticFilesHandler который обслуживает и медиа и статику
|
||||
if env.bool('ALLOW_MEDIA_SERVE', default=False):
|
||||
# Для локального тестирования добавляем обслуживание медиа через Django
|
||||
# ВАЖНО: это очень медленно и небезопасно для production!
|
||||
from django.conf.urls.static import static
|
||||
# Будет добавлено в urls.py при импорте: urlpatterns += static(MEDIA_URL, document_root=MEDIA_ROOT)
|
||||
|
||||
|
||||
|
||||
@@ -1,28 +1,84 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""oknardia Конфигурация URL
|
||||
|
||||
Список `urlpatterns` направляет URL-адреса в представления. Дополнительную информацию см.:
|
||||
https://docs.djangoproject.com/en/4.1/topics/http/urls/
|
||||
Примеры:
|
||||
Представления функций
|
||||
1. Добавьте import: из представлений импорта my_app
|
||||
2. Добавьте URL-адрес в urlpatterns: path('', views.home, name='home')
|
||||
Представления на основе классов
|
||||
1. Добавьте импорт: from other_app.views import Home
|
||||
2. Добавьте URL-адрес в шаблоны URL-адресов: path('', Home.as_view(), name='home')
|
||||
Включение другой конфигурации URL
|
||||
1. Импортируйте функцию include(): из django.urls import include, path
|
||||
2. Добавьте URL-адрес в urlpatterns: path('blog/', include('blog.urls'))
|
||||
"""
|
||||
"""oknardia Конфигурация URL"""
|
||||
from django.contrib import admin
|
||||
from django.urls import path, re_path
|
||||
from django.conf.urls.static import static
|
||||
from oknardia.settings import *
|
||||
from web import views, autocomplete_addr, user_manager, blog, diagrams, report1, report2, catalog, prices, service
|
||||
from django.urls import include, re_path, path
|
||||
from django.http import FileResponse
|
||||
from pathlib import Path
|
||||
import environ
|
||||
import mimetypes
|
||||
import os
|
||||
|
||||
# Инициализируем env
|
||||
env = environ.Env()
|
||||
PROJECT_ROOT = Path(__file__).resolve().parent.parent.parent
|
||||
environ.Env.read_env(str(PROJECT_ROOT / '.env'))
|
||||
from oknardia.settings import *
|
||||
from web import views, autocomplete_addr, user_manager, blog, diagrams, report1, report2, catalog, prices, service, \
|
||||
catalog_profiles, catalog_series, catalog_openings, catalog_companies
|
||||
|
||||
|
||||
def _serve_public_root_file(request, path):
|
||||
"""Отдает статик-файл из public/ с правильным Content-Type и UTF-8 для текстовых файлов."""
|
||||
file_path = PUBLIC_ROOT / path
|
||||
|
||||
# Безопасность: проверяем, что файл гарантированно внутри PUBLIC_ROOT
|
||||
try:
|
||||
file_path = file_path.resolve()
|
||||
file_path.relative_to(PUBLIC_ROOT.resolve())
|
||||
except ValueError:
|
||||
from django.http import Http404
|
||||
raise Http404("Файл не найден")
|
||||
|
||||
if not file_path.is_file():
|
||||
from django.http import Http404
|
||||
raise Http404("Файл не найден")
|
||||
|
||||
# Определяем MIME type
|
||||
mime_type, _ = mimetypes.guess_type(str(file_path))
|
||||
|
||||
# Для текстовых файлов добавляем charset=utf-8
|
||||
if mime_type and mime_type.startswith('text'):
|
||||
mime_type = f"{mime_type}; charset=utf-8"
|
||||
|
||||
response = FileResponse(open(file_path, 'rb'), content_type=mime_type)
|
||||
|
||||
# Определяем Content-Disposition в зависимости от типа файла:
|
||||
# inline — браузер отображает/обрабатывает (иконки, картинки, CSS, JS, robots.txt, manifest)
|
||||
# attachment — браузер скачивает (архивы, документы и т.д.)
|
||||
inline_types = {
|
||||
'image/', # все картинки (PNG, SVG, JPG и т.д.)
|
||||
'text/', # все текстовые файлы
|
||||
'application/javascript',
|
||||
'application/json',
|
||||
'application/xml',
|
||||
'application/rss+xml',
|
||||
'application/x-web-app-manifest+json', # manifest
|
||||
}
|
||||
|
||||
if mime_type and any(mime_type.lower().startswith(t) for t in inline_types):
|
||||
response['Content-Disposition'] = 'inline'
|
||||
else:
|
||||
# Для остального (неизвестные типы, архивы и т.д.) — скачивание
|
||||
response['Content-Disposition'] = f'attachment; filename="{file_path.name}"'
|
||||
|
||||
return response
|
||||
|
||||
|
||||
def _iter_public_root_files():
|
||||
"""Находит все обычные файлы в корне public/, кроме служебных артефактов (начинающихся с точки)."""
|
||||
if not PUBLIC_ROOT.exists():
|
||||
return
|
||||
|
||||
for file_path in sorted(PUBLIC_ROOT.iterdir()):
|
||||
if not file_path.is_file():
|
||||
continue
|
||||
# Пропускаем служебные файлы (начинающиеся с точки)
|
||||
if file_path.name.startswith('.'):
|
||||
continue
|
||||
yield file_path.name
|
||||
|
||||
urlpatterns = [
|
||||
path('admin/', admin.site.urls),
|
||||
path(ADMIN_URL, admin.site.urls),
|
||||
|
||||
# главная страница
|
||||
re_path(r'^$', views.main_init),
|
||||
@@ -56,56 +112,104 @@ urlpatterns = [
|
||||
re_path(r'^stat/series/geo[/*]$', diagrams.statistic_menu), # дубль для старых ссылок
|
||||
re_path(r'^stat/rating[/*]$', report2.ratings),
|
||||
re_path(r'^stat/rating/profiles_rank[/*]$', report2.profiles_rating),
|
||||
# --- Каталог
|
||||
# --- --- Каталог профилей
|
||||
re_path(r'^catalog[/*]$', catalog.catalog_root),
|
||||
re_path(r'^catalog/profile[/*]$', catalog.catalog_profile),
|
||||
# --- КАТАЛОГ
|
||||
re_path(r'^catalog[/*]$', catalog.catalog_root), # ГЛАВНАЯ СТРАНИЦА КАТАЛОГА
|
||||
# --- --- КАТАЛОГ ПРОФИЛЕЙ
|
||||
re_path(r'^catalog/profile[/*]$', catalog_profiles.catalog_profile), # СПИСОК ВСЕХ ПРОФИЛЕЙ И ПРОИЗВОДИТЕЛЕЙ
|
||||
re_path(r'^catalog/profile/(?P<manufacture_id>\d+)-(?P<manufacture_name>\S*)'
|
||||
r'/(?P<model_id>\d+)-(?P<model_name>\S*)[/*]$', catalog.catalog_profile_model),
|
||||
r'/(?P<model_id>\d+)-(?P<model_name>\S*)[/*]$',
|
||||
catalog_profiles.catalog_profile_model), # СТРАНИЦА ОПИСАНИЯ МОДЕЛИ ПРОФИЛЯ
|
||||
re_path(r'^catalog/profile/(?P<manufacture_id>\d+)-(?P<manufacture_name>\S*)[/*]$',
|
||||
catalog.catalog_profile_manufacture),
|
||||
# --- --- Каталог серий типового строительства
|
||||
re_path(r'^catalog/seria[/*]$', catalog.catalog_seria),
|
||||
re_path(r'^catalog/seria/(?P<seria_name_translit>[^/]*)/all(?P<seria_id>\d+)[/*]$', catalog.catalog_seria_info),
|
||||
catalog_profiles.catalog_profile_manufacture), # КАРТОЧКА ОПИСАНИЯ ПРОИЗВОДИТЕЛЯ ПРОФИЛЯ
|
||||
# --- --- КАТАЛОГ СЕРИЙ ТИПОВОГО СТРОИТЕЛЬСТВА
|
||||
re_path(r'^catalog/seria[/*]$', catalog_series.catalog_seria), # СПИСОК ВСЕХ СЕРИЙ ЗДАНИЙ
|
||||
re_path(r'^catalog/seria/(?P<seria_name_translit>[^/]*)/all(?P<seria_id>\d+)[/*]$',
|
||||
catalog_series.catalog_seria_info), # КАРТОЧКА СЕРИИ ДОМА И ЕЕ СТАТИСТИКА
|
||||
re_path(r'^seria_[^/]*/all(?P<seria_id>\d+)/\S*$', catalog.report_all_info_seria_redirect), # для старых ссылок
|
||||
# --- --- Каталог стандартных проёмов и схем открывания длч типовых серий строительства
|
||||
re_path(r'^catalog/standard_opening[/*]$', catalog.standard_opening),
|
||||
# --- --- Каталог производителей окон
|
||||
re_path(r'^catalog/company[/*]$', catalog.catalog_company),
|
||||
re_path(r'^catalog/company/(?P<company_id>\d+)-(?P<company_name_slug>\S*)[/*]$', catalog.catalog_company_detail),
|
||||
# --- --- КАТАЛОГ СТАНДАРТНЫХ ПРОЁМОВ И СХЕМ ОТКРЫВАНИЯ ДЛЧ ТИПОВЫХ СЕРИЙ СТРОИТЕЛЬСТВА
|
||||
re_path(r'^catalog/standard_opening[/*]$', catalog_openings.standard_opening), # СТРАНИЦА С ТАБЛИЦЕЙ ПРОЁМОМ
|
||||
# --- --- КАТАЛОГ ПРОИЗВОДИТЕЛЕЙ ОКОН
|
||||
re_path(r'^catalog/company[/*]$', catalog_companies.catalog_company), # СПИСОК ВСЕХ ПРОИЗВОДИТЕЛЕЙ ОКОН
|
||||
re_path(r'^catalog/company/(?P<company_id>\d+)-(?P<company_name_slug>\S*)[/*]$',
|
||||
catalog_companies.catalog_company_detail), # КАРТОЧКА ПРОИЗВОДИТЕЛЯ-УСТАНОВЩИКА ОКОН
|
||||
# --- --- КАТАЛОГ ОКОННЫХ НАБОРОВ (SetKit) — список комплектаций с переходом к сравнению
|
||||
re_path(r'^catalog/sets[/*]$', catalog.catalog_sets),
|
||||
# ЦЕНОВЫЕ ПРЕДЛОЖЕНИЯ
|
||||
# --- Одиночное окно
|
||||
# --- ОДИНОЧНОЕ ОКНО
|
||||
re_path(r'^catalog/standard_opening/price-(?P<win_width_mm>\d+)x(?P<win_height_mm>\d+)mm-tip(?P<win_id>\d+)[/*]$',
|
||||
prices.report_one_win_price), # КАНОНИЧЕСКИЙ SEO-URL СТРАНИЦЫ ЦЕН ДЛЯ ОДНОГО ПРОЕМА
|
||||
re_path(r'^tsena-odnogo-okna/(?P<win_width_mm>\d+)x(?P<win_height_mm>\d+)mm/tip(?P<win_id>\d+)[/*]$',
|
||||
prices.report_one_win_price),
|
||||
re_path(r'^next_price_one_flap_frame/idW(?P<win_id>\d+)N(?P<frame_begin_n>\d+)\S*$', prices.next_one_win_price),
|
||||
# --- Ценовая выдача
|
||||
re_path(r'^(?P<build_id>\d+)/(?P<apart_id>\d+)/(?P<slug>[\s\S]*)$', prices.report_price),
|
||||
# --- Подгружаемый фрейм ценовая выдачи
|
||||
prices.redirect_one_win_price_legacy), # LEGACY-URL: 301 -> КАНОНИЧЕСКИЙ ПУТЬ
|
||||
re_path(r'^next_price_one_flap_frame/idW(?P<win_id>\d+)N(?P<frame_begin_n>\d+)\S*$',
|
||||
prices.next_one_win_price), # ПОДГРУЖАЕМЫЙ ФРЕЙМ С ЦЕНОВЫМИ ПРЕДЛОЖЕНИЯМИ ДЛЯ ОДНОГО ПРОЕМА
|
||||
# --- ЦЕНОВАЯ ВЫДАЧА (НОВЫЙ РОУТИНГ)
|
||||
# НОВЫЙ КРАСИВЫЙ URL С ПРЕФИКСАМИ SERIAID, APPARTAD, ADDRESSID
|
||||
re_path(r'^price/seriaID(?P<seria_id>\d+)--(?P<seria_slug>[^/]+)/appartID(?P<apart_id>\d+)/addressID(?P<address_id>\d+)--(?P<address_slug>[^/]+)/?$', prices.report_price_new),
|
||||
# --- ПОДГРУЖАЕМЫЙ ФРЕЙМ ЦЕНОВОЙ ВЫДАЧИ (ОСТАВЛЯЕМ СТАРЫЙ)
|
||||
re_path(r'^next_price_frame/idA(?P<apart_id>\d+)MDPO(?P<mount_dim_per_offer>\d+)LON(?P<address_longitude>\d+)'
|
||||
r'LAT(?P<address_latitude>\d+\.*\d*)N(?P<frame_begin_n>\d+\.*\d*)\S*[/*]$', prices.next_price_frame),
|
||||
# --- СТАРЫЙ URL ЦЕНОВОЙ ВЫДАЧИ (ДОБАВИМ РЕДИРЕКТ) ДЛЯ ПОИСКОВИКОВ
|
||||
# --- НЕ УДАЛЯТЬ! КАРТА С СЕРИЯМИ ДОМОВ ИСПОЛЬЗУЕТ ЭТОТ РОУТИНГ, Т.К. ТАКИЕ URL КОРОЧЕ И ДЕЛАЮТ JS КОПАКТНЕЕ
|
||||
re_path(r'^(?P<build_id>\d+)/(?P<apart_id>\d+)/(?P<slug>[\s\S]*)$', prices.report_price_legacy_redirect),
|
||||
# СРАВНЕНИЕ ОКОННЫХ НАБОРОВ
|
||||
re_path(r'^compare_sets/(?P<to_compare>[\s\S]+|.*)$', report1.compare_offers), # дубль для старых ссылок
|
||||
re_path(r'^compare_offers/(?P<to_compare>[\s\S]+|.*)$', report1.compare_offers),
|
||||
re_path(r'^specification_set/\d$', views.main_init), # заглушка (позже будет спецификация оконного набора)
|
||||
# отображение всех составлющих рейтинга
|
||||
re_path( r'^show_rating_components/(?P<win_set>\d+)$', report1.show_rating_components),
|
||||
# СЛУЖЕБНЫЕ СТРАНИЦЫ (для администратора)
|
||||
# --- страничка "главная сервис-утилит"
|
||||
re_path(r'^service[/*]$', service.service),
|
||||
# --- страничка для тестирования верстки текста в блоге
|
||||
re_path(r'^service/tmp[/*]$', service.tmp),
|
||||
# --- страничка "нет доступа"
|
||||
re_path(r'^service/not-denice[/*]$', service.not_denice),
|
||||
# --- создание файлов sitemap.xml
|
||||
re_path(r'^service/make_sitemaps[/*]$', service.make_site_maps),
|
||||
|
||||
]
|
||||
|
||||
if DEBUG:
|
||||
urlpatterns += static(MEDIA_URL, document_root=MEDIA_ROOT)
|
||||
|
||||
# ___ ____ _ _____ _ _ _____ _
|
||||
# | | | | \ ___| |_ _ _ ___ |_ _|___ ___| | |_ ___ ___ | _ |___ ___ ___| |
|
||||
# |_ | | | | -_| . | | | . | | | | . | . | | . | .'| _| | __| .'| | -_| |
|
||||
# |_| |____/|___|___|___|_ | |_| |___|___|_|___|__,|_| |__| |__,|_|_|___|_|
|
||||
# |___|
|
||||
# Динамическая генерация URL patterns для всех файлов в корне public/
|
||||
# (кроме служебных файлов, начинающихся с точки)
|
||||
# В DEV режиме и при ALLOW_MEDIA_SERVE=True отдаются через Django,
|
||||
# в PRODUCTION это должно обслуживать Nginx/WhiteNoise
|
||||
PUBLIC_ROOT_URLPATTERNS = [
|
||||
path(filename, _serve_public_root_file, {'path': filename})
|
||||
for filename in _iter_public_root_files()
|
||||
]
|
||||
|
||||
# Добавляем static файлы для корня ДО основных URL patterns
|
||||
# (чтобы отдавать файлы быстро и не проверять остальные рулы)
|
||||
urlpatterns = [*PUBLIC_ROOT_URLPATTERNS, *urlpatterns]
|
||||
|
||||
handler404 = 'web.views.handler404'
|
||||
handler400 = 'web.views.handler400'
|
||||
handler403 = 'web.views.handler403'
|
||||
handler500 = 'web.views.handler500'
|
||||
|
||||
# Для локального тестирования production конфига: отдача медиа через Django
|
||||
# В реальном production медиа обслуживает Nginx!
|
||||
if DEBUG or env.bool('ALLOW_MEDIA_SERVE', default=False):
|
||||
from django.views.static import serve as serve_static
|
||||
# Проверяем что директория медиа существует
|
||||
if os.path.isdir(MEDIA_ROOT):
|
||||
# Добавляем URL pattern для отдачи медиа файлов
|
||||
urlpatterns += [
|
||||
re_path(
|
||||
r'^media/(?P<path>.*)$',
|
||||
serve_static,
|
||||
{'document_root': MEDIA_ROOT},
|
||||
name='media'
|
||||
),
|
||||
]
|
||||
|
||||
|
||||
if DEBUG:
|
||||
# --- страничка для тестирования верстки текста в блоге
|
||||
urlpatterns += [re_path(r'^blog/tmp[/*]$', service.tmp),]
|
||||
# --- странички для тестирования страниц с кодами ошибок
|
||||
urlpatterns += [
|
||||
re_path(r'^400$', views.handler400),
|
||||
re_path(r'^403$', views.handler403),
|
||||
re_path(r'^404$', views.handler404),
|
||||
re_path(r'^500$', views.handler500),
|
||||
]
|
||||
# ___ ____ _ _____ _ _ _____ _
|
||||
# | | | | \ ___| |_ _ _ ___ |_ _|___ ___| | |_ ___ ___ | _ |___ ___ ___| |
|
||||
# |_ | | | | -_| . | | | . | | | | . | . | | . | .'| _| | __| .'| | -_| |
|
||||
# |_| |____/|___|___|___|_ | |_| |___|___|_|___|__,|_| |__| |__,|_|_|___|_|
|
||||
# |___|
|
||||
urlpatterns = [path('__debug__/', include('debug_toolbar.urls')), *urlpatterns]
|
||||
|
||||
|
||||
@@ -4,25 +4,31 @@
|
||||
<meta charset="utf-8" />
|
||||
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
||||
<meta http-equiv="content-language" content="ru" />
|
||||
<meta http-equiv="Date" content="{% block Date4Meta %}{% now "c" %}{% endblock %}" />
|
||||
<meta http-equiv="Last-Modified" content="{% block Last4Meta %}{% now "c" %}{% endblock %}" />
|
||||
<meta http-equiv="Expires" content="{% block Expires4Meta %}{% now "c" %}{% endblock %}" />
|
||||
<meta http-equiv="Date" content="{% block Date4Meta %}{% now "Y-m-d" %}{% endblock %}" />
|
||||
<meta http-equiv="Last-Modified" content="{% block Last4Meta %}{% now "Y-m-d" %}{% endblock %}" />
|
||||
<meta http-equiv="Expires" content="{% block Expires4Meta %}{% now "Y-m-d" %}{% endblock %}" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||
<meta name="description" content="{% block Description %}{{ META_DESCRIPTION|default:"" }}Здесь вы можете узнать цены и скидки на пластиковые окна. Просто введите адрес, укажите планировку квартиры и узнайте размеры проёмов, актуальные предложения и стоимость установки окон. Сравнивайте характеристики стеклопакетов, оконных профилей, схем открывания, расценки, условия установки, обслуживания и гарантии.{% endblock %}" />
|
||||
<meta name="keywords" content="{% block Keywords %}цены на пластиковые окна, скидки на пластиковые окна, окна в квартиру, размеры окон в доме серии, скидки на пластиковые окна, характеристики пластиковых окон{{ META_KEYWORDS|default:"" }}{% endblock %}" />
|
||||
<meta name="author" content="{% block Author4Meta %}{% endblock %}OKNARDIA.RU" />
|
||||
<meta name="copyright" lang="ru" content="{% block CopyrightAuthor4Meta %}{% endblock %}OKNARDIA.RU" />
|
||||
<meta name="author" content="{% block Author4Meta %}{% endblock %} OKNARDIA.RU" />
|
||||
<meta name="copyright" lang="ru" content="{% block CopyrightAuthor4Meta %}{% endblock %} OKNARDIA.RU" />
|
||||
<meta name="robots" content="index,follow" />
|
||||
<meta name="document-state" content="{{ META_DOCUMENT_STATE|default:"Dynamic" }}" />
|
||||
<meta name="generator" content="OKNARDIA 0.3β by Python/Django" />
|
||||
<title>{% block Title %}{% endblock %} : ОКНАРДИЯ</title>
|
||||
<link rel="alternate" href="https://oknardia.ru" hreflang="ru-ru" />
|
||||
<link rel="icon" type="image/png" href="{{ request.scheme }}://{{ request.get_host }}/favicon.png" sizes="96x96" />
|
||||
<link rel="icon" type="image/svg+xml" href="{{ request.scheme }}://{{ request.get_host }}/favicon.svg" />
|
||||
<link rel="shortcut icon" href="/favicon.ico" />
|
||||
<link rel="apple-touch-icon" sizes="180x180" href="{{ request.scheme }}://{{ request.get_host }}/apple-touch-icon.png" />
|
||||
<meta name="apple-mobile-web-app-title" content="oknardia.ru" />
|
||||
<link rel="manifest" href="{{ request.scheme }}://{{ request.get_host }}/site.webmanifest" />
|
||||
<link href="{% static 'css/bootstrap.min.css' %}" rel="stylesheet" type="text/css" />{# <link href="//maxcdn.bootstrapcdn.com/bootstrap/3.3.6/css/bootstrap.min.css" rel="stylesheet" type="text/css" />#}
|
||||
<link href="{% static 'css/bootstrap-theme.min.css' %}" rel="stylesheet" type="text/css" />{# <link href="//maxcdn.bootstrapcdn.com/bootstrap/3.3.6/css/bootstrap-theme.min.css" rel="stylesheet" type="text/css" />#}
|
||||
<link href="{% static 'css/oknardia1.css' %}" rel="stylesheet" type="text/css" />{% block Top_CSS1 %}{% endblock %}{% block Top_CSS2 %}{% endblock %}{% block Top_CSS3 %}{% endblock %}
|
||||
<script src="{% static 'js/jquery-2.1.1.min.js' %}" type="text/javascript"></script>{# <script src="//ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js" type="text/javascript"></script>#}
|
||||
<script src="{% static 'js/bootstrap.min.js' %}" type="text/javascript"></script>{# <script src="//maxcdn.bootstrapcdn.com/bootstrap/3.3.6/js/bootstrap.min.js" type="text/javascript"></script>#}{% block Top_JS1 %}{% endblock %}{% block Top_JS2 %}{% endblock %}{% block Top_JS3 %}{% endblock %}{% block Top_JS4 %}{% endblock %}{% block Top_JS5 %}{% endblock %}{% block Top_Meta1 %}{% endblock %}
|
||||
<script type="text/javascript">$(document).ready(function(){ $('#login-logout').load('/login-logout' ); })</script>
|
||||
{# Аналитика: Google Analytics 4, Yandex.Metrika, Top.Mail.Ru #}<script src="{% static 'js/analytics.js' %}" type="text/javascript"></script>
|
||||
{# Модуль авторизации: управление dropdown меню логина/логаута #}<script src="{% static 'js/auth.js' %}" type="text/javascript"></script>{% block ADD_TO_HEAD %}{% endblock %}
|
||||
</head>
|
||||
|
||||
<body{% block Add_Body_Attribute %}{% endblock %}>
|
||||
@@ -48,13 +54,13 @@
|
||||
<li><a href="/stat_all">Статистика</a></li>
|
||||
<li role="separator" class="divider"></li>
|
||||
<li><a href="/contact">Контакты</a></li>
|
||||
<li><a href="/blogpost/2/My_zhdem_vashi_prajs-listy!">Сотрудничество</a></li>
|
||||
<li><a href="/blogpost/2/myi-zhdyom-vashi-prajs-listyi">Сотрудничество</a></li>
|
||||
<li><a href="/tariff">Услуги и тарифы</a></li>
|
||||
<!-- li class="divider"></li>
|
||||
<li><a href="#" rel="nofollow">Обратная связь</a></li -->
|
||||
</ul>
|
||||
</li>
|
||||
<li class="dropdown" id="login-logout"><!--- Сюда подгружают AJAX-ом блок login-logout ---><small><br />Авторизации.<noscript style="color:red;">Для авторизации необходимо включить JavaScript.</noscript></small></li>
|
||||
<li class="dropdown" id="login-logout"><a href="#" onclick="return openLoginLogout(event);" rel="nofollow"><span class="glyphicon glyphicon-user"></span> {% if LOGGED_USER != "" %}{{ user.username|truncatechars:12 }}{% else %}Вход{% endif %}</a><small>{# Авторизации. #}<noscript style="color:red;">Для авторизации необходимо включить JavaScript.</noscript></small></li>
|
||||
</ul>
|
||||
</div>
|
||||
</nav>{% endblock %}
|
||||
@@ -65,25 +71,10 @@
|
||||
{% block Bottom_Nav_Bar %}
|
||||
<div class="row panel-footer">
|
||||
<div class="col-xs-12">
|
||||
<span style="top:-200px;left:-8000px;position: absolute;"><script type="text/javascript">
|
||||
{# <!-- Google Analylics --> #}(function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){(i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m)})(window,document,'script','//www.google-analytics.com/analytics.js','ga');
|
||||
ga('create', 'UA-9116991-5', 'auto'); ga('send', 'pageview');
|
||||
{# <!-- Rating@Mail.ru counter --> #}var _tmr=_tmr||[];_tmr.push({id:"2018432",type:"pageView",start:(new Date()).getTime()});(function (d,w,id){if(d.getElementById(id))return;var ts=d.createElement("script");ts.type="text/javascript";ts.async=true;ts.id=id;ts.src=(d.location.protocol=="https:"?"https:":"http:")+"//top-fwz1.mail.ru/js/code.js";var f=function(){var s=d.getElementsByTagName("script")[0];s.parentNode.insertBefore(ts, s);};if(w.opera=="[object Opera]"){ d.addEventListener("DOMContentLoaded",f,false);}else{f();}})(document,window,"topmailru-code");
|
||||
</script><noscript><div style="position:absolute;left:-10000px;">
|
||||
{# <!-- Rating@Mail.ru nosript --> #}<img src="//top-fwz1.mail.ru/counter?id=2018432;js=na" style="border:0;height:1px;width:1px" alt="" />
|
||||
{# <!-- Yandex.Metrika counter --> #}<img src="//mc.yandex.ru/watch/32997984" style="border:0;height:1px;width:1px" alt="" />{# <!-- /Yandex.Metrika counter --> #}
|
||||
</div></noscript>
|
||||
{#<!-- Rating@Mail.ru logo -->#}<a target="_blank" href="http://top.mail.ru/jump?from=2018432"><img src="//top-fwz1.mail.ru/counter?id=2018432;t=216;l=1" style="border:0;padding-top:8px;" rel="nofollow" alt="Рейтинг@Mail.ru"></a>{#<!-- //Rating@Mail.ru logo -->#}
|
||||
{# <!-- Yandex.Metrika informer --> #}<a href="https://metrika.yandex.ru/stat/?id=32997984&from=informer" target="_blank" rel="nofollow"><img src="https://informer.yandex.ru/informer/32997984/3_0_E0E0E0FF_C0C0C0FF_0_pageviews" style="width:88px; height:31px; border:0;" alt="Яндекс.Метрика" title="Яндекс.Метрика: данные за сегодня (просмотры, визиты и уникальные посетители)" onclick="try{Ya.Metrika.informer({i:this,id:32997984,lang:'ru'});return false}catch(e){}" /></a>{# <!-- /Yandex.Metrika informer --> #}
|
||||
{# <!-- begin of Top100 code --> #}<span id="rambler"><script id="top100Counter" type="text/javascript" src="//counter.rambler.ru/top100.jcn?3148853"></script><noscript><a href="http://top100.rambler.ru/navi/3148853/"><img src="http://counter.rambler.ru/top100.cnt?3148853" alt="Rambler's Top100" border="0"/></a></noscript></span>{# <!-- end of Top100 code --> #}
|
||||
<script type="text/javascript"><!--
|
||||
{#<!--LiveInternet counter-->#}document.write("<a href='//www.liveinternet.ru/click' target=_blank><img src='//counter.yadro.ru/hit?t50.2;r"+escape(document.referrer)+((typeof(screen)=="undefined")?"":";s"+screen.width+"*"+screen.height+"*"+(screen.colorDepth?screen.colorDepth:screen.pixelDepth))+";u"+escape(document.URL)+";"+Math.random()+"' alt='' title='LiveInternet' style='border:0;padding-top:8px;'><\/a>");
|
||||
{# <!-- Yandex.Metrika counter --> #}(function(d,w,c){(w[c]=w[c]||[]).push(function(){try{w.yaCounter32997984=new Ya.Metrika({id:32997984,clickmap:true,trackLinks:true,accurateTrackBounce:true,webvisor:true,trackHash:true});}catch(e){}});var n=d.getElementsByTagName("script")[0],s=d.createElement("script"),f=function(){n.parentNode.insertBefore(s,n);};s.type="text/javascript";s.async=true;s.src="https://mc.yandex.ru/metrika/watch.js";if(w.opera=="[object Opera]"){d.addEventListener("DOMContentLoaded",f,false);}else{ f();}})(document,window,"yandex_metrika_callbacks");
|
||||
//--></script>{# <!--/LiveInternet--> #}
|
||||
</span>
|
||||
<small>© oknardia.ru, 2015-{% now "Y" %}. <a href="/blogpost/18/Ob-avtorskih-pravah">Все права защищены</a>.<!--- Время отработки скрипта: {{ ticks }}{{ TAU }} сек---></small>
|
||||
<small>© oknardia.ru, 2015-{% now "Y" %}. <a href="/blogpost/18/Ob-avtorskih-pravah">Все права защищены</a>.<!--- Время отработки скрипта: {{ ticks }}{{ TAU }} сек---> {{ ticks }}{{ TAU }} сек </small>
|
||||
</div>
|
||||
</div>{% endblock %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
{# ######################################## Нижнее меню-футер КОНЕЦ ######################################## #}
|
||||
|
||||
{# Модальное окно SOCIAL LOGIN НАЧАЛО #}
|
||||
|
||||
@@ -1,20 +1,121 @@
|
||||
{% extends "base.html" %}{% load static %}
|
||||
|
||||
{% block Title %}Блоги: Стр.{{ PAGE_BACK|add:"1" }}{% endblock %}
|
||||
{% block Title %}Блог Окнардии для компаний-поставщиков окон и их клиентов — Страница {{ PAGE_BACK|add:"1" }}{% endblock %}
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}Блоги «Окнардия» :: {% for i1 in DIM_BLOGPOST %}{{ i1.HEADER_D }}{% if not forloop.last %}, {% endif %}{% endfor %}{% endblock %}
|
||||
{% block Description %}Блог Окнардии для компаний-поставщиков окон и их клиентов: публикации о пластиковых окнах, продвижении услуг замены окон, ценах и трендах — Страница {{ PAGE_BACK|add:"1" }}{% endblock %}
|
||||
|
||||
{% block Keywords %}oknardia, окнардия, blogs, блоги, публикации, цены пластиковых окон, стоимость пластиковых окон, скидки на пластиковые окна, предложения пластиковых окон, {{ META_KEYWORDS|default:"" }} {% endblock %}
|
||||
{% block Keywords %}{{ META_KEYWORDS }}{% endblock %}
|
||||
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Date4Meta %}{% if META_DATA_PUB %}{{ META_DATA_PUB|date:"Y-m-d" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Last4Meta %}{% if META_DATA_MODIFY %}{{ META_DATA_MODIFY|date:"Y-m-d" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Author4Meta %}: Блоги{% endblock %}
|
||||
{% block Author4Meta %}Окнардия{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}: Блоги{% endblock %}
|
||||
{% block CopyrightAuthor4Meta %}Окнардия{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- Canonical (текущая страница) и pagination разметка --> #}
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/blog/P{{ PAGE_BACK }}" />
|
||||
{% if PAGE_BACK > 0 %}<link rel="prev" href="{{ request.scheme }}://{{ request.get_host }}/blog/P{{ PAGE_BACK|add:'-1' }}" />{% endif %}
|
||||
{% if FORW_BUTTON %}<link rel="next" href="{{ request.scheme }}://{{ request.get_host }}/blog/P{{ PAGE_BACK|add:'1' }}" />{% endif %}
|
||||
{# <!-- Meta-теги для улучшения индексирования в социальных сетях (B2B) --> #}
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:type" content="website" />
|
||||
<meta property="og:title" content="Блог Окнардии для компаний-поставщиков окон и их клиентов" />
|
||||
<meta property="og:description" content="Публикации о пластиковых окнах, продвижении услуг замены окон, ценах и трендах — Страница {{ PAGE_BACK|add:"1" }}" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/blog/P{{ PAGE_BACK }}" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
{% if META_IMAGE %}<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}{{ META_IMAGE }}" />{% else %}<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/oknardia_logo.svg" />{% endif %}
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:title" content="Блог Окнардии — для компаний и их клиентов" />
|
||||
<meta name="twitter:description" content="Статьи о продвижении услуг замены окон, ценах и трендах в оконной индустрии — Страница {{ PAGE_BACK|add:"1" }}" />
|
||||
{% if META_IMAGE %}<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}{{ META_IMAGE }}" />{% else %}<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/oknardia_logo.svg" />{% endif %}
|
||||
{# <!-- /Meta-теги --> #}{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}{# <!-- Schema.org JSON-LD разметка для списка блога (B2B для компаний и их клиентов) --> #}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "CollectionPage",
|
||||
"name": "Блог Окнардии — для компаний-поставщиков окон и их клиентов",
|
||||
"description": "Блог Окнардии для компаний-поставщиков окон и их клиентов: публикации о пластиковых окнах, продвижении услуг замены окон, ценах и трендах. Ресурс для расширения продаж и улучшения видимости в выдаче поисковиков.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/blog/P{{ PAGE_BACK }}",
|
||||
"image": "{% if META_IMAGE %}{{ request.scheme }}://{{ request.get_host }}{{ META_IMAGE }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/oknardia_logo.svg{% endif %}",
|
||||
"audience": [
|
||||
{
|
||||
"@type": "AudienceType",
|
||||
"name": "B2B: Компании-поставщики и производители оконных конструкций"
|
||||
},
|
||||
{
|
||||
"@type": "AudienceType",
|
||||
"name": "B2C: Клиенты компаний, ищущие информацию о заменке окон"
|
||||
}
|
||||
],
|
||||
"breadcrumb": {
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Блог",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/blog/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 3,
|
||||
"name": "Страница {{ PAGE_BACK|add:'1' }}",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/blog/P{{ PAGE_BACK }}"
|
||||
}
|
||||
]
|
||||
},
|
||||
"mainEntity": {
|
||||
"@type": "Blog",
|
||||
"name": "Блог Окнардии",
|
||||
"alternateName": "Блог для компаний-поставщиков окон и их клиентов",
|
||||
"description": "Профессиональный блог для компаний, занимающихся производством, поставкой и установкой пластиковых окон, а также их клиентов. Публикации об оконных конструкциях, продвижении услуг замены окон, ценах, трендах и инновациях в оконной индустрии.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/blog/",
|
||||
"image": "{{ request.scheme }}://{{ request.get_host }}/static/img/oknardia_logo.svg",
|
||||
"audience": [
|
||||
{
|
||||
"@type": "AudienceType",
|
||||
"name": "Компании-поставщики, производители, установщики окон"
|
||||
},
|
||||
{
|
||||
"@type": "AudienceType",
|
||||
"name": "Конечные клиенты, ищущие информацию об окнах и услугах"
|
||||
}
|
||||
],
|
||||
"blogPosts": [
|
||||
{% for POST in DIM_BLOGPOST %}
|
||||
{
|
||||
"@type": "BlogPosting",
|
||||
"headline": "{{ POST.HEADER|escapejs }}",
|
||||
"description": "{% if POST.META_DESC %}{{ POST.META_DESC|escapejs }}{% else %}Публикация в блоге Окнардии{% endif %}",
|
||||
"image": "{% if POST.IMG_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ POST.IMG_BLOG|escapejs }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/oknardia_logo.svg{% endif %}",
|
||||
"datePublished": "{{ POST.PUB_DAT|date:'Y-m-d' }}T{{ POST.PUB_DAT|date:'H:i:s' }}Z",
|
||||
"dateModified": "{{ POST.MOD_DAT|date:'Y-m-d' }}T{{ POST.MOD_DAT|date:'H:i:s' }}Z",
|
||||
"author": {
|
||||
"@type": "Person",
|
||||
"name": "{{ POST.NAME1 }}{% if POST.NAME2 %} {{ POST.NAME2 }}{% endif %}"
|
||||
},
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/blogpost/{{ POST.POST_ID }}/{{ POST.HEADER_T }}",
|
||||
"keywords": "{% if POST.META_KW %}{{ POST.META_KW|escapejs }}{% else %}блог, публикация, окна, поставщики{% endif %}"
|
||||
}{% if not forloop.last %},{% endif %}
|
||||
{% endfor %}
|
||||
]
|
||||
}
|
||||
}
|
||||
</script>
|
||||
{# <!-- /Schema.org JSON-LD --> #}{% endblock %}
|
||||
|
||||
{% block Top_JS3%}
|
||||
<script>
|
||||
@@ -24,29 +125,35 @@ $(window).load(function(){var images = $('.half');images.each(function(i){$(this
|
||||
|
||||
{% block Main_Content %}
|
||||
<div class="container-fluid">
|
||||
{# <!--- Хлебные крошки --> #}<div class="row">
|
||||
{# Хлебные крошки #}
|
||||
<div class="row">
|
||||
<div class="col-md-11 col-xs-12">
|
||||
<ol class="breadcrumb">
|
||||
<li><a href="/">Главная</a></li>
|
||||
<li><a href="/blog/">Блог</a></li>
|
||||
<li>Стр.{{ PAGE_BACK|add:"1" }}</li>
|
||||
<li>Страница {{ PAGE_BACK|add:"1" }}</li>
|
||||
</ol>
|
||||
<h1>Блог</h1>
|
||||
</div>
|
||||
</div>{# <!--- /Хлебные крошки ---> #}
|
||||
</div>
|
||||
|
||||
{# Список постов #}
|
||||
{% for POST in DIM_BLOGPOST %}<div class="row">
|
||||
<div class="col-md-11 col-xs-12 blog-list-header">
|
||||
<hr class="dotted-black" />
|
||||
<p>{{ POST.PUB_DAT|date:"d.F.Y (l) H:i" }}</p>
|
||||
<p><img src="/media/{{ POST.USER_AVATAR }}" /> {% if POST.NAME1 != "" or POST.NAME2 != "" %} <i>{{ POST.NAME1 }}{% if POST.NAME2 != "" %} {{ POST.NAME2 }}{% endif %}</i>{% endif %}</p>
|
||||
<p><time datetime="{{ POST.PUB_DAT|date:'Y-m-d\TH:i:s\Z' }}">{{ POST.PUB_DAT|date:"d.F.Y (l) H:i" }}</time></p>
|
||||
<p><img src="/media/{{ POST.USER_AVATAR }}" alt="{{ POST.NAME1 }}{% if POST.NAME2 %} {{ POST.NAME2 }}{% endif %}" />
|
||||
{% if POST.NAME1 != "" or POST.NAME2 != "" %}<i>{{ POST.NAME1 }}{% if POST.NAME2 != "" %} {{ POST.NAME2 }}{% endif %}</i>{% endif %}
|
||||
</p>
|
||||
<h2>{{ POST.HEADER|safe }}</h2>
|
||||
</div>
|
||||
<div class="col-md-11 blog-list-tizer">
|
||||
{# <!--- Тизер поста в блоге ---> #}{{ POST.CONTENT_CUT|safe|truncatechars:4096 }}{# <!--- /Тизер поста в блоге ---> #}
|
||||
{# Тизер поста в блоге #}{{ POST.CONTENT_CUT|safe|truncatechars:4096 }}{# /Тизер поста в блоге #}
|
||||
{% if POST.CUT_TEXT != "NONE" %}<p><a href="/blogpost/{{ POST.POST_ID }}/{{ POST.HEADER_T }}?page-back={{ PAGE_BACK }}" class="btn btn-default">{{ POST.CUT_TEXT|safe }}</a></p>{% endif %}
|
||||
</div>
|
||||
</div>{% endfor %}
|
||||
{# <!--- Листалка ---> #}<div class="row">
|
||||
|
||||
{# Листалка пагинации #}<nav class="row">
|
||||
<div class="col-md-11 col-xs-12">
|
||||
<hr class="dotted-black" />
|
||||
<nav aria-label="переходы на страницы">
|
||||
@@ -66,13 +173,10 @@ $(window).load(function(){var images = $('.half');images.each(function(i){$(this
|
||||
</ul>
|
||||
</nav>
|
||||
</div>
|
||||
</div>{# <!--- /Листалка: ---> #}
|
||||
{# <!--- Баннер ---> #}<div class="row"><div class="col-md-12 col-xs-12">{% include "ad/bannet-wide.html" %}</div></div>{# <!--- Баннер: конец --- #}
|
||||
</nav>
|
||||
|
||||
{# Баннер #}
|
||||
<div class="row"><div class="col-md-12 col-xs-12">{% include "ad/bannet-wide.html" %}</div></div>
|
||||
</div>{% endblock %}
|
||||
|
||||
{% comment %}
|
||||
{% block Top_Nav_Bar %}
|
||||
{# ОТЛАДКА, ГАСИМ ВЕРХНЕЕ МЕНЮ #}
|
||||
{% endblock %}
|
||||
{% endcomment %}
|
||||
|
||||
|
||||
@@ -1,57 +1,103 @@
|
||||
{% extends "base.html" %}{% load static %}
|
||||
|
||||
{% block Title %}Блог :: {{ HEADER|striptags }}{% endblock %}
|
||||
{% block Title %}{{ HEADER|striptags }}{% endblock %}
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}{{ TIZER|striptags|truncatewords:25 }}{% endblock %}
|
||||
{% block Description %}{% if META_DESC %}{{ META_DESC }}{% else %}{{ TIZER|striptags|truncatewords:25 }}{% endif %}{% endblock %}
|
||||
|
||||
{% block Keywords %}oknardia, окнардия, blog, блог, публикация, {{ HEADER|striptags }}{% endblock %}
|
||||
{% block Keywords %}{% if META_KW %}{{ META_KW }}{% else %}oknardia, окнардия, блог, публикация, {{ HEADER|striptags }}{% endif %}{% endblock %}
|
||||
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"Y-m-d" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"Y-m-d" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Author4Meta %}: {{ USERNAME }}{% if NAME1 != "" or NAME2 != "" %} ({{ NAME1 }}{% if NAME2 != "" %} {{ NAME2 }}{% endif %}){% endif %}{% endblock %}
|
||||
{% block Author4Meta %}{{ NAME1 }}{% if NAME2 %} {{ NAME2 }}{% endif %} ({{ USERNAME }}) в Блоге Окнардия{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}: Блоги{% endblock %}
|
||||
{% block CopyrightAuthor4Meta %}Окнардия — Блог{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}
|
||||
{# <!-- Дополнительные Metatags --> #}{% if NAME1 != '' or NAME2 != '' %}
|
||||
<meta itemprop="author" content="{{ NAME1 }} {{ NAME2 }}" />{% endif %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/media/{{ USER_AVATAR }}" />
|
||||
<meta itemprop="datePublished" content="{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}" />
|
||||
<span itemprop="publisher" itemscope itemtype="http://schema.org/Organization"><meta itemprop="name" content="ОКНАРДИЯ: сборник цен на пластиковые окна" /></span>
|
||||
<span itemprop="author" itemscope itemtype="http://schema.org/Person"><meta itemprop="name" content="{% if NAME1 != '' or NAME2 != '' %}{{ NAME1 }}{% if NAME2 != '' %} {{ NAME2 }}{% endif %}{% endif %}" /></span>
|
||||
<meta itemprop="articleSection" content="ОКНАРДИЯ: Блог «{{ USERNAME }}»" />
|
||||
<meta itemprop="headline" content="{{ TIZER|striptags|truncatewords_html:25 }}" />
|
||||
<meta name="news_keywords" content="{{ HEADER|striptags }}" />
|
||||
<link rel="canonical" href="https://oknardia.ru/blogpost/{{ ID }}/{{ HEADER_T }}" />
|
||||
<link rel="standout" href="https://oknardia.ru/blogpost/{{ ID }}/{{ HEADER_T }}" />
|
||||
{# <!-- Разметка для соц-сетей Facebook Open Graph --> #}<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
<meta property="fb:app_id" content="258354027974262" />
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
{# <!-- Разметка OG-теги для соц-сетей и мессенджеров --> #}<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="https://oknardia.ru/blogpost/{{ ID }}/{{ HEADER_T }}" />
|
||||
{% block Top_Meta1 %}{# <!-- Canonical разметка --> #}
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/blogpost/{{ ID }}/{{ HEADER_T }}" />{% if not BACK_DISABLE %}
|
||||
<link rel="prev" href="{{ request.scheme }}://{{ request.get_host }}/blogpost/{{ BACK_ID }}/{{ BACK_HEADER_T }}?page-back={{ PAGE_BACK|add:'-1' }}" />{% endif %}{% if not FORW_DISABLE %}
|
||||
<link rel="next" href="{{ request.scheme }}://{{ request.get_host }}/blogpost/{{ FORW_ID }}/{{ FORW_HEADER_T }}?page-back={{ PAGE_BACK }}" />{% endif %}
|
||||
{# <!-- Meta-теги для социальных сетей (B2B/B2C для компаний и клиентов) --> #}
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:type" content="article" />
|
||||
<meta property="og:title" content="{{ HEADER|striptags }} | oknardia.ru" />
|
||||
<meta property="og:description" content="{{ TIZER|striptags|truncatewords_html:25 }}" />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="{{ HEADER|striptags }} | oknardia.ru" />
|
||||
<meta name="twitter:description" content="{{ TIZER|striptags|truncatewords_html:25 }}" />
|
||||
<meta name="twitter:card" content="summary">
|
||||
<meta property="og:title" content="{{ HEADER|striptags }}" />
|
||||
<meta property="og:description" content="{% if META_DESC %}{{ META_DESC|escapejs|truncatewords:30 }}{% else %}{{ TIZER|striptags|truncatewords:25 }}{% endif %}" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/blogpost/{{ ID }}/{{ HEADER_T }}" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:domain" content="oknardia.ru" />
|
||||
<meta property="twitter:url" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}">{% endblock %}
|
||||
<meta name="twitter:title" content="{{ HEADER|striptags }}" />
|
||||
<meta name="twitter:description" content="{% if META_DESC %}{{ META_DESC|escapejs|truncatewords:30 }}{% else %}{{ TIZER|striptags|truncatewords:25 }}{% endif %}" />
|
||||
<meta name="twitter:image" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
{# <!-- /Meta-теги --> #}{% endblock %}
|
||||
|
||||
{% block Top_JS3%}
|
||||
<script type="text/javascript">$(window).load(function(){var images = $('.half');images.each(function(i){$(this).width($(this).width()/2);});});</script>{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}{# <!-- Schema.org JSON-LD разметка для отдельного блог-поста --> #}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BlogPosting",
|
||||
"headline": "{{ HEADER|escapejs }}",
|
||||
"description": "{% if META_DESC %}{{ META_DESC|escapejs }}{% else %}{{ TIZER|striptags|escapejs|truncatewords:25 }}{% endif %}",
|
||||
"image": "{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG|escapejs }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}",
|
||||
"datePublished": "{{ PUB_DAT|date:'Y-m-d' }}T{{ PUB_DAT|date:'H:i:s' }}Z",
|
||||
"dateModified": "{% if PUB_MODIFY %}{{ PUB_MODIFY|date:'Y-m-d' }}T{{ PUB_MODIFY|date:'H:i:s' }}Z{% else %}{{ PUB_DAT|date:'Y-m-d' }}T{{ PUB_DAT|date:'H:i:s' }}Z{% endif %}",
|
||||
"mainEntityOfPage": {
|
||||
"@type": "WebPage",
|
||||
"@id": "{{ request.scheme }}://{{ request.get_host }}/blogpost/{{ ID }}/{{ HEADER_T }}"
|
||||
},
|
||||
"breadcrumb": {
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Блог",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/blog/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 3,
|
||||
"name": "Страница {{ PAGE_BACK|add:'1' }}",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/blog/P{{ PAGE_BACK }}"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 4,
|
||||
"name": "{{ HEADER|escapejs }}",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/blogpost/{{ ID }}/{{ HEADER_T }}"
|
||||
}
|
||||
]
|
||||
},
|
||||
"author": {
|
||||
"@type": "Person",
|
||||
"name": "{{ NAME1 }}{% if NAME2 %} {{ NAME2 }}{% endif %}"
|
||||
},
|
||||
"publisher": {
|
||||
"@type": "Organization",
|
||||
"name": "ОКНАРДИЯ",
|
||||
"logo": {
|
||||
"@type": "ImageObject",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/static/img/oknardia_logo.svg"
|
||||
}
|
||||
},
|
||||
"keywords": "{% if META_KW %}{{ META_KW|escapejs }}{% else %}блог, публикация, окна, поставщики{% endif %}"
|
||||
}
|
||||
</script>
|
||||
{# <!-- /Schema.org JSON-LD --> #}{% endblock %}
|
||||
|
||||
{% block Main_Content %}
|
||||
<dIv class="container-fluid" itemscope itemtype="http://schema.org/Article">
|
||||
<div class="row">{% if not IS_ARCHIVE %}
|
||||
@@ -78,7 +124,7 @@
|
||||
</DIv>
|
||||
</DiV>
|
||||
{# Листалка: НАЧАЛО #}<div class="row">
|
||||
<div class="col-md-11 col-xs-12">
|
||||
<nav class="col-md-11 col-xs-12">
|
||||
<hr class="dotted-black" />
|
||||
{% if not IS_ARCHIVE %}<nav aria-label="перелистывание записей блога">
|
||||
<ul class="pager">
|
||||
@@ -90,7 +136,7 @@
|
||||
{% else %}<li class="next"><a href="/blogpost/{{ FORW_ID }}/{{ FORW_HEADER_T }}?page-back={{ PAGE_BACK }}">Следующая запись <span aria-hidden="true">→</span></a></li>{% endif %}
|
||||
</ul>
|
||||
</nav>{% endif %}
|
||||
</div>
|
||||
</nav>
|
||||
</div>{# Листалка: КОНЕЦ #}
|
||||
{# --- Баннер: НАЧАЛО --- #}
|
||||
<div class="row"><div class="col-md-12 col-xs-12"><hr class="dotted-black" />{% include "ad/bannet-wide.html" %}</div></div>
|
||||
@@ -103,4 +149,3 @@
|
||||
{# ОТЛАДКА, ГАСИМ ВЕРХНЕЕ МЕНЮ #}
|
||||
{% endblock %}
|
||||
{% endcomment %}
|
||||
|
||||
|
||||
@@ -1,58 +1,81 @@
|
||||
{% extends "base.html" %}
|
||||
{% load static %}{% load filters %}
|
||||
|
||||
{% block Title %}Каталог изготовителей и поставщиков окон{% endblock %}
|
||||
{% block Title %}Каталог оконных компаний: производители и поставщики окон, рейтинг и цены{% endblock %}
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}Каталог изготовителей окон, партнёры «Окнардия», рейтинг, {% for i in COMPANIES %}{{ i.sMerchantName }}, {% endfor %} средняя цена окна{% endblock %}
|
||||
{% block Description %}Актуальный каталог оконных компаний России. Сравните производителей и поставщиков пластиковых окон по рейтингу, ассортименту, средней цене и дате последнего обновления.{% endblock %}
|
||||
|
||||
{% block Keywords %}Оконные компании, {% for i in COMPANIES %}{{ i.sMerchantName }}, {% endfor %} изготовители окон, производители окон, постащики окон, партнёры, каталог компаний, каталог оконных компаний, oknardia, окнардия {{ META_KEYWORDS|default:"" }} {% endblock %}
|
||||
{% block Keywords %}оконные компании, каталог компаний, производители окон, поставщики окон, рейтинг оконных компаний, сравнить цены на окна, oknardia, окнардия{% endblock %}
|
||||
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Author4Meta %}: Каталог «Окнардия»{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Author4Meta %}: Каталог изготовителей окон{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}: Каталог изготовителей окон{% endblock %}
|
||||
{% block CopyrightAuthor4Meta %}: Каталог «Окнардия»{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metatags --> #}
|
||||
<meta itemprop="author" content="Каталог «Окнардия»" />{% if IMG_FOR_BLOG %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/media/{{ IMG_FOR_BLOG }}" />{% else %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/static/img/MerDY3gpU0w.jpg" />{% endif %}
|
||||
<meta itemprop="datePublished" content="{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}" />
|
||||
<span itemprop="publisher" itemscope itemtype="http://schema.org/Organization"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<span itemprop="author" itemscope itemtype="http://schema.org/Person"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<meta itemprop="articleSection" content="Каталог производителей окон" />
|
||||
<meta itemprop="headline" content="Компании-партнёры «Окнардии», их рейтинг, число оконных наборов и вариантов расчёта цен для типовых проёмов, средняя цена окна..." />
|
||||
<meta name="news_keywords" content="{{ HEADER }}" />
|
||||
<link rel="canonical" href="https://oknardia.ru/catalog/company/" />
|
||||
<link rel="standout" href="https://oknardia.ru/catalog/company/" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/catalog/company/" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
<meta property="fb:app_id" content="258354027974262" />
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="https://oknardia.ru//catalog/company/" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/company/" />
|
||||
<meta property="og:type" content="article" />
|
||||
<meta property="og:title" content="Каталог производителей окон | oknardia.ru" />
|
||||
<meta property="og:description" content="Компании-партнеры «Окнардии», их средний рейтинг, число оконных наборов и вариантов расчета цен для типовых проёмов, средняя цена окна..." />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<meta property="og:title" content="Каталог оконных компаний: производители и поставщики окон, рейтинг и цены | oknardia.ru" />
|
||||
<meta property="og:description" content="Актуальный каталог оконных компаний России. Сравните производителей и поставщиков пластиковых окон по рейтингу, ассортименту, средней цене и дате последнего обновления." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="Каталог производителей окон | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Компании-партнеры «Окнардии», их средний рейтинг, число оконных наборов и вариантов расчета цен для типовых проёмов, средняя цена окна..." />
|
||||
<meta name="twitter:title" content="Каталог оконных компаний: производители и поставщики окон, рейтинг и цены | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Актуальный каталог оконных компаний России. Сравните производителей и поставщиков пластиковых окон по рейтингу, ассортименту, средней цене и дате последнего обновления." />
|
||||
<meta name="twitter:card" content="summary">
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:domain" content="oknardia.ru" />
|
||||
<meta property="twitter:url" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}">
|
||||
{# Удалить: <meta name="twitter:domain"> — устаревший тег #}
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<meta name="relap-image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg">
|
||||
{# <!-- END Дополнительные Metatags --> #}{% endblock %}
|
||||
|
||||
|
||||
{% block ADD_TO_HEAD %}{% comment %}
|
||||
JSON-LD для страницы-списка компаний: CollectionPage + ItemList с элементами Organization.
|
||||
Это понятнее для поисковиков, чем legacy microdata на метатегах.
|
||||
{% endcomment %}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "CollectionPage",
|
||||
"name": "Каталог оконных компаний: производители и поставщики окон",
|
||||
"description": "Актуальный каталог оконных компаний России с рейтингами, средней ценой и составом наборов.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/company/",
|
||||
"inLanguage": "ru-RU",
|
||||
"isPartOf": {
|
||||
"@type": "WebSite",
|
||||
"name": "Окнардия",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}"
|
||||
},
|
||||
"mainEntity": {
|
||||
"@type": "ItemList",
|
||||
"name": "Производители и поставщики окон",
|
||||
"numberOfItems": {{ COMPANIES|length }},
|
||||
"itemListElement": [
|
||||
{% for i in COMPANIES %}
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": {{ forloop.counter }},
|
||||
"item": {
|
||||
"@type": "Organization",
|
||||
"name": "{{ i.sMerchantName|escapejs }}",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/company/{{ i.id }}-{{ i.sMerchantMainURL }}",
|
||||
"logo": "{{ request.scheme }}://{{ request.get_host }}/media/{{ i.pMerchantLogo }}"
|
||||
}
|
||||
}{% if not forloop.last %},{% endif %}
|
||||
{% endfor %}
|
||||
]
|
||||
}
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
|
||||
{% block Main_Content %}
|
||||
<div class="container-fluid">
|
||||
{# <!--- Хлебные крошки: НАЧАЛО --> #}<div class="row">
|
||||
@@ -90,7 +113,3 @@
|
||||
{% include "report/report_log_user_visit.html" %}
|
||||
</div>
|
||||
</div>{% endblock %}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
@@ -5,30 +5,20 @@
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}«{{ COMPANY }}», описание компании «{{ COMPANY }}», оконные наборы от «{{ COMPANY }}» и их состав, характеристики «{{ COMPANY }}», рейтинг «{{ COMPANY }}», средние цены и отклонение цен «{{ COMPANY }}».{% endblock %}
|
||||
{% block Description %}Производитель окон «{{ COMPANY }}» в каталоге Окнардии: оконные наборы, их состав и характеристики, независимый рейтинг качества, средние цены на замену оконных конструкций в типовых домах.{% endblock %}
|
||||
|
||||
{% block Keywords %}{{ COMPANY }}, компания {{ COMPANY }}, окна {{ COMPANY }}, изготовитель окон {{ COMPANY }}, производитель окон {{ COMPANY }}, поставщик окон {{ COMPANY }}, партнёр, каталог компаний, каталог оконных компаний, oknardia, окнардия {{ META_KEYWORDS|default:"" }} {% endblock %}
|
||||
{% block Keywords %}{{ COMPANY }}, компания {{ COMPANY }}, окна {{ COMPANY }}, изготовитель окон {{ COMPANY }}, производитель окон {{ COMPANY }}, поставщик окон {{ COMPANY }}, партнёр, каталог компаний, каталог оконных компаний, oknardia, окнардия{% endblock %}
|
||||
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Author4Meta %}Каталог изготовителей окон{% endblock %}
|
||||
|
||||
{% block Author4Meta %}: Каталог изготовителей окон{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}: Каталог изготовителей окон{% endblock %}
|
||||
{% block CopyrightAuthor4Meta %}Каталог изготовителей окон{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metatags --> #}
|
||||
<meta itemprop="author" content="Каталог «Окнардия»" />{% if IMG_FOR_BLOG %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/media/{{ IMG_FOR_BLOG }}" />{% else %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/static/img/MerDY3gpU0w.jpg" />{% endif %}
|
||||
<meta itemprop="datePublished" content="{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}" />
|
||||
<span itemprop="publisher" itemscope itemtype="http://schema.org/Organization"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<span itemprop="author" itemscope itemtype="http://schema.org/Person"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<meta itemprop="articleSection" content="Каталог производителей окон" />
|
||||
<meta itemprop="headline" content="Изготовитель окон «{{ COMPANY }}», описание, производимые им оконные наборы и их состав, характеристики, рейтинг, средние цены и отклонение цен." />
|
||||
{# Microdata (itemprop) убрана — заменена на JSON-LD в блоке ADD_TO_HEAD ниже (чище, надёжнее) #}
|
||||
<meta name="news_keywords" content="{{ HEADER|striptags }}" />
|
||||
<link rel="canonical" href="https://oknardia.ru/catalog/company/{{ COMPANY_ID }}-{{ COMPANY_T }}" />
|
||||
<link rel="standout" href="https://oknardia.ru/catalog/company/{{ COMPANY_ID }}-{{ COMPANY_T }}" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/catalog/company/{{ COMPANY_ID }}-{{ COMPANY_T }}" />
|
||||
{# Удалить: <link rel="standout"> — тег Google News 2011 г., отменён в 2014, поисковики игнорируют #}
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
@@ -36,24 +26,60 @@
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="https://oknardia.ru//catalog/company/{{ COMPANY_ID }}-{{ COMPANY_T }}" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/company/{{ COMPANY_ID }}-{{ COMPANY_T }}" />
|
||||
<meta property="og:type" content="article" />
|
||||
<meta property="og:title" content="Окна «{{ COMPANY }}» | oknardia.ru" />
|
||||
<meta property="og:description" content="Окна «{{ COMPANY }}», описание окон «{{ COMPANY }}», производимые им оконные наборы и их состав, характеристики, рейтинг, средние цены и отклонение цен." />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<meta property="og:description" content="«{{ COMPANY }}» — оконные наборы, состав и характеристики, независимый рейтинг качества, средние цены на установку. Агрегатор Окнардия." />
|
||||
{# Нельзя вкладывать {{ }} внутрь аргумента фильтра |default — используем {% if %}{% else %} #}
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="Производителей окон «{{ COMPANY }}» | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Изготовитель окон «{{ COMPANY }}», описание, производимые им оконные наборы и их состав, характеристики, рейтинг, средние цены и отклонение цен." />
|
||||
<meta name="twitter:title" content="Производитель окон «{{ COMPANY }}» | oknardia.ru" />
|
||||
<meta name="twitter:description" content="«{{ COMPANY }}» в каталоге Окнардии: наборы, характеристики, рейтинг и цены на установку окон в типовых домах." />
|
||||
<meta name="twitter:card" content="summary">
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:domain" content="oknardia.ru" />
|
||||
<meta property="twitter:url" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}">
|
||||
{# Удалить: <meta name="twitter:domain"> — устарело с 2015, Twitter его не использует #}
|
||||
<meta property="twitter:url" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}">
|
||||
{# <!-- END Дополнительные Metatags --> #}{% endblock %}
|
||||
|
||||
{% block Top_JS5 %}
|
||||
<script src="https://api-maps.yandex.ru/2.1/?lang=ru_RU" type="text/javascript"></script>{% endblock %}
|
||||
|
||||
{% block Top_JS5 %}<script src="https://api-maps.yandex.ru/2.1/?lang=ru_RU" type="text/javascript"></script>{% endblock %}
|
||||
{% block ADD_TO_HEAD %}{% comment %}
|
||||
JSON-LD разметка Schema.org для страницы производителя окон.
|
||||
Тип LocalBusiness описывает компанию-поставщика окон: название, контакты, адрес, геокоординаты,
|
||||
логотип и ссылку на официальный сайт производителя.
|
||||
Данные берутся из первого набора в SETS (все наборы принадлежат одному офису/бренду),
|
||||
поэтому достаточно SETS.0 для контактной информации.
|
||||
Документация: https://schema.org/LocalBusiness #}{% endcomment %}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "LocalBusiness",
|
||||
"name": "{{ COMPANY|escapejs }}",
|
||||
"description": "Производитель окон «{{ COMPANY|escapejs }}»: оконные наборы, характеристики профилей и стеклопакетов, цены на установку в типовых домах.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/company/{{ COMPANY_ID }}-{{ COMPANY_T }}",
|
||||
"image": "{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}",
|
||||
"logo": {
|
||||
"@type": "ImageObject",
|
||||
"url": "{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}"
|
||||
}{% if SETS %},
|
||||
"telephone": "{{ SETS.0.sOfficePhones|striptags|escapejs }}",
|
||||
"address": {
|
||||
"@type": "PostalAddress",
|
||||
"streetAddress": "{{ SETS.0.sOfficeAddress|escapejs }}",
|
||||
"addressCountry": "RU"
|
||||
},
|
||||
"geo": {
|
||||
"@type": "GeoCoordinates",
|
||||
"latitude": {{ SETS.0.fOfficeGeoCode_Latitude|stringformat:".7f" }},
|
||||
"longitude": {{ SETS.0.fOfficeGeoCode_Longitude|stringformat:".7f" }}
|
||||
},
|
||||
"sameAs": "{{ SETS.0.sMerchantMainURL.URL|escapejs }}"{% endif %}
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
|
||||
{% block Main_Content %}
|
||||
@@ -112,7 +138,14 @@
|
||||
{# ПРАВАЯ КОЛОНКА: НАЧАЛО #}<div class="col-md-6 col-xs-12">
|
||||
<table class="head3">
|
||||
<tr>
|
||||
<td><h3>Оконный набор: «{{ i.sSetName|safe }}»</h3></td>
|
||||
<td>
|
||||
<h3>
|
||||
Оконный набор: «{{ i.sSetName|safe }}»
|
||||
<small style="font-size:xx-small;font-weight:normal;">
|
||||
<a href="/catalog/sets/#kit-card-{{ i.idSetKit }}" title="Открыть карточку этого набора в каталоге оконных наборов">в каталоге наборов</a>
|
||||
</small>
|
||||
</h3>
|
||||
</td>
|
||||
<td align="right"><nobr class="badge badge4price" title="Рейтинг «Окнардии» для оконного набора «{{ i.sSetName }}»{% if i.fSetRating.RATING > -0.01 %} — {{ i.fSetRating.RATING|stringformat:".2f" }} баллов{% endif %}">{% for Star in i.fSetRating.STARS %}{% if Star == 0 %}<b class="glyphicon glyphicon-star-empty"></b>{% else %}<b class="glyphicon glyphicon-star"></b>{% endif %}{% endfor %} {% if i.fSetRating.RATING > -0.01 %} {{ i.fSetRating.RATING|stringformat:".2f" }}{% endif %}</nobr></td>
|
||||
</tr>
|
||||
</table>
|
||||
|
||||
@@ -5,18 +5,70 @@
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}Каталог оконных профилей{% endblock %}
|
||||
{% block Description %}Подберите оконный профиль под свои требования: в каталоге «Окнардии» собраны производители, марки и ключевые характеристики.{% endblock %}
|
||||
|
||||
{% block Keywords %}каталог оконных профилей, каталог производителей оконных профилей, каталог профилей, оконные профили, oknardia, окнардия {{ META_KEYWORDS|default:"" }} {% endblock %}
|
||||
|
||||
{% block Date4Meta %}{{ CATALOG_LAST_UPDATE|date:"c" }}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{{ CATALOG_LAST_UPDATE|date:"c" }}{% endblock %}
|
||||
{% block Keywords %}оконные профили, каталог профилей, сравнение профилей, производители оконных профилей, характеристики оконных профилей, oknardia {{ META_KEYWORDS|default:"" }} {% endblock %}
|
||||
|
||||
{% block Author4Meta %}: Каталог{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}: Каталог{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metatags --> #}
|
||||
<meta name="news_keywords" content="каталог оконных профилей, производители профилей, марки профилей" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/catalog/profile/" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/profile/" />
|
||||
<meta property="og:type" content="website" />
|
||||
<meta property="og:title" content="Каталог оконных профилей | oknardia.ru" />
|
||||
<meta property="og:description" content="Производители и модели оконных профилей с характеристиками и рейтингом в каталоге Окнардии." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<meta name="twitter:title" content="Каталог оконных профилей | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Производители и модели оконных профилей с характеристиками и рейтингом в каталоге Окнардии." />
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta property="twitter:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/profile/" />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
{# <!-- END Дополнительные Metатags --> #}{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "CollectionPage",
|
||||
"name": "Каталог оконных профилей",
|
||||
"description": "Список производителей и моделей оконных профилей с переходом на карточки и характеристики.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/"
|
||||
}
|
||||
</script>
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Каталог",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 3,
|
||||
"name": "Оконные профили",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>{% endblock %}
|
||||
|
||||
{% block Main_Content %}
|
||||
<div class="container-fluid">
|
||||
{# <!--- Хлебные крошки: НАЧАЛО --> #}<div class="row">
|
||||
@@ -27,7 +79,7 @@
|
||||
<li>Оконные профили</li>
|
||||
</ol>
|
||||
<h1>Каталог оконных профилей</h1>
|
||||
<p>Узнать о производителях, познакомиться с детальными характеристики и описаниями оконных профилей можно кликнув по ссылкам. Сейчас в каталоге «Окнардии» представлено {{ CATALOG_MANUFACT_NUM_W }} профилей ({{ CATALOG_PROFILE_NUM }} в базе). Последнее обновление {{ CATALOG_LAST_UPDATE_W }}.</p>
|
||||
<p>Узнать о производителях, познакомиться с детальными характеристики и описаниями оконных профилей можно кликнув по ссылкам. Сейчас в каталоге «Окнардии» представлено {{ CATALOG_MANUFACT_NUM_W }} профилей ({{ CATALOG_PROFILE_NUM }} в базе).</p>
|
||||
</div>
|
||||
</div>{# <!--- Хлебные крошки: КОНЕЦ ---> #}
|
||||
<div class="row">
|
||||
@@ -61,4 +113,3 @@
|
||||
{# ОТЛАДКА, ГАСИМ ВЕРХНЕЕ МЕНЮ #}
|
||||
{% endblock %}
|
||||
{% endcomment %}
|
||||
|
||||
|
||||
@@ -9,48 +9,35 @@
|
||||
|
||||
{% block Keywords %}{{ CATALOG_MANUFACT }}, оконные профили {{ CATALOG_MANUFACT }}, производитель {{ CATALOG_MANUFACT }}, {% for i in PROFILES %}{{ i.PROFILE_NAME }}, {% endfor %}каталог оконных профилей, каталог производителей оконных профилей, каталог профилей, оконные профили, oknardia, окнардия {{ META_KEYWORDS|default:"" }} {% endblock %}
|
||||
|
||||
{% block Date4Meta %}{{ PUB_DAT|date:"c" }}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{{ PUB_DAT|date:"c" }}{% endblock %}
|
||||
|
||||
{% block Author4Meta %}: Каталог{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}: Каталог{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}
|
||||
<!-- Дополнительные Metatags -->
|
||||
<meta itemprop="author" content="Каталог «Окнардия»" />{% if IMG_FOR_BLOG %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/media/{{ IMG_FOR_BLOG }}" />{% else %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/static/img/MerDY3gpU0w.jpg" />{% endif %}
|
||||
<meta itemprop="datePublished" content="{{ PUB_DAT|date:"c" }}" />
|
||||
<span itemprop="publisher" itemscope itemtype="http://schema.org/Organization"><meta itemprop="name" content="Каталог «Окнардия»: оконные профили" /></span>
|
||||
<span itemprop="author" itemscope itemtype="http://schema.org/Person"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<meta itemprop="articleSection" content="Каталог «Окнардия»: оконные профили {{ CATALOG_MANUFACT }}" />
|
||||
<meta itemprop="headline" content="{{ TIZER|striptags|truncatewords_html:25 }}" />
|
||||
<meta name="news_keywords" content="{{ HEADER|striptags }}" />
|
||||
<link rel="canonical" href="https://oknardia.ru//catalog/profile/{{ CATALOG_URL }}" />
|
||||
<link rel="standout" href="https://oknardia.ru//catalog/profile/{{ CATALOG_URL }}" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metатеги --> #}
|
||||
{# Удалить: itemprop microdata и rel=standout в head (устаревшее), используем JSON-LD ниже #}
|
||||
{# Удалить: twitter:domain (устаревшее поле) #}
|
||||
<meta name="news_keywords" content="{{ HEADER|striptags|default:CATALOG_MANUFACT }}" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL }}/" />
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
<meta property="fb:app_id" content="258354027974262" />
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="https://oknardia.ru//catalog/profile/{{ CATALOG_URL }}" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL }}/" />
|
||||
<meta property="og:type" content="article" />
|
||||
<meta property="og:title" content="{{ HEADER|striptags }} | oknardia.ru" />
|
||||
<meta property="og:description" content="{{ TIZER|striptags|truncatewords_html:25 }}" />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="{{ HEADER|striptags }} | oknardia.ru" />
|
||||
<meta name="twitter:description" content="{{ TIZER|striptags|truncatewords_html:25 }}" />
|
||||
<meta name="twitter:card" content="summary">
|
||||
<meta property="og:title" content="{{ HEADER|striptags|default:CATALOG_MANUFACT }} | oknardia.ru" />
|
||||
<meta property="og:description" content="{{ TIZER|striptags|truncatewords_html:25|default:'Оконные профили производителя в каталоге Окнардии.' }}" />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
<meta name="twitter:title" content="{{ HEADER|striptags|default:CATALOG_MANUFACT }} | oknardia.ru" />
|
||||
<meta name="twitter:description" content="{{ TIZER|striptags|truncatewords_html:25|default:'Оконные профили производителя в каталоге Окнардии.' }}" />
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:domain" content="oknardia.ru" />
|
||||
<meta property="twitter:url" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}">{% endblock %}
|
||||
<meta property="twitter:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL }}/" />
|
||||
<meta name="twitter:image" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}">
|
||||
{# <!-- END Дополнительные Metatags --> #}{% endblock %}
|
||||
|
||||
{% block Top_JS4 %}
|
||||
<script type="text/javascript" src="//www.gstatic.com/charts/loader.js"></script>
|
||||
@@ -79,6 +66,73 @@
|
||||
}
|
||||
</script>{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "CollectionPage",
|
||||
"name": "Оконные профили производства {{ CATALOG_MANUFACT|escapejs }}",
|
||||
"description": "Страница производителя {{ CATALOG_MANUFACT|escapejs }}: список профилей, рейтинг и описание производителя в каталоге Окнардии.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL }}/",
|
||||
"about": {
|
||||
"@type": "Organization",
|
||||
"name": "{{ CATALOG_MANUFACT|escapejs }}"
|
||||
},
|
||||
"subjectOf": {
|
||||
"@type": "CreativeWork",
|
||||
"name": "Рейтинг Окнардии и как он устроен",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/blogpost/13/rejting-oknardii-i-kak-on-ustroen-2"
|
||||
}
|
||||
}
|
||||
</script>
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Каталог",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 3,
|
||||
"name": "Оконные профили",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 4,
|
||||
"name": "{{ CATALOG_MANUFACT|escapejs }}",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL }}/"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "ItemList",
|
||||
"name": "Профили {{ CATALOG_MANUFACT|escapejs }}",
|
||||
"itemListElement": [{% for i in PROFILES %}
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": {{ forloop.counter }},
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ i.PROFILE_ID }}-{{ CATALOG_MAN2URL }}/{{ i.PROFILE_ID }}-{{ i.PROFILE_URL }}/",
|
||||
"name": "{{ i.PROFILE_NAME|escapejs }}"
|
||||
}{% if not forloop.last %},{% endif %}{% endfor %}
|
||||
]
|
||||
}
|
||||
</script>{% endblock %}
|
||||
|
||||
{% block Main_Content %}
|
||||
<div class="container-fluid">
|
||||
{# <!--- Хлебные крошки: НАЧАЛО --> #}<div class="row">
|
||||
@@ -105,7 +159,9 @@
|
||||
</tr>{% endfor %}
|
||||
</tbody>
|
||||
</table>
|
||||
<p class="small-note">Сравнить компонеты рейтинга профилей можно в разделе <a href="/stat/rating/profiles_rank/">Ретинги</a>.</p>{% if not OFFERS_BY_MAUFACTURE == 0 %}
|
||||
<p class="small-note">Сравнить компонеты рейтинга профилей можно в разделе <a href="/stat/rating/profiles_rank/">Ретинги</a>.<br />
|
||||
Методика расчёта: <a href="/blogpost/13/rejting-oknardii-i-kak-on-ustroen-2" target="_blank"
|
||||
rel="nofollow">«Рейтинг Окнардии и как он устроен»</a>.</p>{% if not OFFERS_BY_MAUFACTURE == 0 %}
|
||||
<h4>Доля предложений окон на основе профилей {{ CATALOG_MANUFACT }} в базе «Окнардия»</h4>
|
||||
<div id="donutchart"></div>
|
||||
<h5>Партнёры «Окнардия» использующие профили производства {{ CATALOG_MANUFACT }} в своих предложениях:</h5>
|
||||
|
||||
@@ -5,31 +5,17 @@
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}Характеристики оконного профиля {{ CATALOG_MODEL.sProfileName }} производства {{ CATALOG_MODEL.sProfileManufacturer }}{% endblock %}
|
||||
{% block Description %}Оконный профиль {{ CATALOG_MODEL.sProfileName }} от {{ CATALOG_MODEL.sProfileManufacturer }}: характеристики, рейтинг, комплектация и применение в оконных предложениях партнёров Окнардии.{% endblock %}
|
||||
|
||||
{% block Keywords %}оконный профиль {{ CATALOG_MODEL.sProfileName }}, характеристики профиля {{ CATALOG_MODEL.sProfileName }}, описание профиля {{ CATALOG_MODEL.sProfileName }}, производитель оконный профилей {{ CATALOG_MODEL.sProfileManufacturer }}, каталог оконных профилей, каталог производителей оконных профилей, каталог профилей, оконные профили, oknardia, окнардия {{ META_KEYWORDS|default:"" }} {% endblock %}
|
||||
|
||||
{% block Date4Meta %}{{ PUB_DAT|date:"c" }}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{{ PUB_DAT|date:"c" }}{% endblock %}
|
||||
|
||||
{% block Author4Meta %}: Каталог{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}: Каталог{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}
|
||||
<!-- Дополнительные Metatags -->
|
||||
<meta itemprop="author" content="Каталог «Окнардия»" />{% if IMG_FOR_BLOG %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/media/{{ IMG_FOR_BLOG }}" />{% else %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/static/img/MerDY3gpU0w.jpg" />{% endif %}
|
||||
<meta itemprop="datePublished" content="{{ PUB_DAT|date:"c" }}" />
|
||||
<span itemprop="publisher" itemscope itemtype="http://schema.org/Organization"><meta itemprop="name" content="Каталог «Окнардия»: оконные профили" /></span>
|
||||
<span itemprop="author" itemscope itemtype="http://schema.org/Person"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<meta itemprop="articleSection" content="Каталог «Окнардия»: оконные профили {{ CATALOG_MODEL.sProfileName }}" />
|
||||
<meta itemprop="headline" content="Описание и характеристики оконных профилей {{ CATALOG_MODEL.sProfileName }} производства {{ CATALOG_MODEL.sProfileManufacturer }}" />
|
||||
<meta name="news_keywords" content="{{ CATALOG_MODEL.sProfileName }}, характеристики {{ CATALOG_MODEL.sProfileName }}, описание {{ CATALOG_MODEL.sProfileName }}, оконные профили {{ CATALOG_MODEL.sProfileName }}, {{ CATALOG_MODEL.sProfileManufacturer }}, производитель {{ CATALOG_MODEL.sProfileManufacturer }}, каталог оконных профилей, каталог производителей оконных профилей, каталог профилей, оконные профили, oknardia, окнардия" />
|
||||
<link rel="canonical" href="https://oknardia.ru//catalog/profile/{{ CATALOG_URL2 }}/" />
|
||||
<link rel="standout" href="https://oknardia.ru//catalog/profile/{{ CATALOG_URL2 }}/" />
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metatags --> #}
|
||||
<meta name="news_keywords" content="{{ CATALOG_MODEL.sProfileName }}, {{ CATALOG_MODEL.sProfileManufacturer }}, каталог оконных профилей, оконные профили" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL2 }}" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
@@ -37,20 +23,100 @@
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="https://oknardia.ru//catalog/profile/{{ CATALOG_URL2 }}" />
|
||||
<meta property="og:type" content="article" />
|
||||
<meta property="og:title" content="Оконные профили {{ CATALOG_MODEL.sProfileName }} | oknardia.ru" />
|
||||
<meta property="og:description" content="Описание и характеристики оконных профилей {{ CATALOG_MODEL.sProfileName }} производства {{ CATALOG_MODEL.sProfileManufacturer }}" />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL2 }}" />
|
||||
<meta property="og:type" content="product" />
|
||||
<meta property="og:title" content="Оконный профиль {{ CATALOG_MODEL.sProfileName }} | oknardia.ru" />
|
||||
<meta property="og:description" content="Оконный профиль {{ CATALOG_MODEL.sProfileName }} от {{ CATALOG_MODEL.sProfileManufacturer }}: характеристики, рейтинг и описание в каталоге Окнардии." />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="Оконные профили {{ CATALOG_MODEL.sProfileName }} | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Описание и характеристики оконных профилей {{ CATALOG_MODEL.sProfileName }} производства {{ CATALOG_MODEL.sProfileManufacturer }}" />
|
||||
<meta name="twitter:card" content="summary">
|
||||
<meta name="twitter:title" content="Оконный профиль {{ CATALOG_MODEL.sProfileName }} | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Характеристики, рейтинг и описание профиля {{ CATALOG_MODEL.sProfileName }} производства {{ CATALOG_MODEL.sProfileManufacturer }}." />
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:domain" content="oknardia.ru" />
|
||||
<meta property="twitter:url" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}">{% endblock %}
|
||||
<meta property="twitter:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL2 }}" />
|
||||
<meta name="twitter:image" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}">
|
||||
{# <!-- END Дополнительные Metатags --> #}{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}
|
||||
{# JSON-LD для карточки оконного профиля и хлебных крошек #}<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "Product",
|
||||
"name": "{{ CATALOG_MODEL.sProfileName|escapejs }}",
|
||||
"description": "Оконный профиль {{ CATALOG_MODEL.sProfileName|escapejs }} производства {{ CATALOG_MODEL.sProfileManufacturer|escapejs }}. Характеристики, рейтинг и применение в наборах партнёров Окнардии.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL2 }}",
|
||||
"image": "{% if IMG_FOR_BLOG %}{{ request.scheme }}://{{ request.get_host }}/media/{{ IMG_FOR_BLOG }}{% else %}{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg{% endif %}",
|
||||
"brand": {
|
||||
"@type": "Brand",
|
||||
"name": "{{ CATALOG_MODEL.sProfileManufacturer|escapejs }}"
|
||||
},
|
||||
"manufacturer": {
|
||||
"@type": "Organization",
|
||||
"name": "{{ CATALOG_MODEL.sProfileManufacturer|escapejs }}"
|
||||
},
|
||||
"category": "Оконные профили",
|
||||
"subjectOf": {
|
||||
"@type": "CreativeWork",
|
||||
"name": "Рейтинг Окнардии и как он устроен",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/blogpost/13/rejting-oknardii-i-kak-on-ustroen-2"
|
||||
},
|
||||
"additionalProperty": [
|
||||
{
|
||||
"@type": "PropertyValue",
|
||||
"name": "Метод ранжирования",
|
||||
"value": "Mann-Whitney (Манна-Уитни)"
|
||||
}
|
||||
]{% if CATALOG_MODEL.fProfileRating > -0.1 %},
|
||||
"aggregateRating": {
|
||||
"@type": "AggregateRating",
|
||||
"ratingValue": "{{ CATALOG_MODEL.fProfileRating|stringformat:'.2f' }}",
|
||||
"bestRating": "5",
|
||||
"worstRating": "0"{% if PROFILE_RATING_SAMPLE_SIZE > 0 %},
|
||||
"ratingCount": "{{ PROFILE_RATING_SAMPLE_SIZE }}"{% endif %}
|
||||
}{% endif %}
|
||||
}
|
||||
</script>
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Каталог",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 3,
|
||||
"name": "Оконные профили",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 4,
|
||||
"name": "{{ CATALOG_MODEL.sProfileManufacturer|escapejs }}",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL }}"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 5,
|
||||
"name": "Профиль {{ CATALOG_MODEL.sProfileName|escapejs }}",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ CATALOG_URL2 }}"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
|
||||
{% block Main_Content %}<div class="container-fluid">
|
||||
@@ -125,7 +191,12 @@
|
||||
<td colspan="3" {% if not CATALOG_MODEL.sProfileColor == "" %} title="Цвет оконного профиля: {{ CATALOG_MODEL.sProfileColor|capfirst }}"{% endif %}>{% if CATALOG_MODEL.sProfileColor == "" %}—{% else %}<small>{{ CATALOG_MODEL.sProfileColor|capfirst }}{% endif %}</small></td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>{% if LIST_OTHER|length > 1 %}
|
||||
</table>{% if CATALOG_MODEL.fProfileRating > -0.1 %}
|
||||
<p class="small-note">Рейтинг модели рассчитан алгоритмом «Окнардии» по статистическому ранжированию
|
||||
характеристик (метод Манна-Уитни){% if PROFILE_RATING_SAMPLE_SIZE > 0 %} на выборке из
|
||||
{{ PROFILE_RATING_SAMPLE_SIZE }} моделей профилей{% endif %}. Методика расчёта:
|
||||
<a href="/blogpost/13/rejting-oknardii-i-kak-on-ustroen-2" target="_blank"
|
||||
rel="nofollow">«Рейтинг Окнардии и как он устроен»</a>.</p>{% endif %}{% if LIST_OTHER|length > 1 %}
|
||||
<h4>Прочие характеристики профиля:</h4>
|
||||
<ul>{% for LI_BULL in LIST_OTHER %}
|
||||
<li>{{ LI_BULL|safe }}</li>{% endfor %}
|
||||
|
||||
@@ -5,30 +5,18 @@
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}Каталог «Окнардия»{% endblock %}
|
||||
{% block Description %}Каталог «Окнардия»: оконные и дверные профили, стеклопакеты, фурнитура, типовые серии домов, стандартные проёмы и партнёры-поставщики окон.{% endblock %}
|
||||
|
||||
{% block Keywords %}Каталог, каталог оконных профилей, каталог стеклопакетов, каталог фурнитуры, каталог серий домов, каталог оконных проёмов, oknardia, окнардия {{ META_KEYWORDS|default:"" }} {% endblock %}
|
||||
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Author4Meta %}: Каталог «Окнардия»{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}: Каталог «Окнардия»{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metatags --> #}
|
||||
<meta itemprop="author" content="Каталог «Окнардия»" />{% if IMG_FOR_BLOG %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/media/{{ IMG_FOR_BLOG }}" />{% else %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/static/img/MerDY3gpU0w.jpg" />{% endif %}
|
||||
<meta itemprop="datePublished" content="{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}" />
|
||||
<span itemprop="publisher" itemscope itemtype="http://schema.org/Organization"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<span itemprop="author" itemscope itemtype="http://schema.org/Person"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<meta itemprop="articleSection" content="Каталог «Окнардия»" />
|
||||
<meta itemprop="headline" content="Главная страница каталога «Окнардия»: оконные и дверные профили, стеклопакеты, фурнитуов, типовые серии домов, стандартные проемы, партнёры..." />
|
||||
<meta name="news_keywords" content="{{ HEADER }}" />
|
||||
<link rel="canonical" href="https://oknardia.ru//catalog/" />
|
||||
<link rel="standout" href="https://oknardia.ru//catalog/" />
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metatags --> #}
|
||||
{# Удалить: itemprop microdata, rel=standout, twitter:domain — устаревшие теги #}
|
||||
<meta name="news_keywords" content="каталог окон, каталог оконных профилей, серии домов, стандартные проёмы" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/catalog/" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
@@ -36,21 +24,81 @@
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="https://oknardia.ru//catalog/" />
|
||||
<meta property="og:type" content="article" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/" />
|
||||
<meta property="og:type" content="website" />
|
||||
<meta property="og:title" content="Каталог «Окнардия» | oknardia.ru" />
|
||||
<meta property="og:description" content="Главная страница каталога «Окнардия»: оконные и дверные профили, стеклопакеты, фурнитуов, типовые серии домов, стандартные проемы, партнёры..." />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<meta property="og:description" content="Оконные и дверные профили, стеклопакеты, типовые серии домов, стандартные проёмы и партнёры-поставщики окон." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="{{ HEADER }}"/>
|
||||
<meta name="twitter:description" content="Главная страница каталога «Окнардия»: оконные и дверные профили, стеклопакеты, фурнитуов, типовые серии домов, стандартные проемы, партнёры..." />
|
||||
<meta name="twitter:card" content="summary">
|
||||
<meta name="twitter:title" content="Каталог «Окнардия» | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Оконные и дверные профили, стеклопакеты, типовые серии домов, стандартные проёмы и партнёры-поставщики окон." />
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:domain" content="oknardia.ru" />
|
||||
<meta property="twitter:url" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
{# <!-- END Дополнительные Metatags --> #}{% endblock %}
|
||||
<meta property="twitter:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/" />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<meta name="relap-image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
{# <!-- END Дополнительные Metатags --> #}{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}
|
||||
{# JSON-LD: корневая страница каталога — CollectionPage + BreadcrumbList + структура разделов #}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "CollectionPage",
|
||||
"name": "Каталог «Окнардия»",
|
||||
"description": "Оконные и дверные профили, стеклопакеты, типовые серии домов, стандартные проёмы и партнёры-поставщики окон.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/",
|
||||
"hasPart": [
|
||||
{
|
||||
"@type": "CollectionPage",
|
||||
"name": "Оконные и дверные профили",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/"
|
||||
},
|
||||
{
|
||||
"@type": "CollectionPage",
|
||||
"name": "Каталог серий домов",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/seria/"
|
||||
},
|
||||
{
|
||||
"@type": "CollectionPage",
|
||||
"name": "Стандартные оконные проёмы и балконные блоки",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/standard_opening/"
|
||||
},
|
||||
{
|
||||
"@type": "CollectionPage",
|
||||
"name": "Производители и поставщики окон",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/company/"
|
||||
},
|
||||
{
|
||||
"@type": "CollectionPage",
|
||||
"name": "Оконные наборы: характеристики, комплектации и сравнение",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/sets/"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Каталог",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Main_Content %}
|
||||
<div class="container-fluid">
|
||||
@@ -66,7 +114,7 @@
|
||||
<dIv class="row -catalog2-">
|
||||
{# ПЕРВЫЙ РАЗДЕЛ С РЕКЛАМОЙ СБОКУ #}<div class="col-md-9 col-xs-8">
|
||||
{# И ЕЩЁ ОДИН РАЗДЕЛ #}<h2 class="header"><a href="/catalog/profile">Оконные и дверные профили</a></h2>
|
||||
<p class="col-md-offset-1 col-xs-offset-1">Каталог систем оконных и дверных профилей и описание <nobr>компаний-производителей</nobr>. Каталог содержит детальные характеристики профилей: сопротивление теплопередаче <nobr><i>Ro</i> (<i>м²×°C/Вт</i>)</nobr>, коэффициент звукоизоляции (<i>дБ</i>), число камер рамы и створки, тип и армирования, монтажная ширина и другие. Пластиковые (ПВХ), деревянные, комбинированные и другие системы профилей.</p>
|
||||
<p class="col-md-offset-1 col-xs-offset-1">Каталог систем оконных и дверных профилей и описание <nobr>компаний-производителей</nobr>. Каталог содержит детальные характеристики профилей: сопротивление теплопередаче <nobr><i>Ro</i> (<i>м²×°C/Вт</i>)</nobr>, коэффициент звукоизоляции (<i>дБ</i>), число камер рамы и створки, тип и армирование, монтажная ширина и другие. Пластиковые (ПВХ), деревянные, комбинированные и другие системы профилей.</p>
|
||||
{# И ЕЩЁ ОДИН РАЗДЕЛ #}<h2 class="header"><a href="/catalog/seria">Каталог серий домов</a></h2>
|
||||
<p class="col-md-offset-1 col-xs-offset-1">Типовые проекты жилого строительства, вхождение стандартных оконных проёмов и балконных блоков в планировки типовых квартир серии, графики ввода в эксплуатацию зданий серии, география строительства, износ жилого фонда…</p>
|
||||
</div>
|
||||
@@ -76,8 +124,10 @@
|
||||
<p class="col-md-offset-1 col-xs-offset-1">Размеры и рекомендованные схемы открывания стандартных проёмов и балконных блоков базы «Окнардия», коммерческие предложения партнёров агрегатора, условия поставки, комплектация, сопутствующие услуги и возможные скидки.</p>
|
||||
</div>
|
||||
{# И ОПЯТЬ РАЗДЕЛ С РЕКЛАМОЙ СБОКУ #}<div class="col-md-9 col-xs-8">
|
||||
{# И ЕЩЁ ОДИН РАЗДЕЛ #}<h2 class="header"><a href="/catalog/company">Производители и поставщики окон</a> <small style="font-size:xx-small;">(в разработке)</small></h2>
|
||||
<p class="col-md-offset-1 col-xs-offset-1">Компании-партнеры «Окнардии», контатная информация, условия и скидки, конфигурации и рейтинги их оконных предложений.</p>
|
||||
{# И ЕЩЁ ОДИН РАЗДЕЛ #}<h2 class="header"><a href="/catalog/company">Производители и поставщики окон</a></h2>
|
||||
<p class="col-md-offset-1 col-xs-offset-1">Компании-партнеры «Окнардии», контактная информация, условия и скидки, конфигурации и рейтинги их оконных предложений.</p>
|
||||
{# И ЕЩЁ ОДИН РАЗДЕЛ #}<h2 class="header"><a href="/catalog/sets/">Оконные наборы: характеристики, комплектации и сравнение</a></h2>
|
||||
<p class="col-md-offset-1 col-xs-offset-1">Готовые комплектации окон разных поставщиков: профиль, стеклопакет, фурнитура и монтаж в одном предложении. Сравнивайте предложения компаний устанавливающих окна по характеристикам (теплопередача, звукоизоляция, состав услуг, рейтинг «Окнардии» и многое другое).</p>
|
||||
{# ВТОРОЙ РАЗДЕЛ #}
|
||||
{# И ЕЩЁ ОДИН РАЗДЕЛ #}<h2 class="header"><a href="javascript://" class="not-ready">Каталог стеклопакетов</a> <small style="font-size:xx-small;">(в разработке)</small></h2>
|
||||
<p class="col-md-offset-1 col-xs-offset-1">Стеклопакеты и описание <nobr>компаний-производителей</nobr> стекла. Каталог содержит детальные характеристики: схемы стеклопакетов, наличие напыления k- и <nobr>i-микропленок</nobr>, тип <nobr>газа-заполнителя</nobr>, сопротивление теплопередаче <nobr><i>Ro</i> (<i>м²×°C/Вт</i>)</nobr>, коэффициент звукоизоляции (<i>дБ</i>), число камер, тонирование…</p>
|
||||
@@ -98,5 +148,3 @@
|
||||
{% include "report/report_log_user_visit.html" %}
|
||||
</div>
|
||||
</div>{% endblock %}
|
||||
|
||||
|
||||
|
||||
@@ -5,30 +5,17 @@
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}Каталог серий зданий и типовое панельное строительство оконного агрегатора Окнардия{% endblock %}
|
||||
{% block Description %}Каталог типовых серий домов России: панельные и блочные серии, ссылки на подробные страницы серий, их планировки и стандартные оконные проёмы.{% endblock %}
|
||||
|
||||
{% block Keywords %}типовое строительство, панельные серии, серии домов, серии зданий, типовые дома, типовые здания, каталог серий типового строительства, oknardia, окнардия {{ META_KEYWORDS|default:"" }} {% endblock %}
|
||||
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Keywords %}типовое строительство, панельные серии, серии домов, серии зданий, типовые дома, типовые здания, каталог серий типового строительства, oknardia, окнардия{% endblock %}
|
||||
|
||||
{% block Author4Meta %}Серии домов : {% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}Cерии домов : {% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metatags --> #}
|
||||
<meta itemprop="author" content="Каталог «Окнардия»" />{% if IMG_FOR_BLOG %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/media/{{ IMG_FOR_BLOG }}" />{% else %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/static/img/MerDY3gpU0w.jpg" />{% endif %}
|
||||
<meta itemprop="datePublished" content="{{ PUB_DAT|date:"c" }}" />
|
||||
<span itemprop="publisher" itemscope itemtype="http://schema.org/Organization"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<span itemprop="author" itemscope itemtype="http://schema.org/Person"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<meta itemprop="articleSection" content="Каталог «Окнардия»" />
|
||||
<meta itemprop="headline" content="Серии типового строительства | oknardia.ru" />
|
||||
<meta name="news_keywords" content="{{ HEADER|striptags }}" />
|
||||
<link rel="canonical" href="https://oknardia.ru//catalog/seria/" />
|
||||
<link rel="standout" href="https://oknardia.ru//catalog/seria/" />
|
||||
{# Legacy microdata (itemprop/itemscope) удалена: используем JSON-LD в ADD_TO_HEAD #}
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/catalog/seria/" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
@@ -36,23 +23,64 @@
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="https://oknardia.ru//catalog/seria/" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/seria/" />
|
||||
<meta property="og:type" content="article" />
|
||||
<meta property="og:title" content="Каталог «Окнардия» | oknardia.ru" />
|
||||
<meta property="og:description" content="Серии типового строительства, типовые проекты жилого строительства, вхождение стандартных оконных проёмов и балконных блоков в планировки типовых квартир серии, графики ввода в эксплуатацию зданий серии, география строительства, износ жилого фонда..." />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<meta property="og:title" content="Каталог типовых серий домов | oknardia.ru" />
|
||||
<meta property="og:description" content="Серии типового строительства, планировки и ссылки на подробные страницы серий домов с данными по стандартным оконным проёмам." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="Каталог типовых серий домов | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Серии типового строительства, типовые проекты жилого строительства, вхождение стандартных оконных проёмов и балконных блоков в планировки типовых квартир серии, графики ввода в эксплуатацию зданий серии, география строительства, износ жилого фонда..." />
|
||||
<meta name="twitter:description" content="Каталог типовых серий домов России: список серий и переходы на подробные страницы с планировками и окнами." />
|
||||
<meta name="twitter:card" content="summary">
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:domain" content="oknardia.ru" />
|
||||
<meta property="twitter:url" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}">
|
||||
<meta property="twitter:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/seria/" />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<meta name="relap-image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg">
|
||||
{# <!-- END Дополнительные Metatags --> #}{% endblock %}
|
||||
|
||||
|
||||
{% block ADD_TO_HEAD %}{% comment %}
|
||||
JSON-LD для страницы списка типовых серий домов.
|
||||
CollectionPage + ItemList помогают поисковику трактовать страницу как каталог сущностей.
|
||||
{% endcomment %}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "CollectionPage",
|
||||
"name": "Каталог серий типового строительства",
|
||||
"description": "Список типовых серий домов России с переходом на подробные страницы серий.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/seria/",
|
||||
"inLanguage": "ru-RU",
|
||||
"isPartOf": {
|
||||
"@type": "WebSite",
|
||||
"name": "Окнардия",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}"
|
||||
},
|
||||
"mainEntity": {
|
||||
"@type": "ItemList",
|
||||
"name": "Типовые серии домов",
|
||||
"numberOfItems": {{ SERIAS|length }},
|
||||
"itemListElement": [
|
||||
{% for i in SERIAS %}
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": {{ forloop.counter }},
|
||||
"item": {
|
||||
"@type": "Thing",
|
||||
"name": "Серия {{ i.NAME|escapejs }}",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/seria/{{ i.NAME_T }}/all{{ i.ID }}",
|
||||
"image": "{{ request.scheme }}://{{ request.get_host }}/media/{{ i.URL }}"
|
||||
}
|
||||
}{% if not forloop.last %},{% endif %}
|
||||
{% endfor %}
|
||||
]
|
||||
}
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
|
||||
{% block Main_Content %}
|
||||
<div class="container-fluid">
|
||||
{# <!--- Хлебные крошки: НАЧАЛО --> #}<div class="row">
|
||||
|
||||
385
oknardia/templates/catalog/catalog_sets.html
Normal file
385
oknardia/templates/catalog/catalog_sets.html
Normal file
@@ -0,0 +1,385 @@
|
||||
{% extends "base.html" %}
|
||||
{% load static %}
|
||||
{% load filters %}
|
||||
|
||||
{% block Title %}Оконные наборы: характеристики, комплектации и сравнение — каталог «Окнардия»{% endblock %}
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}Каталог оконных наборов «Окнардия»: готовые комплектации для замены окон с профилем, стеклопакетом и монтажом в одном предложении. Подробные характеристики, рейтинг и сравнение от разных поставщиков.{% endblock %}
|
||||
|
||||
{% block Keywords %}оконные наборы, комплектации окон, сравнение окон, профиль и стеклопакет, монтаж окон, окнардия{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/catalog/sets/" />
|
||||
<meta property="og:type" content="website" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/sets/" />
|
||||
<meta property="og:title" content="Оконные наборы: характеристики, комплектации и сравнение | oknardia.ru" />
|
||||
<meta property="og:description" content="Каталог готовых комплектаций окон от партнёров «Окнардии»: профиль, стеклопакет, фурнитура и монтаж. Сравнивайте по рейтингу и характеристикам." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:title" content="Оконные наборы: каталог и сравнение | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Готовые комплектации окон — профиль, стеклопакет, фурнитура и монтаж от партнёров «Окнардии»." />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}
|
||||
{# JSON-LD: CollectionPage каталога наборов — BreadcrumbList + ItemList с кратким описанием каждого Product #}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@graph": [
|
||||
{
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{"@type": "ListItem", "position": 1, "name": "Главная", "item": "{{ request.scheme }}://{{ request.get_host }}/"},
|
||||
{"@type": "ListItem", "position": 2, "name": "Каталог", "item": "{{ request.scheme }}://{{ request.get_host }}/catalog/"},
|
||||
{"@type": "ListItem", "position": 3, "name": "Оконные наборы",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/sets/"}
|
||||
]
|
||||
},
|
||||
{
|
||||
"@type": "CollectionPage",
|
||||
"name": "Оконные наборы: характеристики, комплектации и сравнение",
|
||||
"description": "Каталог готовых комплектаций окон от партнёров «Окнардии»: профиль, стеклопакет, фурнитура и монтаж в одном предложении.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/sets/",
|
||||
"mainEntity": {
|
||||
"@type": "ItemList",
|
||||
"name": "Оконные наборы",
|
||||
"numberOfItems": {{ SET_LIST|length }},
|
||||
"itemListElement": [{% for item in SET_LIST %}
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": {{ forloop.counter }},
|
||||
"item": {
|
||||
"@type": "Product",
|
||||
"name": "{{ item.kit.sSetName|escapejs }}",
|
||||
{% if item.merchant_name %}"brand": {"@type": "Brand", "name": "{{ item.merchant_name|escapejs }}"},{% endif %}
|
||||
"description": "Профиль {{ item.profile.sProfileName|escapejs }} ({{ item.profile.sProfileManufacturer|escapejs }}), стеклопакет {{ item.glazing.sGlazingMark|default:item.glazing.sGlazingName|escapejs }}.{% if item.profile.fProfileHeatTransf > 0.1 %} Ro профиля: {{ item.profile.fProfileHeatTransf }} м²·°C/Вт.{% endif %}{% if item.glazing.fGlazingHeatTransfer > 0.1 %} Ro стеклопакета: {{ item.glazing.fGlazingHeatTransfer }} м²·°C/Вт.{% endif %}",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/compare_offers/{{ item.kit.id }}/",
|
||||
{% if item.kit.fSetRating > 0.1 %}
|
||||
"review": {
|
||||
"@type": "Review",
|
||||
"author": {
|
||||
"@type": "Organization",
|
||||
"name": "Окнардия",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}"
|
||||
},
|
||||
"reviewRating": {
|
||||
"@type": "Rating",
|
||||
"ratingValue": "{{ item.kit.fSetRating|stringformat:".2f" }}",
|
||||
"bestRating": "5",
|
||||
"worstRating": "0"
|
||||
},
|
||||
"reviewBody": "Алгоритмический рейтинг «Окнардии», рассчитанный по критерию Манна-Уитни на основе характеристик профиля и стеклопакета, а также дополнительных услу и скидок"
|
||||
},
|
||||
{% endif %}
|
||||
"additionalProperty": [
|
||||
{"@type": "PropertyValue", "name": "Профиль", "value": "{{ item.profile.sProfileName|escapejs }}"},
|
||||
{"@type": "PropertyValue", "name": "Стеклопакет", "value": "{{ item.glazing.sGlazingName|escapejs }}"},
|
||||
{% if item.profile.iProfileThickness > 5 %}{"@type": "PropertyValue", "name": "Монтажная ширина профиля", "unitCode": "MMT", "unitText": "мм", "value": {{ item.profile.iProfileThickness }}},{% endif %}
|
||||
{% if item.glazing.iGlazingCamerasN >= 1 %}{"@type": "PropertyValue", "name": "Камер в стеклопакете", "unitText": "шт.", "value": {{ item.glazing.iGlazingCamerasN }}},{% endif %}
|
||||
{"@type": "PropertyValue", "name": "Доставка", "value": "{{ item.kit.sSetDelivery|escapejs }}"},
|
||||
{"@type": "PropertyValue", "name": "Монтаж", "value": "{{ item.kit.sSetUninstallInstall|escapejs }}"},
|
||||
{"@type": "PropertyValue", "name": "Метод ранжирования", "value": "Mann-Whitney (Манна-Уитни)"},
|
||||
{"@type": "PropertyValue", "name": "Источник данных", "value": "oknardia.ru"}
|
||||
]
|
||||
}
|
||||
}{% if not forloop.last %},{% endif %}{% endfor %}
|
||||
]
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
|
||||
{# CSS для плавающей панели выбора сравнения #}
|
||||
<style>
|
||||
/* Карточка набора — небольшой отступ снизу */
|
||||
.kit-card { margin-bottom: 16px; }
|
||||
/* Таблица характеристик — минимальные отступы */
|
||||
.kit-specs th { font-weight: normal; color: #777; white-space: nowrap; }
|
||||
.kit-specs th, .kit-specs td { padding: 2px 6px !important; font-size: 12px; }
|
||||
</style>
|
||||
{% endblock %}
|
||||
|
||||
{% block Main_Content %}<!--- ------------------------------------------------------------------------------------------------------------------------- --->
|
||||
<div class="container-fluid">
|
||||
|
||||
{# Хлебные крошки #}
|
||||
<div class="row">
|
||||
<div class="col-md-11 col-xs-12">
|
||||
<ol class="breadcrumb">
|
||||
<li><a href="/">Главная</a></li>
|
||||
<li><a href="/catalog/">Каталог</a></li>
|
||||
<li>Оконные наборы</li>
|
||||
</ol>
|
||||
<h1>Оконные наборы: характеристики, комплектации и сравнение</h1>
|
||||
<p>Оконный набор — готовая комплектация для замены окон в вашем доме: профиль, стеклопакет,
|
||||
фурнитура и монтаж в одном предложении от компаний-партнёров «Окнардии».
|
||||
Отметьте несколько интересных наборов и сравните их детально по всем характеристикам.</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Список карточек #}
|
||||
{% for item in SET_LIST %}
|
||||
<div class="panel panel-default kit-card" id="kit-card-{{ item.kit.id }}">
|
||||
|
||||
{# ---- ШАПКА КАРТОЧКИ: название + рейтинг + логотип ---- #}
|
||||
<div class="panel-heading">
|
||||
<div class="row" style="display:flex;align-items:center;">
|
||||
|
||||
{# Название + звёздочки рейтинга #}
|
||||
<div class="col-xs-9 col-md-10">
|
||||
<h2 class="panel-title" style="font-size:1.15em;">{{ item.kit.sSetName }}</h2>
|
||||
<div style="margin-top:3px;"{% if item.kit.sSetDescription %} title="{{ item.kit.sSetDescription }}"{% endif %}>
|
||||
<nobr>{% for star in item.stars %}{% if star %}<b class="glyphicon glyphicon-star" style="color:#f0a500;"></b>{% else %}<i class="glyphicon glyphicon-star-empty" style="color:#ccc;"></i>{% endif %}{% endfor %}{% if item.kit.fSetRating > 0.1 %} <tt class="badge">{{ item.kit.fSetRating|stringformat:".2f" }}</tt>{% endif %}</nobr>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Логотип компании — кликабельный, ведёт на карточку компании в каталоге #}
|
||||
<div class="col-xs-3 col-md-2 text-right">
|
||||
{% if item.merchant_logo %}
|
||||
{% if item.merchant_id %}<a href="/catalog/company/{{ item.merchant_id }}-{{ item.merchant_slug }}/" title="{{ item.merchant_name }}">{% endif %}
|
||||
<img src="http://oknardia.ru/media/{{ item.merchant_logo }}"
|
||||
style="max-height:36px;max-width:110px;object-fit:contain;"
|
||||
alt="{{ item.merchant_name }}" />
|
||||
{% if item.merchant_id %}</a>{% endif %}
|
||||
{% endif %}
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</div>{# /panel-heading #}
|
||||
|
||||
{# ---- ТЕЛО КАРТОЧКИ: три колонки — условия | профиль | стеклопакет ---- #}
|
||||
<div class="panel-body">
|
||||
<div class="row">
|
||||
|
||||
{# == Колонка 1: компания и условия поставки == #}
|
||||
<div class="col-md-3 col-xs-12" style="border-right:1px solid #eee;margin-bottom:8px;">
|
||||
<h3 style="font-size:1em;font-weight:bold;margin-top:0;">Поставщик</h3>
|
||||
{% if item.merchant_id %}
|
||||
<p style="margin-bottom:6px;">
|
||||
<a href="/catalog/company/{{ item.merchant_id }}-{{ item.merchant_slug }}/">
|
||||
<strong>{{ item.merchant_name }}</strong>
|
||||
</a>
|
||||
</p>
|
||||
{% elif item.merchant_name %}
|
||||
<p style="margin-bottom:6px;"><strong>{{ item.merchant_name }}</strong></p>
|
||||
{% endif %}
|
||||
|
||||
<table class="table kit-specs" style="margin-bottom:4px;">
|
||||
{% if item.kit.sSetImplementAll %}
|
||||
<tr><th>Фурнитура:</th><td>{{ item.kit.sSetImplementAll|capfirst }}</td></tr>
|
||||
{% endif %}
|
||||
{% if item.kit.sSetImplementHandles %}
|
||||
<tr><th><sup>∟</sup> Ручки:</th><td>{{ item.kit.sSetImplementHandles|capfirst }}</td></tr>
|
||||
{% endif %}
|
||||
{% if item.kit.sSetImplementHinges %}
|
||||
<tr><th><sup>∟</sup> Петли:</th><td>{{ item.kit.sSetImplementHinges|capfirst }}</td></tr>
|
||||
{% endif %}
|
||||
{% if item.kit.sSetImplementLatch %}
|
||||
<tr><th><sup>∟</sup> Запоры:</th><td>{{ item.kit.sSetImplementLatch|capfirst }}</td></tr>
|
||||
{% endif %}
|
||||
{% if item.kit.sSetImplementLimiter %}
|
||||
<tr><th><sup>∟</sup> Огранич.:</th><td>{{ item.kit.sSetImplementLimiter|capfirst }}</td></tr>
|
||||
{% endif %}
|
||||
{% if item.kit.sSetImplementCatch %}
|
||||
<tr><th><sup>∟</sup> Фиксаторы:</th><td>{{ item.kit.sSetImplementCatch|capfirst }}</td></tr>
|
||||
{% endif %}
|
||||
{% if item.kit.sSetClimateControl|length > 3 %}
|
||||
<tr><th>Климат-конт.:</th><td style="color:green;">{{ item.kit.sSetClimateControl|capfirst }}</td></tr>
|
||||
{% endif %}
|
||||
<tr>
|
||||
<th>Подоконник:</th>
|
||||
<td {% if item.kit.sSetSill|capfirst == "Нет" or item.kit.sSetSill|length < 4 %}style="color:red;"{% endif %}>
|
||||
{% if item.kit.sSetSill %}{{ item.kit.sSetSill|capfirst }}{% else %}—{% endif %}
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Водоотлив:</th>
|
||||
<td {% if item.kit.sSetPanes|capfirst == "Нет" or item.kit.sSetPanes|length < 4 %}style="color:red;"{% endif %}>
|
||||
{% if item.kit.sSetPanes %}{{ item.kit.sSetPanes|capfirst }}{% else %}—{% endif %}
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Откос:</th>
|
||||
<td {% if item.kit.sSetSlope|capfirst == "Нет" or item.kit.sSetSlope|length < 4 %}style="color:red;"{% endif %}>
|
||||
{% if item.kit.sSetSlope %}{{ item.kit.sSetSlope|capfirst }}{% else %}—{% endif %}
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Доставка:</th>
|
||||
<td style="color:{% if item.kit.bSetDelivery %}green{% else %}red{% endif %};">
|
||||
{{ item.kit.sSetDelivery|capfirst }}
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Монтаж:</th>
|
||||
<td style="color:{% if item.kit.bSetUninstallInstall %}green{% else %}red{% endif %};">
|
||||
{{ item.kit.sSetUninstallInstall|capfirst }}
|
||||
</td>
|
||||
</tr>
|
||||
{% if item.kit.sSetOtherConditions %}
|
||||
<tr><th>Прочее:</th><td><small>{{ item.kit.sSetOtherConditions|capfirst }}</small></td></tr>
|
||||
{% endif %}
|
||||
</table>
|
||||
</div>{# /col компания #}
|
||||
|
||||
{# == Колонка 2: профиль == #}
|
||||
<div class="col-md-4 col-xs-12" style="border-right:1px solid #eee;margin-bottom:8px;">
|
||||
<h3 style="font-size:1em;font-weight:bold;margin-top:0;">
|
||||
Профиль:
|
||||
<a href="/catalog/profile/{{ item.profile.id }}-{{ item.profile_manufacturer_slug }}/{{ item.profile.id }}-{{ item.profile_slug }}/">{{ item.profile.sProfileName }}</a>
|
||||
<small style="font-weight:normal;color:#777;">— <a href="/catalog/profile/{{ item.profile.id }}-{{ item.profile_manufacturer_slug }}/">{{ item.profile.sProfileManufacturer }}</a></small>
|
||||
</h3>
|
||||
{% if item.profile.sProfileBriefDescription %}
|
||||
<p style="font-size:small;color:#666;margin-bottom:6px;">{{ item.profile.sProfileBriefDescription }}</p>
|
||||
{% endif %}
|
||||
<table class="table kit-specs">
|
||||
<tr><th>Производитель:</th><td>{{ item.profile.sProfileManufacturer }}</td></tr>
|
||||
{% if item.profile.iProfileCameras %}<tr><th>Камер рамы/створки:</th><td>{{ item.profile.iProfileCameras }} шт.</td></tr>{% endif %}
|
||||
{% if item.profile.iProfileThickness > 5 %}<tr><th>Монтажная ширина:</th><td>{{ item.profile.iProfileThickness }} мм</td></tr>{% endif %}
|
||||
{% if item.profile.iProfileGlazingThickness > 4 %}<tr><th>Макс. толщина СП:</th><td>{{ item.profile.iProfileGlazingThickness }} мм</td></tr>{% endif %}
|
||||
{% if item.profile.fProfileHeatTransf > 0.1 %}<tr><th>Теплопередача <i>Ro</i>:</th><td>{{ item.profile.fProfileHeatTransf }} м²·°C/Вт</td></tr>{% endif %}
|
||||
{% if item.profile.fProfileSoundproofing > 1 %}<tr><th>Звукоизоляция:</th><td>{{ item.profile.fProfileSoundproofing }} дБ</td></tr>{% endif %}
|
||||
{% if item.profile.fProfileSeals > 0 %}<tr><th>Контуры уплотнения:</th><td>{{ item.profile.fProfileSeals }} шт.</td></tr>{% endif %}
|
||||
{% if item.profile.iProfileHeight > 15 %}<tr><th>Высота в проёме:</th><td>{{ item.profile.iProfileHeight }} мм</td></tr>{% endif %}
|
||||
{% if item.profile.iProfileRabbet > 1 %}<tr><th>Фальц рамы:</th><td>{{ item.profile.iProfileRabbet }} мм</td></tr>{% endif %}
|
||||
{% if item.profile.sProfileColor %}<tr><th>Цвет:</th><td>{{ item.profile.sProfileColor|capfirst }}</td></tr>{% endif %}
|
||||
{% if item.profile.sProfileReinforcement %}<tr><th>Армирование:</th><td>{{ item.profile.sProfileReinforcement }}</td></tr>{% endif %}
|
||||
{% if item.profile.sProfileSealDescription %}<tr><th>Уплотнитель:</th><td>{{ item.profile.sProfileSealDescription|capfirst }}</td></tr>{% endif %}
|
||||
{% if item.profile.sProfileFillet %}<tr><th>Штапик:</th><td>{{ item.profile.sProfileFillet }}</td></tr>{% endif %}
|
||||
{% if item.profile.sProfileOther %}<tr><th>Прочие хар-ки:</th><td><small>{{ item.profile.sProfileOther }}</small></td></tr>{% endif %}
|
||||
</table>
|
||||
</div>{# /col профиль #}
|
||||
|
||||
{# == Колонка 3: стеклопакет == #}
|
||||
<div class="col-md-5 col-xs-12" style="margin-bottom:8px;">
|
||||
<h3 style="font-size:1em;font-weight:bold;margin-top:0;">
|
||||
Стеклопакет: {{ item.glazing.sGlazingName }}
|
||||
</h3>
|
||||
{% if item.glazing.sGlazingBriefDescription %}
|
||||
<p style="font-size:small;color:#666;margin-bottom:6px;">{{ item.glazing.sGlazingBriefDescription|capfirst }}</p>
|
||||
{% endif %}
|
||||
<table class="table kit-specs">
|
||||
{% if item.glazing.sGlazingMark and item.glazing.sGlazingMark != "—" %}<tr><th>Схема:</th><td>{{ item.glazing.sGlazingMark }}</td></tr>{% endif %}
|
||||
{% if item.glazing.sGlazingManufacturer and item.glazing.sGlazingManufacturer != "—//—" and item.glazing.sGlazingManufacturer != "—" %}<tr><th>Производитель:</th><td>{{ item.glazing.sGlazingManufacturer }}</td></tr>{% endif %}
|
||||
{% if item.glazing.iGlazingCamerasN >= 1 %}<tr><th>Камер:</th><td>{{ item.glazing.iGlazingCamerasN }} шт.</td></tr>{% endif %}
|
||||
{% if item.glazing.iGlazingThickness >= 3 %}<tr><th>Толщина:</th><td>{{ item.glazing.iGlazingThickness }} мм</td></tr>{% endif %}
|
||||
{% if item.glazing.fGlazingHeatTransfer > 0.1 %}<tr><th>Теплопередача <i>Ro</i>:</th><td>{{ item.glazing.fGlazingHeatTransfer }} м²·°C/Вт</td></tr>{% endif %}
|
||||
{% if item.glazing.fGlazingSoundproofing >= 10 %}<tr><th>Звукоизоляция:</th><td>{{ item.glazing.fGlazingSoundproofing }} дБ</td></tr>{% endif %}
|
||||
{% if item.glazing.fGlazingLightTransmission >= 1 %}<tr><th>Светопропускание:</th><td>{{ item.glazing.fGlazingLightTransmission }} %</td></tr>{% endif %}
|
||||
{% if item.glazing.fGlazingPassingSun >= 1 %}<tr><th>Солнцепропускание:</th><td>{{ item.glazing.fGlazingPassingSun }} %</td></tr>{% endif %}
|
||||
{% if item.glazing.sGlazingLightReflectance and item.glazing.sGlazingLightReflectance != "—/—" %}<tr><th>Светоотражение:</th><td>{{ item.glazing.sGlazingLightReflectance }} %</td></tr>{% endif %}
|
||||
{% if item.glazing.sGlazingReflectionAndAbsorptionOfHeat and item.glazing.sGlazingReflectionAndAbsorptionOfHeat != "—/—" %}<tr><th>Теплоотражение/погл.:</th><td>{{ item.glazing.sGlazingReflectionAndAbsorptionOfHeat }} %</td></tr>{% endif %}
|
||||
{% if item.glazing.sGlazingToning %}<tr><th>Тонирование:</th><td>{{ item.glazing.sGlazingToning|capfirst }}</td></tr>{% endif %}
|
||||
</table>
|
||||
</div>{# /col стеклопакет #}
|
||||
|
||||
</div>{# /row #}
|
||||
</div>{# /panel-body #}
|
||||
|
||||
{# ---- ПОДВАЛ КАРТОЧКИ: чекбокс «отметить» + кнопка сравнения ---- #}
|
||||
<div class="panel-footer" style="padding:6px 12px;">
|
||||
{# Чекбокс «отметить для сравнения» — учитывается при клике на кнопку любой карточки #}
|
||||
<label style="font-weight:normal;cursor:pointer;margin-right:14px;font-size:small;color:#555;">
|
||||
<input type="checkbox"
|
||||
class="kit-compare-check"
|
||||
value="{{ item.kit.id }}"
|
||||
style="vertical-align:middle;margin-right:3px;" />
|
||||
отметить
|
||||
</label>
|
||||
{# Кнопка: сравнивает текущую карточку с отмеченными (или с лучшей по рейтингу, если ничего не отмечено) #}
|
||||
<button type="button"
|
||||
class="btn btn-default btn-xs"
|
||||
onclick="compareWithKit('{{ item.kit.id }}')">
|
||||
<b class="glyphicon glyphicon-th-list"></b> Сравнить с другими
|
||||
</button>
|
||||
</div>
|
||||
|
||||
</div>{# /panel kit-card #}
|
||||
{% empty %}
|
||||
<div class="alert alert-info">Нет доступных оконных наборов.</div>
|
||||
{% endfor %}
|
||||
|
||||
{# --- Баннер --- #}
|
||||
<div class="row"><div class="col-md-12 col-xs-12"><hr class="dotted-black" />{% include "ad/bannet-wide.html" %}</div></div>
|
||||
|
||||
<div class="row">
|
||||
{% include "report/report_last_user_visit.html" %}
|
||||
{% include "report/report_log_user_visit.html" %}
|
||||
</div>
|
||||
|
||||
</div>{# /container-fluid #}
|
||||
|
||||
<!--- ------------------------------------------------------------------------------------------------------------------------- --->{% endblock %}
|
||||
|
||||
{% block Top_JS3 %}<script>
|
||||
/* Логика кнопки «Сравнить с другими» на карточке набора.
|
||||
ALL_KIT_IDS — все ID в порядке убывания рейтинга (как отсортирован queryset).
|
||||
best2 — два лучших ID по рейтингу (порядок как в queryset, т.е. убывание рейтинга).
|
||||
|
||||
compareWithKit(currentId):
|
||||
checked.length === 0 (ничего не отмечено):
|
||||
— currentId входит в best2 → сравниваем best2 (оба лидера);
|
||||
— иначе → сравниваем currentId с best2[0] (лучшим).
|
||||
checked.length >= 1 (что-то отмечено):
|
||||
— берём checked (до 5 штук), добавляем currentId без дублей → до 6;
|
||||
— если после сборки осталась только одна карточка (отметил только текущую
|
||||
и нажал её) → добавляем best2[0] или best2[1] как партнёра;
|
||||
— сортируем по возрастанию ID (стабильный URL).
|
||||
*/
|
||||
(function () {
|
||||
var MAX_COMPARE = 6;
|
||||
|
||||
/* Все ID наборов в порядке убывания рейтинга */
|
||||
var ALL_KIT_IDS = [{% for item in SET_LIST %}"{{ item.kit.id }}"{% if not forloop.last %},{% endif %}{% endfor %}];
|
||||
|
||||
/* Два лучших по рейтингу ID (первые два из отсортированного queryset) */
|
||||
var best2 = ALL_KIT_IDS.slice(0, 2);
|
||||
|
||||
function getChecked() {
|
||||
/* Возвращает массив ID всех отмеченных чекбоксов на странице */
|
||||
return Array.from(document.querySelectorAll('.kit-compare-check:checked'))
|
||||
.map(function (el) { return el.value; });
|
||||
}
|
||||
|
||||
window.compareWithKit = function (currentId) {
|
||||
currentId = String(currentId);
|
||||
var checked = getChecked();
|
||||
var ids;
|
||||
|
||||
if (checked.length === 0) {
|
||||
/* Ничего не отмечено → сравниваем currentId с лучшим по рейтингу */
|
||||
if (best2.indexOf(currentId) !== -1) {
|
||||
/* currentId уже один из двух лидеров → берём обоих лидеров */
|
||||
ids = best2.slice();
|
||||
} else {
|
||||
/* currentId — не лидер → сравниваем его с лучшим */
|
||||
ids = [currentId, best2[0]];
|
||||
}
|
||||
} else {
|
||||
/* Есть отмеченные → берём checked (до MAX_COMPARE-1=5), добавляем currentId без дублей */
|
||||
ids = checked.slice(0, MAX_COMPARE - 1);
|
||||
if (ids.indexOf(currentId) === -1) {
|
||||
ids.push(currentId);
|
||||
}
|
||||
/* Краевой случай: отмечен только текущий и нажали его же кнопку → одна карточка.
|
||||
Добавляем лучшего по рейтингу как партнёра для сравнения. */
|
||||
if (ids.length === 1) {
|
||||
var extra = (best2[0] !== currentId) ? best2[0] : (best2[1] || null);
|
||||
if (extra) { ids.push(extra); }
|
||||
}
|
||||
/* Сортируем по возрастанию ID для стабильного URL */
|
||||
ids.sort(function (a, b) { return parseInt(a, 10) - parseInt(b, 10); });
|
||||
}
|
||||
|
||||
window.location.href = '/compare_offers/' + ids.join(',');
|
||||
};
|
||||
}());
|
||||
</script>{% endblock %}
|
||||
|
||||
@@ -1,33 +1,23 @@
|
||||
{% extends "base.html" %}{% load static %}
|
||||
|
||||
{% block Title %} Стандартные оконные проёмы типовых серий домов :: каталог{% endblock %}
|
||||
{% block Title %}Стандартные оконные проёмы и балконные блоки для типовых серий домов: размеры, схемы, каталог{% endblock %}
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}Каталог «Окнардия»: стандартные оконные проёмы типовых серий домов...{% endblock %}
|
||||
{% block Description %}Найдите точные размеры (ширину и высоту) и схемы стандартных оконных проёмов и балконных блоков для самых распространённых типовых серий домов в России. Удобный каталог для подбора окон.{% endblock %}
|
||||
|
||||
{% block Keywords %}оконные проёмы, стандартные окна, стандартные оконные проемы, каталог, каталог оконных проёмов, oknardia, окнардия {{ META_KEYWORDS|default:"" }} {% endblock %}
|
||||
{% block Keywords %}типовые окна, размеры окон, оконные проемы, балконный блок, стандартные окна, размеры окон в панельном доме, серия дома, каталог окон, схемы открывания окон, П-44, II-49, 1-515, oknardia, окнардия{% endblock %}
|
||||
|
||||
{% block Date4Meta %}{{ PUB_DAT|date:"c" }}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{{ PUB_DAT|date:"c" }}{% endblock %}
|
||||
{# Date4Meta/Last4Meta не переопределяем: используем дефолт из base.html #}
|
||||
|
||||
{% block Author4Meta %}: Каталог «Окнардия»{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}: Каталог «Окнардия»{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metatags --> #}
|
||||
<meta itemprop="author" content="Каталог «Окнардия»" />{% if IMG_FOR_BLOG %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/media/{{ IMG_FOR_BLOG }}" />{% else %}
|
||||
<meta itemprop="image" content="https://oknardia.ru/static/img/MerDY3gpU0w.jpg" />{% endif %}
|
||||
<meta itemprop="datePublished" content="{{ PUB_DAT|date:"c" }}" />
|
||||
<span itemprop="publisher" itemscope itemtype="http://schema.org/Organization"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<span itemprop="author" itemscope itemtype="http://schema.org/Person"><meta itemprop="name" content="Каталог «Окнардия»" /></span>
|
||||
<meta itemprop="articleSection" content="Каталог «Окнардия»" />
|
||||
<meta itemprop="headline" content="Каталог «Окнардия»: стандартные оконные проёмы типовых серий домов..." />
|
||||
{# Legacy microdata (itemprop/itemscope) удалена: используем JSON-LD в ADD_TO_HEAD #}
|
||||
<meta name="news_keywords" content="{{ HEADER|striptags }}" />
|
||||
<link rel="canonical" href="https://oknardia.ru//catalog/" />
|
||||
<link rel="standout" href="https://oknardia.ru//catalog/" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/catalog/standard_opening" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
@@ -35,21 +25,67 @@
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="https://oknardia.ru//catalog/" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/standard_opening" />
|
||||
<meta property="og:type" content="article" />
|
||||
<meta property="og:title" content="Каталог «Окнардия» | oknardia.ru" />
|
||||
<meta property="og:description" content="Каталог «Окнардия»: стандартные оконные проёмы типовых серий домов..." />
|
||||
<meta property="og:image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<link rel="image_src" href="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:"https://oknardia.ru/static/img/MerDY3gpU0w.jpg" }}" />
|
||||
<meta property="og:title" content="Каталог «Окнардия»: стандартные оконные проёмы типовых серий домов | oknardia.ru" />
|
||||
<meta property="og:description" content="Каталог стандартных оконных проёмов и балконных блоков: размеры, схемы открывания и серии домов, в которых они встречаются." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="Каталог «Окнардия»: стандартные оконные проёмы типовых серий домов... | oknardia.ru"/>
|
||||
<meta name="twitter:description" content="Каталог «Окнардия»: стандартные оконные проёмы типовых серий домов..." />
|
||||
<meta name="twitter:title" content="Стандартные оконные проёмы и балконные блоки | oknardia.ru"/>
|
||||
<meta name="twitter:description" content="Размеры, схемы открывания и типовые серии домов для стандартных оконных проёмов в каталоге Окнардии." />
|
||||
<meta name="twitter:card" content="summary">
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:domain" content="oknardia.ru" />
|
||||
<meta property="twitter:url" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
<meta name="relap-image" content="{% if IMG_FOR_BLOG %}https://oknardia.ru/media/{% endif %}{{ IMG_FOR_BLOG|default:'https://oknardia.ru/static/img/MerDY3gpU0w.jpg' }}" />
|
||||
{# <!-- END Дополнительные Metatags --> #}{% endblock %}
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<meta name="relap-image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
{# <!-- END Дополнительные Metатags --> #}{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}{% comment %}
|
||||
JSON-LD для страницы-списка типовых оконных проемов.
|
||||
CollectionPage + ItemList помогают поисковику понять структуру каталога.
|
||||
{% endcomment %}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "CollectionPage",
|
||||
"name": "Стандартные оконные проёмы и балконные блоки",
|
||||
"description": "Каталог стандартных оконных проёмов и балконных блоков с размерами, схемами открывания и привязкой к типовым сериям домов.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/standard_opening",
|
||||
"inLanguage": "ru-RU",
|
||||
"isPartOf": {
|
||||
"@type": "WebSite",
|
||||
"name": "Окнардия",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}"
|
||||
},
|
||||
"mainEntity": {
|
||||
"@type": "ItemList",
|
||||
"name": "Типовые оконные проёмы",
|
||||
"numberOfItems": {{ LIST_WIN_OPENING|length }},
|
||||
"itemListElement": [
|
||||
{% for i in LIST_WIN_OPENING %}
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": {{ forloop.counter }},
|
||||
"item": {
|
||||
"@type": "Thing",
|
||||
"name": "{{ i.DESCRIPTION|escapejs }}",
|
||||
"description": "{{ i.DESCRIPTION_L|escapejs }}",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/standard_opening/price-{{ i.W|stringformat:'.0f' }}x{{ i.H|stringformat:'.0f' }}mm-tip{{ i.ID }}",
|
||||
"image": "{{ request.scheme }}://{{ request.get_host }}{% static i.URL2IMG %}",
|
||||
"additionalProperty": [
|
||||
{"@type": "PropertyValue", "name": "Ширина", "value": "{{ i.W|stringformat:'.0f' }} мм"},
|
||||
{"@type": "PropertyValue", "name": "Высота", "value": "{{ i.H|stringformat:'.0f' }} мм"},
|
||||
{"@type": "PropertyValue", "name": "Балконный блок: окно", "value": "{% if i.IS_NEAR_DOOR %}да{% else %}нет{% endif %}"},
|
||||
{"@type": "PropertyValue", "name": "Балконный блок: дверь", "value": "{% if i.IS_DOOR %}да{% else %}нет{% endif %}"}
|
||||
]
|
||||
}
|
||||
}{% if not forloop.last %},{% endif %}
|
||||
{% endfor %}
|
||||
]
|
||||
}
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Main_Content %}
|
||||
<div class="container-fluid">
|
||||
@@ -58,18 +94,18 @@
|
||||
<ol class="breadcrumb">
|
||||
<li><a href="/">Главная</a></li>
|
||||
<li><a href="/catalog">Каталог</a></li>
|
||||
<li>Оконные проёмы и балконные блоки</li>
|
||||
<li>Оконные проёмы и балконные блоки</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>{# <!--- Хлебные крошки: КОНЕЦ ---> #}
|
||||
<dIv class="row">
|
||||
{# ПЕРВЫЙ РАЗДЕЛ #}<div class="col-md-9 col-xs-8">
|
||||
<h1>Стандартные оконные проёмы и балконные блоки</h1>
|
||||
<p>Ценовая выдача «Окнардии» основана на базе стандартных оконных проёмов в типовых сериях домов. Для каждого проёма существуют рекомендованные организациями-проектировщиками схемы открывание, но партнёры «Окнардии» могут предложить свои, более расширенные или наоборот сокращенные. В таблице приведены параметры стандартных проёмов базы.</p>
|
||||
<h1>Стандартные оконные проёмы и балконные блоки</h1>
|
||||
<p>Ценовая выдача «Окнардии» основана на базе стандартных оконных проёмов в типовых сериях домов. Для каждого проёма существуют рекоме­ндованные органи­зациями-проекти­ровщиками схемы открывание, но партнёры «Окнардии» могут предложить свои, более расширенные или наоборот сокращенные. В таблице приведены параметры стандартных проёмов базы.</p>
|
||||
</div>
|
||||
{# реклама Oknardia 250x250 СБОКУ #}<div class="col-md-3 col-xs-4 float-right">{% include "ad/bannet-250x250.html" %}</div>
|
||||
<div class="col-md-11 col-xs-12 catalog scrolled">
|
||||
<table class="table-striped table-hover table-responsive flap-catalog">
|
||||
<table class="table-striped table-hover table-responsive flap-catalog" style="margin-top: 1em">
|
||||
<thead>
|
||||
<tr>
|
||||
<th colspan="3" class="c">Размеры (мм)</th>
|
||||
@@ -97,25 +133,16 @@
|
||||
<td data-sort="{% if i.IS_DOOR %}1{% else %}0{% endif %}">{% if i.IS_DOOR %}да{% else %}—{% endif %}</td>
|
||||
<td>{{ i.DESCRIPTION }}</td>
|
||||
<td>{% for j in i.INCLUDING_IN_SERIA %}<a href="/catalog/seria/{{ j.NAME_T }}/all{{ j.ID }}">{{ j.NAME }}</a>{% if not forloop.last %}, {% endif %}{% endfor %}</td>
|
||||
<td><a class="btn btn-default btn-xs" href="/tsena-odnogo-okna/{{ i.W|stringformat:".0f" }}x{{ i.H|stringformat:".0f" }}mm/tip{{ i.ID }}">цены</a></td>
|
||||
{# {% for j in SERIAS %}#}
|
||||
{# <td>{% if j.id in i.INCLUDING_IN_SERIA %}#{% endif %}</td>{% endfor %}#}
|
||||
<td><a class="btn btn-default btn-xs" href="/catalog/standard_opening/price-{{ i.W|stringformat:".0f" }}x{{ i.H|stringformat:".0f" }}mm-tip{{ i.ID }}">цены</a></td>
|
||||
</tr>{% endfor %}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
|
||||
|
||||
<DIV class="col-xs-12" style="height:6em;"></DIV>
|
||||
</dIv>
|
||||
|
||||
{# --- Баннер: НАЧАЛО --- #}
|
||||
<div class="row"><div class="col-md-12 col-xs-12"><hr class="dotted-black" />{% include "ad/bannet-wide.html" %}</div></div>
|
||||
{# --- Баннер: конец --- #}
|
||||
{# --- Баннер: НАЧАЛО --- #}<div class="row"><div class="col-md-12 col-xs-12"><hr class="dotted-black" />{% include "ad/bannet-wide.html" %}</div></div>{# --- Баннер: конец --- #}
|
||||
<div class="row">
|
||||
{% include "report/report_last_user_visit.html" %}
|
||||
{% include "report/report_log_user_visit.html" %}
|
||||
</div>
|
||||
</div>{% endblock %}
|
||||
|
||||
|
||||
|
||||
@@ -1,16 +1,20 @@
|
||||
{% extends "base.html" %}{% load static %}
|
||||
|
||||
{% block Title %}Контакты{% endblock %}
|
||||
{% block Title %}Контакты маркетплейса Окнардия | Адрес, телефон, email для связи{% endblock %}
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"Y-m-d" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"Y-m-d" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Description %}Контактная информация маркетплейс-агрегатора «Окнардии»: адрес, телефоны и email для связи, персоны.{% endblock %}
|
||||
{% block Description %}Контактная информация маркетплейса Окнардия: адрес офиса, email info@oknardia.ru, руководство и учредители, персоны компании.{% endblock %}
|
||||
|
||||
{% block Keywords %}Контакты, контактная информация, телефон для связи, email для связи, адрес, адрес офиса, персоны, Окнардия, маркетплейс-агрегатор «Окнардии»{% endblock %}
|
||||
{% block Keywords %}контакты окнардия, контактная информация, email для связи, адрес офиса, маркетплейс окон, агрегатор окон, руководство окнардии, организационные вопросы, партнерство{% endblock %}
|
||||
|
||||
{% block Author4Meta %}Контакты маркетплейса «Окнардия»{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}Маркетплейс «Окнардия»{% endblock %}
|
||||
|
||||
{% block Top_JS1%}
|
||||
<script type="text/javascript">
|
||||
@@ -18,7 +22,101 @@ $(window).load(function(){var images = $('.half');images.each(function(i){$(this
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metатags для контактов --> #}
|
||||
{# Удалить: устаревшие теги если появятся #}
|
||||
<meta name="news_keywords" content="контакты окнардия, адрес офиса, email, организационные вопросы, партнерство" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/contact/" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:app_id" content="258354027974262" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/contact/" />
|
||||
<meta property="og:type" content="contact" />
|
||||
<meta property="og:title" content="Контакты маркетплейса Окнардия" />
|
||||
<meta property="og:description" content="Свяжитесь с командой Окнардия: адрес офиса, email info@oknardia.ru, руководство и учредители компании." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="Контакты маркетплейса Окнардия" />
|
||||
<meta name="twitter:description" content="Адрес офиса, email для связи и руководство компании Окнардия." />
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta property="twitter:url" content="{{ request.scheme }}://{{ request.get_host }}/contact/" />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
{# <!-- END Дополнительные Metатags для контактов --> #}{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}
|
||||
{# JSON-LD: страница контактов — Organization + ContactPoint + BreadcrumbList #}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "Organization",
|
||||
"name": "Окнардия",
|
||||
"legalName": "Маркетплейс-агрегатор Окнардия",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/",
|
||||
"logo": "{{ request.scheme }}://{{ request.get_host }}/static/img/logo.png",
|
||||
"description": "Маркетплейс-агрегатор для сравнения цен на установку пластиковых и деревянных окон в зданиях типового строительства в России.",
|
||||
"contactPoint": {
|
||||
"@type": "ContactPoint",
|
||||
"contactType": "Customer Support",
|
||||
"email": "info@oknardia.ru",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/contact/"
|
||||
},
|
||||
"sameAs": [
|
||||
"https://www.facebook.com/oknardia",
|
||||
"https://t.me/oknardia"
|
||||
],
|
||||
"founders": [
|
||||
{
|
||||
"@type": "Person",
|
||||
"name": "Сергей Еремин",
|
||||
"jobTitle": "CEO/CTO",
|
||||
"description": "Организационные вопросы, технические решения, партнерство"
|
||||
},
|
||||
{
|
||||
"@type": "Person",
|
||||
"name": "Тимофей Молдованин",
|
||||
"jobTitle": "CFO/COO",
|
||||
"description": "Финансовые и коммерческие вопросы"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Контакты",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/contact/"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Main_Content %}<div class="container-fluid">
|
||||
{# Хлебные крошки: НАЧАЛО #}
|
||||
<div class="row">
|
||||
<div class="col-md-11 col-xs-12">
|
||||
<ol class="breadcrumb">
|
||||
<li><a href="/">Главная</a></li>
|
||||
<li class="active">Контакты</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
{# Хлебные крошки: КОНЕЦ #}
|
||||
|
||||
<div class="row">
|
||||
<div class="col-md-9"><h1>Контакты «Окнардия»</h1></div>
|
||||
</div>
|
||||
@@ -39,7 +137,7 @@ img {background-color: whitesmoke;}
|
||||
|
||||
<div class="row">
|
||||
<div class="col-md-3" >
|
||||
<img class="img-circle" src="https://oknardia.ru/media/img_avatar/avatar_eserg_160x160.png" width="160" height="160" alt="Сергей Еремин — CEO/CTO «Окнардия», организационные вопросы, технические решения, партнерство" />
|
||||
<img class="img-circle" src="{{ request.scheme }}://{{ request.get_host }}/media/img_avatar/avatar_eserg_160x160.png" width="160" height="160" alt="Сергей Еремин — CEO/CTO «Окнардия», организационные вопросы, технические решения, партнерство" />
|
||||
<h3>Сергей Еремин</h3>
|
||||
<h4>CEO/CTO</h4>
|
||||
<h5>организационные вопросы, технические решения, партнерство</h5>
|
||||
@@ -61,7 +159,7 @@ img {background-color: whitesmoke;}
|
||||
</dl>
|
||||
</div>
|
||||
<div class="col-md-3" >
|
||||
<img class="img-circle" src="https://oknardia.ru/media/img_avatar/timofei_molodovanin.jpg" width="160" height="160" alt="Тимофей Молдованин — CFO/COO «Окнардия», финансовые и коммерческие вопросы" />
|
||||
<img class="img-circle" src="{{ request.scheme }}://{{ request.get_host }}/media/img_avatar/timofei_molodovanin.jpg" width="160" height="160" alt="Тимофей Молдованин — CFO/COO «Окнардия», финансовые и коммерческие вопросы" />
|
||||
<h3>Тимофей Молдованин</h3>
|
||||
<h4>CFO/COO</h4>
|
||||
<h5>финансовые и коммерческие вопросы</h5>
|
||||
|
||||
31
oknardia/templates/error/400.html
Normal file
31
oknardia/templates/error/400.html
Normal file
@@ -0,0 +1,31 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>400 — Некорректный запрос</title>
|
||||
<!-- Аналитика: Google Analytics 4, Yandex.Metrika, Top.Mail.Ru -->
|
||||
<script src="/static/js/analytics.js" type="text/javascript"></script>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6; /* Полупрозрачность только для картинки */
|
||||
z-index: -1; /* Уводит картинку под текст */
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 120px; margin: 0; color: #ccc; background-image: url('/static/img/oknardia_logo.gif'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">400</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Некорректный запрос</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">Сервер не может обработать ваш запрос. Пожалуйста, проверьте корректность данных.</h4>
|
||||
<p style="margin-top: 55px;"><a href="/">Вернуться на главную oknardia.ru</a></p>
|
||||
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
29
oknardia/templates/error/401.html
Normal file
29
oknardia/templates/error/401.html
Normal file
@@ -0,0 +1,29 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>401 — Требуется авторизация</title>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6;
|
||||
z-index: -1;
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 120px; margin: 0; color: #ccc; background-image: url('/media/img_seria/1LG-504D12_2.jpg'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">401</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Требуется авторизация</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">Для доступа к этому ресурсу требуется авторизация.</h4>
|
||||
<p style="margin-top: 55px;"><a href="/login-logout">Войти в систему</a> или <a href="/">вернуться на главную</a></p>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
|
||||
32
oknardia/templates/error/403.html
Normal file
32
oknardia/templates/error/403.html
Normal file
@@ -0,0 +1,32 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>403 — Доступ запрещён</title>
|
||||
<!-- Аналитика: Google Analytics 4, Yandex.Metrika, Top.Mail.Ru -->
|
||||
<script src="/static/js/analytics.js" type="text/javascript"></script>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6; /* Полупрозрачность только для картинки */
|
||||
z-index: -1; /* Уводит картинку под текст */
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 120px; margin: 0; color: #ccc; background-image: url('/static/img/oknardia_logo.gif'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">403</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Доступ запрещён</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">У вас нет прав для доступа к этому ресурсу. Если вы считаете это ошибкой, свяжитесь с администратором.</h4>
|
||||
<p style="margin-top: 55px;"><a href="/">Вернуться на главную oknardia.ru</a></p>
|
||||
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
|
||||
31
oknardia/templates/error/404.html
Normal file
31
oknardia/templates/error/404.html
Normal file
@@ -0,0 +1,31 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>404 — Страница не найдена</title>
|
||||
<!-- Аналитика: Google Analytics 4, Yandex.Metrika, Top.Mail.Ru -->
|
||||
<script src="/static/js/analytics.js" type="text/javascript"></script>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6; /* Полупрозрачность только для картинки */
|
||||
z-index: -1; /* Уводит картинку под текст */
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 120px; margin: 0; color: #ccc; background-image: url('/static/img/oknardia_logo.gif'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">404</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Страница не найдена</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">К сожалению, запрашиваемая страница не существует или была удалена.</h4>
|
||||
<p style="margin-top: 55px;"><a href="/">Вернуться на главную oknardia.ru</a></p>
|
||||
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
29
oknardia/templates/error/413.html
Normal file
29
oknardia/templates/error/413.html
Normal file
@@ -0,0 +1,29 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>413 — Файл слишком большой</title>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6;
|
||||
z-index: -1;
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 120px; margin: 0; color: #ccc; background-image: url('/media/img_seria/1LG-504D12_2.jpg'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">413</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Файл слишком большой</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">Размер загружаемого файла превышает допустимый лимит (максимум 10 МБ).</h4>
|
||||
<p style="margin-top: 55px;"><a href="/">Вернуться на главную oknardia.ru</a></p>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
|
||||
29
oknardia/templates/error/429.html
Normal file
29
oknardia/templates/error/429.html
Normal file
@@ -0,0 +1,29 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>429 — Слишком много запросов</title>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6;
|
||||
z-index: -1;
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 120px; margin: 0; color: #ccc; background-image: url('/media/img_seria/1LG-504D12_2.jpg'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">429</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Слишком много запросов</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">Вы отправляете слишком много запросов. Пожалуйста, немного подождите и попробуйте снова.</h4>
|
||||
<p style="margin-top: 55px;"><a href="/">Вернуться на главную oknardia.ru</a></p>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
|
||||
31
oknardia/templates/error/500.html
Normal file
31
oknardia/templates/error/500.html
Normal file
@@ -0,0 +1,31 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>500 — Внутренняя ошибка сервера</title>
|
||||
<!-- Аналитика: Google Analytics 4, Yandex.Metrika, Top.Mail.Ru -->
|
||||
<script src="/static/js/analytics.js" type="text/javascript"></script>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6; /* Полупрозрачность только для картинки */
|
||||
z-index: -1; /* Уводит картинку под текст */
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 120px; margin: 0; color: #ccc; background-image: url('/media/img_seria/1LG-504D12_2.jpg'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">500</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Внутренняя ошибка сервера</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">На нашем сервере произошла непредвиденная ошибка. Команда разработчиков уже работает над решением проблемы.</h4>
|
||||
<p style="color: #999; margin: 20px 0 0 30px; font-size: 14px;">Если проблема сохраняется, пожалуйста, свяжитесь с администратором: <a href="mailto:info@oknardia.ru">info@oknardia.ru</a></p>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
29
oknardia/templates/error/502.html
Normal file
29
oknardia/templates/error/502.html
Normal file
@@ -0,0 +1,29 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>502 — Плохой шлюз</title>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6;
|
||||
z-index: -1;
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 120px; margin: 0; color: #ccc; background-image: url('/media/img_seria/1LG-504D12_2.jpg'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">502</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Плохой шлюз</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">Сервер приложения временно недоступен. Мы работаем над восстановлением работоспособности.</h4>
|
||||
<p style="color: #999; margin: 20px 0 0 30px; font-size: 14px;">Пожалуйста, попробуйте перезагрузить страницу через несколько минут. Если проблема сохраняется, свяжитесь с нами: <a href="mailto:info@oknardia.ru">info@oknardia.ru</a></p>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
|
||||
29
oknardia/templates/error/503.html
Normal file
29
oknardia/templates/error/503.html
Normal file
@@ -0,0 +1,29 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>503 — Сервис недоступен</title>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6;
|
||||
z-index: -1;
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 120px; margin: 0; color: #ccc; background-image: url('/media/img_seria/1LG-504D12_2.jpg'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">503</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Сервис недоступен</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">Сайт временно недоступен для плановых работ. Мы скоро вернёмся!</h4>
|
||||
<p style="color: #999; margin: 20px 0 0 30px; font-size: 14px;">Информация об обслуживании обновляется автоматически. Спасибо за терпение!</p>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
|
||||
29
oknardia/templates/error/504.html
Normal file
29
oknardia/templates/error/504.html
Normal file
@@ -0,0 +1,29 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>504 — Тайм-аут шлюза</title>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6;
|
||||
z-index: -1;
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 120px; margin: 0; color: #ccc; background-image: url('/media/img_seria/1LG-504D12_2.jpg'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">504</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Тайм-аут шлюза</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">Сервер приложения слишком долго отвечает на запрос. Это может быть перегрузка или технический сбой.</h4>
|
||||
<p style="color: #999; margin: 20px 0 0 30px; font-size: 14px;">Попробуйте перезагрузить страницу через несколько минут. Если проблема повторится, свяжитесь с поддержкой: <a href="mailto:info@oknardia.ru">info@oknardia.ru</a></p>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
|
||||
29
oknardia/templates/error/under_reconstruction.html
Normal file
29
oknardia/templates/error/under_reconstruction.html
Normal file
@@ -0,0 +1,29 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>Технические работы</title>
|
||||
<style>
|
||||
.container::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0; left: 0; right: 0; bottom: 0;
|
||||
background-image: url('/media/img_seria/1LG-504D12_2.jpg');
|
||||
opacity: 0.6;
|
||||
z-index: -1;
|
||||
background-position: 18% 15%;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 200px auto;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container" style="margin: 10% 0 0 20%;">
|
||||
<h1 style="font-size: 80px; margin: 0; color: #ccc; background-image: url('/media/img_seria/1LG-504D12_2.jpg'); background-position: left 65%; background-repeat: no-repeat; background-size: 400px auto;"><i style="padding-left: 420px">⚙</i></h1>
|
||||
<h2 style="color: #333; font-size: 40px;">Технические работы</h2>
|
||||
<h4 style="color: #666; margin: 0 0 0 30px; font-size: 20px;">Приносим извинения. В данный момент сайт проходит техническое обслуживание. Мы скоро вернёмся!</h4>
|
||||
<p style="margin-top: 55px;"><a href="/">Вернуться на главную oknardia.ru</a></p>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
|
||||
@@ -1,24 +1,77 @@
|
||||
{% extends "base.html" %}
|
||||
{% load static %}
|
||||
|
||||
{% block Title %}: выбор пластиковых окон в квартиру. Поставщики, цены, описания, характеристики, отзывы.{% endblock %}
|
||||
{% block Title %}Окнардия: агрегатор цен на пластиковые окна и услуги их установки{% endblock %}
|
||||
|
||||
{% block Description %}Окнардия: Здесь собраны цены на установку пластиковых окон. Просто введите адрес и получите актуальные предложения от ведущих поставщиков окон, подробные характеристики профилей и стеклопакетов, информацию о скидках. Никаких предварительных замеров! Мы уже знаем размеры проёмов в квартире, рекомендованные схемы открывания, требования к стеклопакетам, профилю и многое другое. Замена пластиковых окон — ответственное мероприятие. Мы помогаем сделать объективный выбор.{% endblock %}
|
||||
{% block Description %}Агрегатор цен на пластиковые окна в типовых домах России. Введите адрес, укажите тип квартиры и сравните цены поставщиков на установку и характеристики предложений: оконный профиль, стеклопакет, скидки, дополнительные услуги.{% endblock %}
|
||||
|
||||
{% block Keywords %}Цены на окна, цены на пластиковые окна, стоимость замены окон, пластиковые окна в квартиру, скидки на пластиковые окна, окна в квартиру, размеры окон, скидки на пластиковые окна, характеристики пластиковых окон, окна в панельный дом, окна в блочный дом.{% endblock %}
|
||||
{% block Keywords %}цены на окна, пластиковые окна, замена окон, услуги установки окон, профили окон, стеклопакеты, скидки на окна, окна в квартиру, размеры оконных проёмов, поставщики окон{% endblock %}
|
||||
|
||||
{% block Top_JS1 %}{# comment #}
|
||||
<!-- script src="{% static 'js/gears_init.js' %}" type="text/javascript"></script>
|
||||
<script src="{% static 'js/geo.js' %}" type="text/javascript"></script>
|
||||
<script src="http://maps.google.com/maps/api/js?sensor=false" type="text/javascript"></script -->{# endcomment #}
|
||||
<script src="//api-maps.yandex.ru/2.1/?lang=ru_RU" type="text/javascript"></script>
|
||||
{% block Author4Meta %}ОКНАРДИЯ — Оконный Агрегатор{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}ОКНАРДИЯ — Оконный Агрегатор{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- Каноничная ссылка и мета-теги для соцсетей --> #}
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/" />
|
||||
{# <!-- Open Graph теги --> #}
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:type" content="website" />
|
||||
<meta property="og:title" content="Окнардия: агрегатор цен на пластиковые окна" />
|
||||
<meta property="og:description" content="Введите адрес вашего дома — получите справочную информацию о размерах оконных проёмов и сравните цены от поставщиков на типовые комплекты окон с профилями, стеклопакетами, условиями установки." />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/oknardia_logo.svg" />
|
||||
{# <!-- Twitter Cards --> #}
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:title" content="Окнардия — справочник цен на пластиковые окна" />
|
||||
<meta name="twitter:description" content="Сравните характеристики и цены окон от производителей и поставщиков по адресу вашего дома" />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/oknardia_logo.svg" />
|
||||
{# <!-- /Meta-теги --> #}{% endblock %}
|
||||
|
||||
{% block Top_JS1 %}<script src="//api-maps.yandex.ru/2.1/?lang=ru_RU" type="text/javascript"></script>
|
||||
<script src="{% static 'js/jquery-ui.min.js' %}" type="text/javascript"></script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Top_CSS2 %}
|
||||
<link href="{% static 'css/jquery-ui.min.css' %}" rel="stylesheet" type="text/css" />
|
||||
{% block Top_CSS2 %} <link href="{% static 'css/jquery-ui.min.css' %}" rel="stylesheet" type="text/css" />
|
||||
{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}{# <!-- Schema.org JSON-LD разметка для информационного агрегатора --> #}
|
||||
<script type="application/ld+json">
|
||||
[
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "Organization",
|
||||
"name": "ОКНАРДИЯ",
|
||||
"alternateName": "Информационный каталог цен на пластиковые окна",
|
||||
"description": "Независимый информационный сервис для сравнения цен на пластиковые окна, выбору оконного профиля, стеклопакета, условий и услуг их установки от производителей, поставщиков и монтажных компаний.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/",
|
||||
"logo": "{{ request.scheme }}://{{ request.get_host }}/static/img/oknardia_logo.svg",
|
||||
"image": "{{ request.scheme }}://{{ request.get_host }}/static/img/oknardia_logo.svg",
|
||||
"sameAs": [
|
||||
"https://www.facebook.com/oknardia/",
|
||||
"https://twitter.com/oknardia"
|
||||
]
|
||||
},
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "WebSite",
|
||||
"name": "ОКНАРДИЯ",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/",
|
||||
"potentialAction": {
|
||||
"@type": "SearchAction",
|
||||
"target": {
|
||||
"@type": "EntryPoint",
|
||||
"urlTemplate": "{{ request.scheme }}://{{ request.get_host }}/get_address?addr={search_term_string}"
|
||||
},
|
||||
"query-input": "required name=search_term_string"
|
||||
},
|
||||
"description": "Поиск и сравнение цен на пластиковые окна по адресу типового дома"
|
||||
}
|
||||
]
|
||||
</script>
|
||||
{# <!-- /Schema.org JSON-LD --> #}{% endblock %}
|
||||
|
||||
{% block Main_Content %}
|
||||
<script type="text/javascript">
|
||||
// Дождёмся загрузки API и готовности DOM.
|
||||
|
||||
@@ -75,11 +75,7 @@
|
||||
<button type="submit" class="btn btn-primary btn-add">Найти</button>
|
||||
</span>
|
||||
</div>
|
||||
{% if LAST_VISIT %}<div><h5>Ваши последние просмотры:</h5>
|
||||
<ul style="font-size:small">{% for ITEM in LAST_VISIT %}
|
||||
<li><a href="{{ ITEM.LastURL }}">{{ ITEM.LastApart }} <small>({{ ITEM.LastAddress }})</small></a> <small style="font-size: xx-small;">{{ ITEM.Time }}</small></li>{% endfor %}
|
||||
</ul>
|
||||
</div>{% endif %}
|
||||
{% include 'report/report_last_user_visit.html' with background_color="None" %}
|
||||
|
||||
</form>
|
||||
<p></p>{% endwith %}
|
||||
@@ -23,7 +23,7 @@
|
||||
<div class="btn-toolbar" style="width:80%">{% for I_APART in LIST_APART %}
|
||||
<button type="button" class="btn btn-default" style="margin:1ex"
|
||||
accesskey="{{ I_APART.id }}"
|
||||
onclick="window.location.href='/{{ ADDRESS_ID }}/{{ I_APART.id }}/{{ addr_T }}';try{yaCounter32997984.reachGoal('CHOICE_APP');}catch(e){}">
|
||||
onclick="window.location.href='/price/seriaID{{ BASE_SERIA_ID }}--{{ BASE_SERIA_LAT }}/appartID{{ I_APART.id }}/addressID{{ ADDRESS_ID }}--{{ addr_T }}';try{yaCounter32997984.reachGoal('CHOICE_APP');}catch(e){}">
|
||||
{{ I_APART.sNameApartment|safe }}
|
||||
</button>{% endfor %}
|
||||
</div>
|
||||
|
||||
@@ -6,17 +6,109 @@
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Date4Meta %}{{ META_DATA_PUBLISH|date:"c" }}{% endblock %}
|
||||
{# SEO-описание: информативно для поисковиков и людей, но кратно. #}
|
||||
{% block Description %}Лучшие цены на пластиковые окна для серии {{ APART|safe }} в доме по адресу {{ ADDRESS }}. Сравните {{ PRICE_FRAME|length }} предложений от ведущих компаний, узнайте стоимость окон для вашей квартиры и получите скидку!{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{{ META_DATA_PUBLISH|date:"c" }}{% endblock %}
|
||||
{# SEO-ключевые слова: расширяем, добавляем вариации, город, преимущества. #}
|
||||
{% block Keywords %}цены на окна, пластиковые окна, серия {{ BASE_SERIA }}, стоимость окон, окна для {{ BASE_SERIA }}, размеры окон, проемы серии {{ BASE_SERIA }}, окна в {{ APART|safe }}, скидки на окна, {{ ADDRESS }}, оконный профиль, монтаж окон, установка окон, сравнение цен, лучшие предложения, акции, рассрочка, {{ KEYWORDS_EXTRA }}{% endblock %}
|
||||
|
||||
{% block Description %}Цены на окна для серии {{ APART|safe }} по адресу {{ ADDRESS }}. Размер окон (см.): {% for I_WIN_DIM in FLAP_DIM %}{{ I_WIN_DIM.iWinWidth|floatformat:0 }}x{{ I_WIN_DIM.iWinHight|floatformat:0 }}{% if forloop.last %}.{% else %}; {% endif %}{% endfor %} Оконные наборы: {% for CurOffer in PRICE_FRAME %}{{ CurOffer.SETS_NAME }} – {{ CurOffer.FIN_PRICE|stringformat:".0f" }} рублей{% if forloop.last %}.{% else %}; {% endif %}{% endfor %}{% endblock %}
|
||||
{% block ADD_TO_HEAD %}{# --- Микроразметка schema.org, Open Graph, Twitter Card, meta-даты --- #}
|
||||
{# --- JSON-LD микроразметка schema.org: хлебные крошки по новой структуре + остальные объекты --- #}
|
||||
<script type="application/ld+json">
|
||||
[
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Серия {{ BASE_SERIA }}",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/seria/{{ BASE_SERIA_LAT }}/all{{ BASE_SERIA_ID }}"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 3,
|
||||
"name": "{{ APART }}"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 4,
|
||||
"name": "Цены замены окон по адресу: {{ ADDRESS }}"
|
||||
}
|
||||
]
|
||||
},
|
||||
{
|
||||
"@context": "https://schema.org/",
|
||||
"@type": "Organization",
|
||||
"name": "ОКНАРДИЯ — агрегатор цен на окна",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/",
|
||||
"logo": "{{ request.scheme }}://{{ request.get_host }}{% static 'img/oknardia_logo.svg' %}",
|
||||
"description": "Сравнение цен на установку оконных конструкций в типовых жилых домах России",
|
||||
"contactPoint": {"@type": "ContactPoint", "contactType": "Customer Service"}
|
||||
},
|
||||
{
|
||||
"@context": "https://schema.org/",
|
||||
"@type": "Product",
|
||||
"name": "Окна для {{ APART|safe }} ({{ ADDRESS }})",
|
||||
"size": "{% for I_WIN_DIM in FLAP_DIM %}{{ I_WIN_DIM.iWinWidth|floatformat:0 }}x{{ I_WIN_DIM.iWinHight|floatformat:0 }}мм — {{ I_WIN_DIM.iQuantity }} шт.{% if not forloop.last %}; {% endif %}{% endfor %}",
|
||||
"description": "Цены на пластиковые окна для серии {{ APART|safe }} по адресу {{ ADDRESS }}. Сравните предложения, комплектации, получите скидки и выберите лучшее решение!",
|
||||
"image": {"@type": "ImageObject", "url": "{{ request.scheme }}://{{ request.get_host }}{% static 'img/oknardia_logo.svg' %}"},
|
||||
"brand": {"@type": "Brand", "name": "ОКНАРДИЯ"},
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}{{ request.path }}",
|
||||
"offers": {"@type": "AggregateOffer", "priceCurrency": "RUB", "itemCondition": "https://schema.org/NewCondition", "availability": "https://schema.org/InStock", "offerCount": "{{ PRICE_FRAME|length }}"}
|
||||
},
|
||||
{# --- ItemList с Offer для каждого предложения (цена, продавец, рейтинг, дата, внутренняя ссылка, профиль, стеклопакет, список окон) --- #}{
|
||||
"@context": "https://schema.org/",
|
||||
"@type": "ItemList",
|
||||
"itemListElement": [
|
||||
{% for CurOffer in PRICE_FRAME %}
|
||||
{
|
||||
"@type": "Offer",
|
||||
"position": {{ forloop.counter }},
|
||||
"name": "{{ CurOffer.SETS_NAME|escapejs }}",
|
||||
"seller": {"@type": "Organization", "name": "{{ CurOffer.MERCHANT|escapejs }}"},
|
||||
"windows": [
|
||||
{% for CurInOffer in CurOffer.DIM %}{"size": "{{ CurInOffer.WIDTH|stringformat:'d' }}x{{ CurInOffer.HIGHT|stringformat:'d' }}", "count": {{ CurInOffer.QUANTITY }}}{% if not forloop.last %}, {% endif %}{% endfor %}
|
||||
],
|
||||
"profile": "{{ CurOffer.PVC_NAME|escapejs }}",
|
||||
"glazing": "{{ CurOffer.GLAZING_NAME_B|escapejs }}",
|
||||
"price": "{{ CurOffer.FIN_PRICE|stringformat:'d' }}",
|
||||
"priceCurrency": "RUB",
|
||||
{% if CurOffer.SETS_RATING %}"aggregateRating": {"@type": "AggregateRating", "ratingValue": "{{ CurOffer.SETS_RATING|stringformat:'.2f' }}"}, {% endif %}
|
||||
{% if CurOffer.SETS_DATA_MODIFY %}"priceValidUntil": "{{ CurOffer.SETS_DATA_MODIFY|date:'Y-m-d' }}", {% endif %}
|
||||
"availability": "https://schema.org/InStock",
|
||||
"itemCondition": "https://schema.org/NewCondition",
|
||||
"url": "#offer_{{ CurOffer.SETS_ID }}"
|
||||
}{% if not forloop.last %},
|
||||
{% endif %}{% endfor %}
|
||||
]
|
||||
}
|
||||
]
|
||||
</script>
|
||||
{# --- Open Graph (OG) --- #}<meta property="og:title" content="Цены на окна для {{ APART|safe }} ({{ ADDRESS }})" />
|
||||
<meta property="og:description" content="Сравните цены, комплектации и получите лучшие предложения на пластиковые окна для серии {{ APART|safe }} по адресу {{ ADDRESS }}!" />
|
||||
<meta property="og:type" content="website" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}{{ request.path }}" />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}{% static 'img/oknardia_logo.svg' %}" />
|
||||
<meta property="og:site_name" content="ОКНАРДИЯ — агрегатор цен на окна" />
|
||||
{# --- Twitter Card --- #}<meta name="twitter:card" content="summary_large_image" />
|
||||
<meta name="twitter:title" content="Цены на окна для {{ APART|safe }} ({{ ADDRESS }})" />
|
||||
<meta name="twitter:description" content="Сравните цены, комплектации и получите лучшие предложения на пластиковые окна для серии {{ APART|safe }} по адресу {{ ADDRESS }}!" />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}{% static 'img/oknardia_logo.svg' %}" />
|
||||
{# --- Даты публикации и обновления --- #}<meta name="date" content="{{ META_DATA_PUBLISH|date:'Y-m-d' }}" />
|
||||
<meta property="article:published_time" content="{{ META_DATA_PUBLISH|date:'Y-m-d' }}" />
|
||||
<meta property="article:modified_time" content="{{ META_DATA_PUBLISH|date:'Y-m-d' }}" />
|
||||
{% endblock %}
|
||||
|
||||
{% comment %}{% block Description %}Цены на плаcтиковые окна для серии {{ BASE_SERIA }} ({{ APART }} квартира, {{ ADDRESS }}) :: {% for CurOffer in PRICE_FRAME %}Поставщик: {{ CurOffer.MERCHANT }}; Комплектация: {{ CurOffer.SETS_NAME }}; Цена: {{ CurOffer.FIN_PRICE }}₽ :: {% endfor %}{% endblock %}{% endcomment %}
|
||||
|
||||
{% block Keywords %}цены окон, серия {{ BASE_SERIA }}, {{ BASE_SERIA }}, стоимость окон, окна для {{ BASE_SERIA }}, размеры окон, проемы серии {{ BASE_SERIA }}, окна в {{ APART|safe }}, скидки на окна, {{ ADDRESS }}, оконный профиль, {% for CurOffer in PRICE_FRAME %}{{ CurOffer.MERCHANT }}, {{ CurOffer.PVC_NAME }}, {{ CurOffer.PVC_MANUFACTURER }}, {{ CurOffer.GLAZING_MARK }}, {% endfor %} характеристики пластиковых окон, {% for I_WIN_DIM in FLAP_DIM %}{{ I_WIN_DIM.iWinWidth|floatformat:0 }}x{{ I_WIN_DIM.iWinHight|floatformat:0 }} см., {% endfor %}{{ META_KEYWORDS|default:"" }}{% endblock %}
|
||||
|
||||
{% block Top_JS3%}<script type="text/javascript">
|
||||
{% block Top_JS3%}<script type="text/javascript" src="{% static 'js/track_user_visit.js' %}"></script>
|
||||
<script type="text/javascript">
|
||||
function show_phone_num( id ){ // колапсатор для отображения контатной информации постафшика окон
|
||||
$('#tel'+id).collapse('show');
|
||||
$('#hid'+id).collapse('hide');
|
||||
@@ -107,26 +199,36 @@ $(function () { // инициализация и обработка попове
|
||||
|
||||
{% block Top_CSS1 %}<link rel="stylesheet" type="text/css" href="{% static "css/csshake-vertical.min.css" %}">{% endblock %}
|
||||
|
||||
{% block Main_Content %}
|
||||
<span itemscope itemtype="http://schema.org/Product">
|
||||
<div class="row col-md-12">
|
||||
{% block Main_Content %}<div class="container-fluid">
|
||||
{# --- Хлебные крошки: НАЧАЛО --- #}<div class="row">
|
||||
<div class="col-md-12">
|
||||
<ol class="breadcrumb">
|
||||
<li><a href="/">Главная</a></li>
|
||||
<li><a href="/catalog/seria/{{ BASE_SERIA_LAT }}/all{{ BASE_SERIA_ID }}">Серия {{ BASE_SERIA }}</a></li>
|
||||
<li>{{ APART }}</li>
|
||||
<li>Цены замены окон по адресу: {{ ADDRESS }}</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>{# --- Хлебные крошки: КОНЕЦ --- #}
|
||||
<div class="row">
|
||||
<div class="col-md-9">
|
||||
<h1>Цены на окна для серии {{ APART|safe }} <small>({{ ADDRESS }})</small></h1>
|
||||
<h1>Цены на окна для серии {{ APART|safe }} <small><nobr>({{ ADDRESS }})</nobr></small></h1>
|
||||
</div>
|
||||
<div class="col-md-9">
|
||||
<p>Квартира имеет проёмы (окна и балконные двери) следующих размеров: {% for I_WIN_DIM in FLAP_DIM %}{% if not forloop.first %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ I_WIN_DIM.iWinWidth|floatformat:0 }}x{{ I_WIN_DIM.iWinHight|floatformat:0 }} см. — {{ I_WIN_DIM.iQuantity }} шт.{% endfor %} Проект (<a href="/catalog/seria/{{ BASE_SERIA_LAT }}/all{{ BASE_SERIA_ID }}">типовая серия {{ BASE_SERIA }}</a>) предполагает следующие схемы открывания окон:</p>
|
||||
</div>
|
||||
{# Микроразмектка: названеи продукта #}<meta itemprop="name" content="Окна {{ APART|safe }} ({{ ADDRESS }})" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="row col-md-12 ShowBigFlapPictures">
|
||||
<div class="col-sm-9">
|
||||
{% include 'report/show_big_flap_pictures.html' %}
|
||||
</div>
|
||||
<div class="col-sm-3 visible-md visible-lg ap_list">
|
||||
<div class="row ShowBigFlapPictures">
|
||||
<div class="col-sm-9">{% include 'report/show_big_flap_pictures.html' %}</div>
|
||||
<div class="col-sm-3 visible-md visible-lg ap_list">
|
||||
<h6>Другие типовые квартиры в этом доме:</h6>
|
||||
<ul>{% for I_APART in APARTMENT_IN_BUILDING %}
|
||||
{% if I_APART.APT_ID == '!' %}<li>{{ I_APART.APT_NAME|safe }}</li>{% else %}<li><a href="/{{ BUILD_ID }}/{{ I_APART.APT_ID }}/{{ ADDRESS_T }}">{{ I_APART.APT_NAME|safe }}</a></li>{% endif %}{% endfor %}
|
||||
{% if I_APART.APT_ID == '!' %}<li>{{ I_APART.APT_NAME|safe }}</li>{% else %}
|
||||
{# Новый формат роутинга для перехода между квартирами #}
|
||||
<li><a href="/price/seriaID{{ BASE_SERIA_ID }}--{{ BASE_SERIA_LAT }}/appartID{{ I_APART.APT_ID }}/addressID{{ BUILD_ID }}--{{ ADDRESS_T }}/">{{ I_APART.APT_NAME|safe }}</a></li>
|
||||
{% endif %}{% endfor %}
|
||||
</ul>
|
||||
<a href="/catalog/seria/{{ BASE_SERIA_LAT }}/all{{ BASE_SERIA_ID }}">Информация по серии {{ BASE_SERIA }}</a>
|
||||
</div>
|
||||
@@ -134,10 +236,8 @@ $(function () { // инициализация и обработка попове
|
||||
|
||||
<div class="row col-md-12">
|
||||
<div class="col-md-12">
|
||||
<p id="tab-note">Таблица содержит цены поставщиков. Клик на название отобразит детальные спецификации каждого предложения: марку профиля рамы и створки, схему стеклопакета, тип фурнитуры, элементы отделки (отлив, подоконник, откос, клапан <nobr>климат-контроля</nobr>) и сопутствующие услуги. Предложения выводятся покадрово, получите следующий кадр кнопкой «Ещё коммерческие предложения окон» под таблицей. Просмотреть и сравнить технические характеристик стеклопакетов, профилей и детальное описание сопутствующих услуг возможно с помощью кнопки «Сравнить выбранные».</p>
|
||||
<p id="tab-note">Таблица содержит цены поставщиков. Клик на название отобразит детальные спецификации каждого предложения: марку профиля рамы и створки, схему стеклопакета, тип фурнитуры, элементы отделки (отлив, подоконник, откос, клапан <nobr>климат-контроля</nobr>) и сопутствующие услуги. Предложения выводятся покадрово, получите следующий кадр кнопкой «Ещё коммерческие предложения окон» под таблицей. Просмотреть и сравнить технические характеристики стеклопакетов, профилей и детальное описание сопутствующих услуг возможно с помощью кнопки «Сравнить выбранные».</p>
|
||||
</div>
|
||||
{# Микроразмектка: названеи продукта #}
|
||||
<meta itemprop="name" content="Окна {{ APART|safe }} ({{ ADDRESS }})"/>
|
||||
</div>
|
||||
|
||||
|
||||
@@ -168,17 +268,23 @@ $(function () { // инициализация и обработка попове
|
||||
{% include "price/price_list_frame.html" %}
|
||||
</tbody>
|
||||
</table>
|
||||
</form>
|
||||
</span>
|
||||
</form>
|
||||
{% with SERIA_BASE=BASE_SERIA %}{% include "report/build_info_in_table.html" %}{% endwith %}
|
||||
{# --- Баннер: НАЧАЛО --- #}
|
||||
<div class="row"><div class="col-md-12 col-xs-12"><hr class="dotted-black" />{% include "ad/bannet-wide.html" %}</div></div>
|
||||
{# --- Баннер: конец --- #}
|
||||
<div class="row">
|
||||
{% include "report/report_last_user_visit.html" %}
|
||||
{% include "report/report_log_user_visit.html" %}
|
||||
<p id="shadow_buffer"></p>
|
||||
</div>
|
||||
<div class="row">
|
||||
{% include "report/report_last_user_visit.html" %}
|
||||
{% include "report/report_log_user_visit.html" %}
|
||||
<p id="shadow_buffer"></p>
|
||||
</div>
|
||||
|
||||
{# Скрытый элемент для отслеживания визитов пользователя (передача данных в JS track_user_visit.js) #}
|
||||
<div id="tracking-data"
|
||||
data-current-url="{{ request.path }}"
|
||||
data-address="{{ ADDRESS }}"
|
||||
data-apart="{{ APART }}"
|
||||
style="display: none;"></div>
|
||||
|
||||
{# модальное окно #}
|
||||
<div class="modal fade bs-example-modal-sm" id="modal-exclamation" tabindex="-1" role="dialog">
|
||||
|
||||
@@ -7,8 +7,8 @@
|
||||
{% if forloop.first %}
|
||||
<th rowspan="{% if CurOffer.DIM|length == 1 %}2{% else %}{{ CurOffer.DIM|length }}{% endif %}" title="Добавить коммерческое предложение окон к сравнению">{# красивые чекбоксы BEGIN #}<div class="checkbox"><label><input id="CHK{{ CurOffer.SETS_ID }}" type="checkbox" name="ForCompare" value="{{ CurOffer.SETS_ID }}" onChange="ChangeCountCheckedBox({{ CurOffer.SETS_ID }});" /><span class="cr"><i class="cr-icon glyphicon glyphicon-ok"></i></span></label></div>{# красивые чекбоксы END #}</th>
|
||||
<td rowspan="{% if CurOffer.DIM|length == 1 %}2{% else %}{{ CurOffer.DIM|length }}{% endif %}"{% if CurOffer.IS_COMMERCIAL %} style="background-image: url(/media/{{ CurOffer.MERCHANT_LOGO }})"{% endif %} title="Краткая спецификация коммерческого предложения">
|
||||
<span itemprop="description">
|
||||
<h3 class="set-name shake-trigger" id="btn{{ CurOffer.SETS_ID }}"><a href="javascript://" onclick="show_dtl({{ CurOffer.SETS_ID }})">{{ CurOffer.MERCHANT }} – {{ CurOffer.SETS_NAME }}<i class="glyphicon glyphicon-chevron-down shake-vertical"></i></a></h3>
|
||||
<span>
|
||||
<h3 class="set-name shake-trigger" id="btn{{ CurOffer.SETS_ID }}"><a href="javascript://" onclick="show_dtl({{ CurOffer.SETS_ID }})">{{ CurOffer.MERCHANT }} – {{ CurOffer.SETS_NAME }}<i class="glyphicon glyphicon-chevron-down shake-vertical"></i></a></h3>
|
||||
|
||||
<DiV id="dtl{{ CurOffer.SETS_ID }}" class="collapse">■ Профиль: <a href="/catalog/profile/{{ CurOffer.PVC_ID }}-{{ CurOffer.PVC_MANUFACTURER_T }}/{{ CurOffer.PVC_ID }}-{{ CurOffer.PVC_NAME_T }}">{{ CurOffer.PVC_NAME|safe }}</a> (<a href="/catalog/profile/{{ CurOffer.PVC_ID }}-{{ CurOffer.PVC_MANUFACTURER_T }}">{{ CurOffer.PVC_MANUFACTURER }}</a>)
|
||||
■ {{ CurOffer.GLAZING_NAME_B|safe }} <nobr>({{ CurOffer.GLAZING_MARK }})</nobr>
|
||||
@@ -40,10 +40,8 @@
|
||||
title="{% if CurOffer.SETS_RATING > 0.01 %}<b> Рейтинг {{ CurOffer.SETS_RATING|stringformat:".2f" }}</b> для оконого набора «{{ CurOffer.SETS_NAME }}» компании «{{ CurOffer.MERCHANT }}» состоит из:{% else %}Рейтинг не присвоен{% endif %}"
|
||||
data-toggle="popover">рейтинг</a>: {% for Star in CurOffer.SETS_RATING_STARTS %}{% if Star == 0 %}<b class="glyphicon glyphicon-star-empty"></b>{% else %}<b class="glyphicon glyphicon-star"></b>{% endif %}{% endfor %} {% if CurOffer.SETS_RATING > -0.1 %} {{ CurOffer.SETS_RATING|stringformat:".2f" }}{% endif %}</nobr>
|
||||
</span>
|
||||
<span itemprop="brand" itemscope itemtype="http://schema.org/Brand">
|
||||
<meta itemprop="name" content="{{ CurOffer.MERCHANT }}" />
|
||||
<meta itemprop="logo" content="https://oknardia.ru/media/{{ CurOffer.MERCHANT_LOGO }}" />
|
||||
</span></td>
|
||||
{# Удалить: старая микроразметка schema.org (brand, meta) #}
|
||||
</td>
|
||||
<!--- Конец большой ячейки со спецификацией оконного предложения --->
|
||||
{% endif %}
|
||||
|
||||
@@ -58,10 +56,9 @@
|
||||
|
||||
<td class="rnw" title="Стоимость {{ CurOffer.TOTAL|stringformat:".2f" }} рублей за все окна квартиры {{ APART|safe }}.">{{ CurOffer.TOTAL|stringformat:".2f"|price_format }}</td>
|
||||
<th{% if CurOffer.DISCOUNT_COLOR2 != "" %} style="background-color:{{ CurOffer.DISCOUNT_COLOR2 }};"{% endif %} title="{% if CurOffer.DISCOUNT < 0.1 %}Нет скидки{% else %}Скидка — {{ CurOffer.DISCOUNT|stringformat:".1f" }}%{% endif %}">{% if CurOffer.DISCOUNT < 0.1 %}—{% else %}−{{ CurOffer.DISCOUNT|stringformat:".1f" }}%{% endif %}</th>
|
||||
<th{% if CurOffer.DISCOUNT_COLOR1 != "" %} style="background-color:{{ CurOffer.DISCOUNT_COLOR1 }};"{% endif %} itemprop="offers" itemscope itemtype="http://schema.org/Offer" title="Итого за все окна с учетом скидки: {{ CurOffer.FIN_PRICE|stringformat:".2f" }} рублей">
|
||||
<th{% if CurOffer.DISCOUNT_COLOR1 != "" %} style="background-color:{{ CurOffer.DISCOUNT_COLOR1 }};"{% endif %} title="Итого за все окна с учетом скидки: {{ CurOffer.FIN_PRICE|stringformat:".2f" }} рублей">
|
||||
Итого: {{ CurOffer.FIN_PRICE|stringformat:".2f"|price_format }} <small class="glyphicon glyphicon-ruble" aria-label="₽ (руб.)" title="₽ (руб.)"></small>
|
||||
<meta itemprop="price" content="{{ CurOffer.FIN_PRICE }}" />
|
||||
<meta itemprop="priceCurrency" content="RUB" />
|
||||
{# Удалить: старая микроразметка schema.org (meta price, priceCurrency) #}
|
||||
</th>
|
||||
{% if CurOffer.DIM|length == 1 %}
|
||||
|
||||
|
||||
@@ -5,11 +5,12 @@
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Date4Meta %}{{ META_DATA_PUBLISH|date:"c" }}{% endblock %}
|
||||
{# SEO блоки дат:#}
|
||||
{# - Date4Meta: дата публикации (первого появления) — используем дату модификации данных. #}
|
||||
{# - Last4Meta: дата последнего обновления — будет по умолчанию now из base.html. #}
|
||||
{% block Date4Meta %}{{ META_DATA_PUBLISH|date:"Y-m-d" }}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{{ META_DATA_PUBLISH|date:"c" }}{% endblock %}
|
||||
|
||||
{% block Top_JS4 %}
|
||||
{% block Top_JS4 %}{# Для построения круговой диаграммы #}
|
||||
<script type="text/javascript" src="//www.gstatic.com/charts/loader.js"></script>
|
||||
<script type="text/javascript">
|
||||
google.charts.load("current", {packages: ["corechart"]});
|
||||
@@ -35,6 +36,140 @@
|
||||
}
|
||||
</script>{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}{% comment %}
|
||||
JSON-LD микроразметка для поисковых систем (Schema.org):
|
||||
- BreadcrumbList: хлебные крошки для навигации в поиске
|
||||
- Organization: информация о бренде/компании
|
||||
- Product: типовое окно с полной информацией
|
||||
- Рейтинги и цены берутся из таблицы предложений (price_offers_for_one_window_frame.html)
|
||||
{% endcomment %}<script type="application/ld+json">
|
||||
[
|
||||
{
|
||||
"@context": "https://schema.org/",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Каталог",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 3,
|
||||
"name": "Оконные проёмы и балконные блоки",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/standard_opening/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 4,
|
||||
"name": "Окно {% for I_WIN_DIM in FLAP_DIM %}{{ I_WIN_DIM.iWinWidth_mm|floatformat:0 }}×{{ I_WIN_DIM.iWinHight_mm|floatformat:0 }}{% endfor %} мм",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}{{ request.path }}"
|
||||
}
|
||||
]
|
||||
},
|
||||
{
|
||||
"@context": "https://schema.org/",
|
||||
"@type": "Organization",
|
||||
"name": "ОКНАРДИЯ — агрегатор цен на окна",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/",
|
||||
"logo": "{{ request.scheme }}://{{ request.get_host }}{% static 'img/oknardia_logo.svg' %}",
|
||||
"description": "Сравнение цен на установку оконных конструкций в типовых жилых домах России",
|
||||
"contactPoint": {
|
||||
"@type": "ContactPoint",
|
||||
"contactType": "Customer Service"
|
||||
}
|
||||
},
|
||||
{
|
||||
"@context": "https://schema.org/",
|
||||
"@type": "Product",
|
||||
"name": "Типовое пластиковое окно {% for I_WIN_DIM in FLAP_DIM %}{{ I_WIN_DIM.iWinWidth|floatformat:0 }}×{{ I_WIN_DIM.iWinHight|floatformat:0 }} см{% endfor %}",
|
||||
"size": "{% for I_WIN_DIM in FLAP_DIM %}{% if not forloop.first %}, {% endif %}{{ I_WIN_DIM.iWinWidth_mm|floatformat:0 }}×{{ I_WIN_DIM.iWinHight_mm|floatformat:0 }} мм{% endfor %}",
|
||||
"description": "Цены на пластиковое окно стандартного размера для типовых жилых домов серий {% for I in SERIA_FOR_WIN %}{% if forloop.last %} и {% elif forloop.first %}{% else %}, {% endif %}{{ I.sName }}{% endfor %}. Сравните предложения различных производителей и установщиков, узнайте актуальные цены, технические характеристики стеклопакетов, профилей, фурнитуры и условия доставки/монтажа.",
|
||||
"image": {
|
||||
"@type": "ImageObject",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}{% static 'img/oknardia_logo.svg' %}"
|
||||
},
|
||||
"brand": {
|
||||
"@type": "Brand",
|
||||
"name": "ОКНАРДИЯ"
|
||||
},
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}{{ request.path }}",
|
||||
"offers": {
|
||||
"@type": "AggregateOffer",
|
||||
"priceCurrency": "RUB",
|
||||
"itemCondition": "https://schema.org/NewCondition",
|
||||
"availability": "https://schema.org/InStock",
|
||||
"offerCount": "{{ NUM_TOTAL_OFFER_N_WORD|safe }}"
|
||||
},
|
||||
"datePublished": "{{ META_DATA_PUBLISH|date:'Y-m-d' }}",
|
||||
"dateModified": "{{ META_DATA_PUBLISH|date:'Y-m-d' }}"
|
||||
},
|
||||
{
|
||||
"@context": "https://schema.org/",
|
||||
"@type": "ItemList",
|
||||
"name": "Коммерческие предложения для типового окна",
|
||||
"numberOfItems": "{{ PRICE_FRAME|length }}",
|
||||
"itemListElement": [
|
||||
{% for CurOffer in PRICE_FRAME %}
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": {{ forloop.counter }},
|
||||
"item": {
|
||||
"@type": "Offer",
|
||||
"name": "{{ CurOffer.SETS_NAME|striptags|escapejs }}",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}{{ request.path }}#btn{{ CurOffer.SETS_ID }}",
|
||||
"price": "{{ CurOffer.FIN_PRICE|stringformat:'.2f' }}",
|
||||
"priceCurrency": "RUB",
|
||||
"itemCondition": "https://schema.org/NewCondition",
|
||||
"availability": "https://schema.org/InStock",
|
||||
"seller": {
|
||||
"@type": "Organization",
|
||||
"name": "{{ CurOffer.MERCHANT|striptags|escapejs }}"
|
||||
},
|
||||
"itemOffered": {
|
||||
"@type": "Product",
|
||||
"name": "{{ CurOffer.SETS_NAME|striptags|escapejs }}",
|
||||
"additionalProperty": [
|
||||
{
|
||||
"@type": "PropertyValue",
|
||||
"name": "Оконный профиль",
|
||||
"value": "{{ CurOffer.PVC_NAME|striptags|escapejs }}{% if CurOffer.PVC_MANUFACTURER %} ({{ CurOffer.PVC_MANUFACTURER|striptags|escapejs }}){% endif %}"
|
||||
}{% if CurOffer.PVC_MANUFACTURER %},
|
||||
{
|
||||
"@type": "PropertyValue",
|
||||
"name": "Производитель профиля",
|
||||
"value": "{{ CurOffer.PVC_MANUFACTURER|striptags|escapejs }}"
|
||||
}{% endif %}{% if CurOffer.GLAZING_MARK %},
|
||||
{
|
||||
"@type": "PropertyValue",
|
||||
"name": "Стеклопакет",
|
||||
"value": "{{ CurOffer.GLAZING_MARK|striptags|escapejs }}"
|
||||
}{% endif %}
|
||||
]
|
||||
},
|
||||
"dateModified": "{{ CurOffer.SETS_DATA_MODIFY|date:'Y-m-d' }}"{% if CurOffer.SETS_RATING > -0.1 %},
|
||||
"aggregateRating": {
|
||||
"@type": "AggregateRating",
|
||||
"ratingValue": "{{ CurOffer.SETS_RATING|stringformat:'.2f' }}",
|
||||
"bestRating": "5",
|
||||
"worstRating": "0",
|
||||
"ratingCount": "1"
|
||||
}{% endif %}
|
||||
}
|
||||
}{% if not forloop.last %},{% endif %}
|
||||
{% endfor %}
|
||||
]
|
||||
}
|
||||
]
|
||||
</script>{% endblock %}
|
||||
|
||||
{% block Description %}Цены на типовое окно {% for I_WIN_DIM in FLAP_DIM %}{{ I_WIN_DIM.iWinWidth|floatformat:0 }}x{{ I_WIN_DIM.iWinHight|floatformat:0 }} см. для домов серий {% for I in SERIA_FOR_WIN %}{% if forloop.last %} и {% elif forloop.first %}{% else %}, {% endif %}{{ I.sName }}{% endfor %}{% endfor %}.{% endblock %}
|
||||
|
||||
{% comment %}{% block Description %}Цены на пластиковые окна для серии {{ BASE_SERIA }} ({{ APART }} квартира, {{ ADDRESS }}) :: {% for CurOffer in PRICE_FRAME %}Поставщик: {{ CurOffer.MERCHANT }}; Комплектация: {{ CurOffer.SETS_NAME }}; Цена: {{ CurOffer.FIN_PRICE }}₽ :: {% endfor %}{% endblock %}{% endcomment %}
|
||||
@@ -146,12 +281,14 @@ $(function () { // инициализация и обработка попове
|
||||
<span itemscope itemtype="http://schema.org/Product">
|
||||
<div class="row">
|
||||
<div class="col-md-9">
|
||||
<h1>Цены на окно {% for I_WIN_DIM in FLAP_DIM %}{{ I_WIN_DIM.iWinWidth_mm|floatformat:0 }}×{{ I_WIN_DIM.iWinHight_mm|floatformat:0 }}{% endfor %} мм. <small>(типовое)</small></h1>
|
||||
<h1 itemprop="name">Цены на окно {% for I_WIN_DIM in FLAP_DIM %}{{ I_WIN_DIM.iWinWidth_mm|floatformat:0 }}×{{ I_WIN_DIM.iWinHight_mm|floatformat:0 }}{% endfor %} мм. <small>(типовое)</small></h1>
|
||||
</div>
|
||||
<div class="col-md-9">
|
||||
<p>Типовой проём {% for I_WIN_DIM in FLAP_DIM %}{{ I_WIN_DIM.iWinWidth|floatformat:1 }}×{{ I_WIN_DIM.iWinHight|floatformat:1 }}{% endfor %} cм. представлен в домах серий: {% for I in SERIA_FOR_WIN %}{% if forloop.last %} и {% elif forloop.first %}{% else %}, {% endif %}<a href="/catalog/seria/{{ I.sNameLat }}/all{{ I.id }}">{{ I.sName }}</a>{% endfor %}. База «Окнардии» размещено {{ NUM_TOTAL_OFFER_N_WORD }} цен для окон в такой проем (из них в архиве {{ NUM_ARCHIVE_OFFER }}). Предложено {{ NUM_FLAP_VARIATION_IN_WORD }} открывания от {{ NUM_TOTAL_FIRM_N_WORD }}.</p>
|
||||
<p itemprop="description">Типовой проём {% for I_WIN_DIM in FLAP_DIM %}{{ I_WIN_DIM.iWinWidth|floatformat:1 }}×{{ I_WIN_DIM.iWinHight|floatformat:1 }}{% endfor %} cм. представлен в домах серий: {% for I in SERIA_FOR_WIN %}{% if forloop.last %} и {% elif forloop.first %}{% else %}, {% endif %}<a href="/catalog/seria/{{ I.sNameLat }}/all{{ I.id }}">{{ I.sName }}</a>{% endfor %}. База «Окнардии» размещено {{ NUM_TOTAL_OFFER_N_WORD }} цен для окон в такой проем (из них в архиве {{ NUM_ARCHIVE_OFFER }}). Предложено {{ NUM_FLAP_VARIATION_IN_WORD }} открывания от {{ NUM_TOTAL_FIRM_N_WORD }}.</p>
|
||||
</div>
|
||||
{# Микроразмектка: названеи продукта #}<meta itemprop="name" content="Окна {{ APART|safe }} ({{ ADDRESS }})" />
|
||||
{# Микроразметка: название продукта и марка #}
|
||||
<meta itemprop="brand" content="ОКНАРДИЯ — агрегатор цен на окна" />
|
||||
<meta itemprop="productionDate" content="{{ META_DATA_PUBLISH|date:'Y-m-d' }}" />
|
||||
</div>
|
||||
|
||||
<div class="row ShowBigFlapPictures">
|
||||
@@ -176,8 +313,6 @@ $(function () { // инициализация и обработка попове
|
||||
<div class="col-md-12">
|
||||
<p id="tab-note">В таблице представлены только цены поставщиков из базы «Окнардия». Клик на названии набора отобразит детальную спецификацию каждого предложения: профиль рамы и створки, схему стеклопакета, фурнитуру, элементы отлива, подоконника, откоса, системы <nobr>климат-контроля</nobr>) и сопутствующие услуги. Предложения выводятся блоками. Очередной блок выводится кнопкой «Ещё коммерческие предложения окон» под таблицей. Детальные технические характеристики стеклопакетов, профилей и описание сопутствующих услуг можно посмотреть и сравнить с помощью кнопки «Сравнить выбранные».</p>
|
||||
</div>
|
||||
{# Микроразмектка: названеи продукта #}
|
||||
<meta itemprop="name" content="Окна {{ APART|safe }} ({{ ADDRESS }})"/>
|
||||
</div>
|
||||
|
||||
|
||||
|
||||
@@ -6,8 +6,8 @@
|
||||
{% if forloop.first %}
|
||||
<th rowspan="{% if CurOffer.DIM|length == 1 %}2{% else %}{{ CurOffer.DIM|length }}{% endif %}" title="Добавить коммерческое предложение окна к сравнению">{# красивые чекбоксы BEGIN #}<div class="checkbox"><label><input id="CHK{{ CurOffer.SETS_ID }}" type="checkbox" name="ForCompare" value="{{ CurOffer.SETS_ID }}" onChange="ChangeCountCheckedBox({{ CurOffer.SETS_ID }});" /><span class="cr"><i class="cr-icon glyphicon glyphicon-ok"></i></span></label></div>{# красивые чекбоксы END #}</th>
|
||||
<td rowspan="{% if CurOffer.DIM|length == 1 %}2{% else %}{{ CurOffer.DIM|length }}{% endif %}"{% if CurOffer.IS_COMMERCIAL %} style="background-image: url(/media/{{ CurOffer.MERCHANT_LOGO }})"{% endif %} title="Краткая спецификация коммерческого предложения">
|
||||
<span itemprop="description">
|
||||
<h3 class="set-name shake-trigger" id="btn{{ CurOffer.SETS_ID }}"><a href="javascript://" onclick="show_dtl({{ CurOffer.SETS_ID }})">{{ CurOffer.MERCHANT }} – {{ CurOffer.SETS_NAME }}<i class="glyphicon glyphicon-chevron-down shake-vertical"></i></a></h3>
|
||||
<span>
|
||||
<h3 class="set-name shake-trigger" id="btn{{ CurOffer.SETS_ID }}"><a href="javascript://" onclick="show_dtl({{ CurOffer.SETS_ID }})">{{ CurOffer.MERCHANT }} – {{ CurOffer.SETS_NAME }}<i class="glyphicon glyphicon-chevron-down shake-vertical"></i></a></h3>
|
||||
|
||||
<DiV id="dtl{{ CurOffer.SETS_ID }}" class="collapse">■ Профиль: <a href="/catalog/profile/{{ CurOffer.PVC_ID }}-{{ CurOffer.PVC_MANUFACTURER_T }}/{{ CurOffer.PVC_ID }}-{{ CurOffer.PVC_NAME_T }}">{{ CurOffer.PVC_NAME|safe }}</a> (<a href="/catalog/profile/{{ CurOffer.PVC_ID }}-{{ CurOffer.PVC_MANUFACTURER_T }}">{{ CurOffer.PVC_MANUFACTURER }}</a>)
|
||||
■ {{ CurOffer.GLAZING_NAME_B|safe }} <nobr>({{ CurOffer.GLAZING_MARK }})</nobr>
|
||||
@@ -29,17 +29,15 @@
|
||||
</DiV>
|
||||
<!-- Дата обновления -->
|
||||
<nobr class="badge badge4price" title="Дата обновления коммерческого предложения окон — {{ CurOffer.SETS_DATA_MODIFY|date:"d.M.Y" }}"><b class="glyphicon glyphicon-calendar"></b> {{ CurOffer.SETS_DATA_MODIFY|date:"d.M.Y" }}</nobr>
|
||||
<!-- Звездочки рейтинга -->
|
||||
<nobr class="badge badge4price" title="Рейтинг «Окнардии»{% if CurOffer.SETS_RATING > -0.1 %} — {{ CurOffer.SETS_RATING|stringformat:".2f" }} баллов{% endif %}"><a
|
||||
<!-- Звездочки рейтинга с микроразметкой Rating -->
|
||||
<nobr class="badge badge4price" title="Рейтинг «Окнардии»{% if CurOffer.SETS_RATING > -0.1 %} — {{ CurOffer.SETS_RATING|stringformat:".2f" }} баллов{% endif %}">
|
||||
<a
|
||||
href="javascript://"
|
||||
id-set="{{ CurOffer.SETS_ID }}"
|
||||
data-trigger="focus" tabindex="0"
|
||||
title="{% if CurOffer.SETS_RATING > 0.01 %}<b> Рейтинг {{ CurOffer.SETS_RATING|stringformat:".2f" }}</b> для оконого набора «{{ CurOffer.SETS_NAME }}» компании «{{ CurOffer.MERCHANT }}» состоит из:{% else %}Рейтинг не присвоен{% endif %}"
|
||||
data-toggle="popover">рейтинг</a>: {% for Star in CurOffer.SETS_RATING_STARTS %}{% if Star == 0 %}<b class="glyphicon glyphicon-star-empty"></b>{% else %}<b class="glyphicon glyphicon-star"></b>{% endif %}{% endfor %} {% if CurOffer.SETS_RATING > -0.1 %} {{ CurOffer.SETS_RATING|stringformat:".2f" }}{% endif %}</nobr>
|
||||
data-toggle="popover">рейтинг</a>: {% for Star in CurOffer.SETS_RATING_STARTS %}{% if Star == 0 %}<b class="glyphicon glyphicon-star-empty"></b>{% else %}<b class="glyphicon glyphicon-star"></b>{% endif %}{% endfor %} {% if CurOffer.SETS_RATING > -0.1 %}{{ CurOffer.SETS_RATING|stringformat:".2f" }}{% endif %}</nobr>
|
||||
</span>
|
||||
<span itemprop="brand" itemscope itemtype="http://schema.org/Brand">
|
||||
<meta itemprop="name" content="{{ CurOffer.MERCHANT }}" />
|
||||
<meta itemprop="logo" content="https://oknardia.ru/media/{{ CurOffer.MERCHANT_LOGO }}" />
|
||||
</span></td>
|
||||
<!--- Конец большой ячейки со спецификацией оконного предложения --->
|
||||
{% endif %}
|
||||
@@ -50,10 +48,8 @@
|
||||
|
||||
<td class="rnw" title="Стоимость {{ CurOffer.TOTAL|stringformat:".2f" }} рублей за все окна квартиры {{ APART|safe }}.">{{ CurOffer.TOTAL|stringformat:".2f"|price_format }}</td>
|
||||
<th{% if CurOffer.DISCOUNT_COLOR2 != "" %} style="background-color:{{ CurOffer.DISCOUNT_COLOR2 }};"{% endif %} title="{% if CurOffer.DISCOUNT < 0.1 %}Нет скидки{% else %}Скидка — {{ CurOffer.DISCOUNT|stringformat:".1f" }}%{% endif %}">{% if CurOffer.DISCOUNT < 0.1 %}—{% else %}−{{ CurOffer.DISCOUNT|stringformat:".1f" }}%{% endif %}</th>
|
||||
<th{% if CurOffer.DISCOUNT_COLOR1 != "" %} style="background-color:{{ CurOffer.DISCOUNT_COLOR1 }};"{% endif %} itemprop="offers" itemscope itemtype="http://schema.org/Offer" title="Итого за все окна с учетом скидки: {{ CurOffer.FIN_PRICE|stringformat:".2f" }} рублей">
|
||||
<th{% if CurOffer.DISCOUNT_COLOR1 != "" %} style="background-color:{{ CurOffer.DISCOUNT_COLOR1 }};"{% endif %} title="Итого за все окна с учетом скидки: {{ CurOffer.FIN_PRICE|stringformat:".2f" }} рублей">
|
||||
Итого: {{ CurOffer.FIN_PRICE|stringformat:".2f"|price_format }} <small class="glyphicon glyphicon-ruble" aria-label="₽ (руб.)" title="₽ (руб.)"></small>
|
||||
<meta itemprop="price" content="{{ CurOffer.FIN_PRICE }}" />
|
||||
<meta itemprop="priceCurrency" content="RUB" />
|
||||
</th>
|
||||
{% if CurOffer.DIM|length == 1 %}
|
||||
|
||||
|
||||
@@ -1,21 +1,101 @@
|
||||
{% extends "base.html" %}{% load static %}
|
||||
|
||||
{% block Title %}: Тарифы и услуги{% endblock %}
|
||||
{% block Title %}Рейтинг оконных профилей | Ранжирование PVC профилей по характеристикам | Окнардия{% endblock %}
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{# block Date4Meta %}{{ META_DATA_PUBLISH|date:"c" }}{% endblock #}
|
||||
{% block Description %}Рейтинг оконных PVC профилей в базе Окнардия: ранжирование моделей профилей по теплопередаче, звукоизоляции и другим характеристикам. Сравнение пластиковых профилей.{% endblock %}
|
||||
|
||||
{# block Last4Meta %}{{ META_DATA_PUBLISH|date:"c" }}{% endblock #}
|
||||
{% block Keywords %}рейтинг окон, рейтинг оконных профилей, рейтинг pvc профилей, производители окон, ранжирование профилей, сравнение пластиковых окон, характеристики окон, теплопередача профилей, звукоизоляция окон{% endblock %}
|
||||
|
||||
{% block Description %}Тарифы и услуги маркетплейс-агрегатора Окнардия. Размещение предложений пластиковых и деревянных окон, обновление цен на окна, рекламные баннеры и виджеты на сайт оконной компании.{% endblock %}
|
||||
{% block Author4Meta %}Рейтинг оконных профилей Окнардия{% endblock %}
|
||||
|
||||
{% block Keywords %}типовые проекты зданий, панельное строительство, {% for CountSeria in SERIA_NAV_DIM %}серия {{ CountSeria.SERIA_R }}, {{ CountSeria.SERIA_R }}, {% endfor %}, года постройки, регионы постройки, распространённость{% endblock %}
|
||||
{% block CopyrightAuthor4Meta %}Окнардия — агрегатор цен на окна{% endblock %}
|
||||
|
||||
{% block Top_JS5 %}
|
||||
<script src="{% static 'js/sortable-table.js' %}" type="text/javascript"></script>{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Метатеги для социальных сетей --> #}
|
||||
<meta name="news_keywords" content="рейтинг окон, рейтинг профилей, производители окон, pvc окна, характеристики профилей" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/stat/rating/profiles_rank/" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:app_id" content="258354027974262" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/stat/rating/profiles_rank/" />
|
||||
<meta property="og:type" content="reference" />
|
||||
<meta property="og:title" content="Рейтинг оконных профилей | Окнардия" />
|
||||
<meta property="og:description" content="Ранжирование PVC профилей по теплопередаче, звукоизоляции и другим характеристикам на основе данных участников маркетплейса Окнардия." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:title" content="Рейтинг оконных профилей | Окнардия" />
|
||||
<meta name="twitter:description" content="Сравнение и ранжирование PVC профилей по характеристикам теплопередачи и звукоизоляции." />
|
||||
<meta property="twitter:url" content="{{ request.scheme }}://{{ request.get_host }}/stat/rating/profiles_rank/" />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
{# <!-- END Метатеги для социальных сетей --> #}{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}
|
||||
{# JSON-LD: BreadcrumbList для рейтинга профилей #}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Статистика",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/stat_all/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 3,
|
||||
"name": "Рейтинг оконных профилей",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/stat/rating/profiles_rank/"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
{# JSON-LD: CollectionPage для рейтинга #}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "CollectionPage",
|
||||
"name": "Рейтинг оконных профилей",
|
||||
"description": "Ранжирование оконных профилей по теплопередаче, звукоизоляции и другим характеристикам на основе данных участников маркетплейса Окнардия.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/stat/rating/profiles_rank/",
|
||||
"isPartOf": {
|
||||
"@type": "WebSite",
|
||||
"name": "Окнардия",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
}
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Main_Content %}<div class="container-fluid">
|
||||
{# Хлебные крошки: НАЧАЛО #}
|
||||
<div class="row">
|
||||
<div class="col-md-11 col-xs-12">
|
||||
<ol class="breadcrumb">
|
||||
<li><a href="/">Главная</a></li>
|
||||
<li><a href="/stat_all/">Статистика</a></li>
|
||||
<li class="active">Рейтинг оконных профилей</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
{# Хлебные крошки: КОНЕЦ #}
|
||||
|
||||
<div class="row">
|
||||
<div class="col-md-9"><h1>Рейтинг оконных профилей базы «Окнардия»</h1></div>
|
||||
</div>
|
||||
|
||||
@@ -2,19 +2,176 @@
|
||||
{% load static %}
|
||||
{% load filters %}
|
||||
|
||||
{% block Title %} Сравнение характеристик оконных профилей: {% for PROFILE in LIST_PROFILE %}{% if forloop.first %}{% else %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ PROFILE }}{% endfor %}. Сравнение характеристик стеклопакетов: {% for GLAZING in LIST_GLAZING %}{% if forloop.first %}{% else %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ GLAZING }}{% endfor %}. Сравнение предложений окон: {% for MERCANT in LIST_MERCHANT %}{% if forloop.first %}{% else %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ MERCANT }}{% endfor %}.{% endblock %}
|
||||
{# Заголовок: человекочитаемый, ключевые слова в начале #}
|
||||
{% block Title %}Сравнение окон: {% for Count in SET_LIST %}{% if not forloop.first %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}«{{ Count.SET_NAME }}» ({{ Count.MERCHANT }}){% endfor %} — характеристики профилей и стеклопакетов{% endblock %}
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Date4Meta %}{{ META_DATA_PUBLISH|date:"c" }}{% endblock %}
|
||||
{% block Date4Meta %}{{ META_DATA_PUBLISH|date:"Y-m-d" }}{% endblock %}
|
||||
{% block Last4Meta %}{{ META_DATA_PUBLISH|date:"Y-m-d" }}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{{ META_DATA_PUBLISH|date:"c" }}{% endblock %}
|
||||
|
||||
|
||||
{% block Description %}Сравнение характеристик окон от поставщиков: {% for MERCANT in LIST_MERCHANT %}{% if forloop.first %}{% else %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ MERCANT }}{% endfor %}. Сравнение профилей пластиковых окон: {% for PROFILE in LIST_PROFILE %}{% if forloop.first %}{% else %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ PROFILE }}{% endfor %}. Сравнение характеристик стеклопакетов: {% for GLAZING in LIST_GLAZING %}{% if forloop.first %}{% else %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ GLAZING }}{% endfor %}.{% endblock %}
|
||||
{# Description: первое слово — целевой запрос, потом конкретика #}
|
||||
{% block Description %}Детальное сравнение оконных наборов: {% for Count in SET_LIST %}{% if not forloop.first %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}«{{ Count.SET_NAME }}» от {{ Count.MERCHANT }}{% endfor %}. Профили: {% for PROFILE in LIST_PROFILE %}{{ PROFILE }}{% if not forloop.last %}, {% endif %}{% endfor %}. Стеклопакеты: {% for GLAZING in LIST_GLAZING %}{{ GLAZING }}{% if not forloop.last %}, {% endif %}{% endfor %}. Теплопередача, звукоизоляция, условия монтажа.{% endblock %}
|
||||
|
||||
{% block Keywords %}сравнение профилей пластиковых окон, {% for PROFILE in LIST_PROFILE %}{{ PROFILE }}, {% endfor %}сравнение стеклопакетов, {% for GLAZING in LIST_GLAZING %}{{ GLAZING }}, {% endfor %}сравнение поставщиков пластиковых окон, {% for MERCANT in LIST_MERCHANT %}{{ MERCANT }}, {% endfor %}характеристики пластиковых окон.{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}
|
||||
{# Canonical — предотвращает дубли при разном порядке ID (1,2 и 2,1 — одна страница) #}
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/compare_offers/{% for Count in SET_LIST %}{{ Count.SET_ID }}{% if not forloop.last %},{% endif %}{% endfor %}/" />
|
||||
<!-- Open Graph -->
|
||||
<meta property="og:type" content="article" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/compare_offers/{% for Count in SET_LIST %}{{ Count.SET_ID }}{% if not forloop.last %},{% endif %}{% endfor %}/" />
|
||||
<meta property="og:title" content="Сравнение окон: {% for Count in SET_LIST %}{% if not forloop.first %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}«{{ Count.SET_NAME }}» ({{ Count.MERCHANT|escapejs }}){% endfor %} | oknardia.ru" />
|
||||
<meta property="og:description" content="Сравнение характеристик оконных профилей и стеклопакетов: {% for Count in SET_LIST %}{% if not forloop.first %}, {% endif %}{{ Count.SET_NAME }} от {{ Count.MERCHANT }}{% endfor %}. Теплопередача, звукоизоляция, условия монтажа — агрегатор Окнардия." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<!-- Twitter Card -->
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:title" content="Сравнение окон: {% for Count in SET_LIST %}{% if not forloop.first %}, {% endif %}{{ Count.SET_NAME }}{% endfor %} | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Детальная таблица сравнения характеристик оконных профилей и стеклопакетов от поставщиков: {% for MERCANT in LIST_MERCHANT %}{{ MERCANT }}{% if not forloop.last %}, {% endif %}{% endfor %}." />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}{% comment %}
|
||||
JSON-LD разметка для страницы сравнения оконных наборов.
|
||||
BreadcrumbList: хлебные крошки в сниппете Google.
|
||||
TechArticle + mainEntity ItemList: описывает страницу как технический сравнительный материал.
|
||||
Каждый набор — Product с:
|
||||
- additionalProperty: полный список условий поставки и монтажа
|
||||
- hasPart[0]: профиль ПВХ как вложенный Product со всеми техническими PropertyValue
|
||||
- hasPart[1]: стеклопакет как вложенный Product со всеми техническими PropertyValue
|
||||
Хак с запятыми: последним в каждом additionalProperty ставим фиксированный элемент
|
||||
{"@type":"PropertyValue","name":"Источник данных","value":"oknardia.ru"} — благодаря этому
|
||||
все условные элементы выше могут безопасно завершаться запятой.
|
||||
{% endcomment %}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@graph": [
|
||||
{
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{"@type": "ListItem", "position": 1, "name": "Главная", "item": "{{ request.scheme }}://{{ request.get_host }}/"},
|
||||
{"@type": "ListItem", "position": 2, "name": "Каталог", "item": "{{ request.scheme }}://{{ request.get_host }}/catalog/"},
|
||||
{"@type": "ListItem", "position": 3, "name": "Оконные наборы", "item": "{{ request.scheme }}://{{ request.get_host }}/catalog/sets/"},
|
||||
{"@type": "ListItem", "position": 4,
|
||||
"name": "Сравнение: {% for Count in SET_LIST %}{% if not forloop.first %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ Count.SET_NAME|escapejs }}{% endfor %}",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/compare_offers/{% for Count in SET_LIST %}{{ Count.SET_ID }}{% if not forloop.last %},{% endif %}{% endfor %}/"}
|
||||
]
|
||||
},
|
||||
{
|
||||
"@type": "TechArticle",
|
||||
"inLanguage": "ru-RU",
|
||||
"headline": "Сравнение оконных наборов: {% for Count in SET_LIST %}{% if not forloop.first %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ Count.SET_NAME|escapejs }} ({{ Count.MERCHANT|escapejs }}){% endfor %}",
|
||||
"description": "Детальная техническая таблица сравнения оконных профилей ({% for PROFILE in LIST_PROFILE %}{{ PROFILE|escapejs }}{% if not forloop.last %}, {% endif %}{% endfor %}) и стеклопакетов ({% for GLAZING in LIST_GLAZING %}{{ GLAZING|escapejs }}{% if not forloop.last %}, {% endif %}{% endfor %}) от поставщиков «Окнардии» — характеристики теплопередачи, звукоизоляции, условия монтажа.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/compare_offers/{% for Count in SET_LIST %}{{ Count.SET_ID }}{% if not forloop.last %},{% endif %}{% endfor %}/",
|
||||
"image": "{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg",
|
||||
"publisher": {
|
||||
"@type": "Organization",
|
||||
"name": "Окнардия",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}",
|
||||
"logo": {"@type": "ImageObject", "url": "{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg"}
|
||||
},
|
||||
"isPartOf": {
|
||||
"@type": "WebSite",
|
||||
"name": "Окнардия — агрегатор цен на замену окон",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}"
|
||||
},
|
||||
"mainEntity": {
|
||||
"@type": "ItemList",
|
||||
"name": "Сравниваемые оконные наборы",
|
||||
"numberOfItems": {{ SET_LIST|length }},
|
||||
"itemListElement": [{% for Count in SET_LIST %}
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": {{ forloop.counter }},
|
||||
"item": {
|
||||
"@type": "Product",
|
||||
"@id": "{{ request.scheme }}://{{ request.get_host }}/compare_offers/{% for C in SET_LIST %}{{ C.SET_ID }}{% if not forloop.last %},{% endif %}{% endfor %}/#set-{{ Count.SET_ID }}",
|
||||
"name": "{{ Count.SET_NAME|escapejs }}",
|
||||
"brand": {"@type": "Brand", "name": "{{ Count.MERCHANT|escapejs }}"},
|
||||
"description": "Оконный набор с профилем {{ Count.PROFILE_NAME|escapejs }} ({{ Count.PROFILE_MANUFACTURER|escapejs }}) и стеклопакетом {{ Count.GLAZING_MARK|escapejs }}.{% if Count.PROFILE_HEAT_TRANSFER > 0.1 %} Сопротивление теплопередаче профиля {{ Count.PROFILE_HEAT_TRANSFER|stringformat:".2f" }} м²·°C/Вт.{% endif %}{% if Count.GLAZING_HEAT_TRANSFER > 0.1 %} Теплопередача стеклопакета {{ Count.GLAZING_HEAT_TRANSFER|stringformat:".2f" }} м²·°C/Вт.{% endif %}",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/company/{{ Count.MERCHANT_ID }}-{{ Count.MERCHANT_T }}/",
|
||||
"image": "http://oknardia.ru/media/{{ Count.MERCHANT_LOGO }}",
|
||||
{% if Count.RATING_SET_N > 0.1 %}
|
||||
"aggregateRating": {
|
||||
"@type": "AggregateRating",
|
||||
"ratingValue": "{{ Count.RATING_SET_N|stringformat:".2f" }}",
|
||||
"bestRating": "5", "worstRating": "0", "reviewCount": "1"
|
||||
},
|
||||
{% endif %}
|
||||
"additionalProperty": [
|
||||
{% if Count.SET_IMPLEMENTS_ALL %}{"@type": "PropertyValue", "name": "Фурнитура", "value": "{{ Count.SET_IMPLEMENTS_ALL|escapejs }}"},{% endif %}
|
||||
{% if Count.SET_IMPLEMENTS_HANDLES %}{"@type": "PropertyValue", "name": "Ручки", "value": "{{ Count.SET_IMPLEMENTS_HANDLES|escapejs }}"},{% endif %}
|
||||
{% if Count.SET_IMPLEMENTS_HINGES %}{"@type": "PropertyValue", "name": "Петли", "value": "{{ Count.SET_IMPLEMENTS_HINGES|escapejs }}"},{% endif %}
|
||||
{% if Count.SET_IMPLEMENTS_LATCH %}{"@type": "PropertyValue", "name": "Механизмы запирания", "value": "{{ Count.SET_IMPLEMENTS_LATCH|escapejs }}"},{% endif %}
|
||||
{% if Count.SET_IMPLEMENTS_LIMITER %}{"@type": "PropertyValue", "name": "Ограничители открывания", "value": "{{ Count.SET_IMPLEMENTS_LIMITER|escapejs }}"},{% endif %}
|
||||
{% if Count.SET_IMPLEMENTS_CATCH %}{"@type": "PropertyValue", "name": "Фиксаторы открывания", "value": "{{ Count.SET_IMPLEMENTS_CATCH|escapejs }}"},{% endif %}
|
||||
{% if Count.SET_CLIMATE_CONTROL|length > 3 %}{"@type": "PropertyValue", "name": "Климат-контроль", "value": "{{ Count.SET_CLIMATE_CONTROL|escapejs }}"},{% endif %}
|
||||
{"@type": "PropertyValue", "name": "Подоконники", "value": "{{ Count.SET_STILL|escapejs }}"},
|
||||
{"@type": "PropertyValue", "name": "Водоотливы", "value": "{{ Count.SET_PANES|escapejs }}"},
|
||||
{"@type": "PropertyValue", "name": "Откосы", "value": "{{ Count.SET_SLOPE|escapejs }}"},
|
||||
{"@type": "PropertyValue", "name": "Доставка", "value": "{{ Count.SET_DELIVERY|escapejs }}"},
|
||||
{"@type": "PropertyValue", "name": "Демонтаж и монтаж", "value": "{{ Count.SET_UNINSTALL_INSTALL|escapejs }}"},
|
||||
{% if Count.SET_OTHER_CONDITIONS %}{"@type": "PropertyValue", "name": "Прочие условия", "value": "{{ Count.SET_OTHER_CONDITIONS|escapejs }}"},{% endif %}
|
||||
{"@type": "PropertyValue", "name": "Источник данных", "value": "oknardia.ru"}
|
||||
],
|
||||
"hasPart": [
|
||||
{
|
||||
"@type": "Product",
|
||||
"@id": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ Count.PROFILE_ID }}-{{ Count.PROFILE_MANUFACTURER_T }}/{{ Count.PROFILE_ID }}-{{ Count.PROFILE_NAME_T }}/",
|
||||
"name": "{{ Count.PROFILE_NAME|escapejs }}",
|
||||
"description": "ПВХ-профиль {{ Count.PROFILE_MANUFACTURER|escapejs }} {{ Count.PROFILE_NAME|escapejs }}",
|
||||
"brand": {"@type": "Brand", "name": "{{ Count.PROFILE_MANUFACTURER|escapejs }}"},
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/profile/{{ Count.PROFILE_ID }}-{{ Count.PROFILE_MANUFACTURER_T }}/{{ Count.PROFILE_ID }}-{{ Count.PROFILE_NAME_T }}/",
|
||||
"additionalProperty": [
|
||||
{% if Count.PROFILE_NUM_COLOR %}{"@type": "PropertyValue", "name": "Цвет профиля", "value": "{{ Count.PROFILE_NUM_COLOR|escapejs }}"},{% endif %}
|
||||
{% if Count.PROFILE_NUM_CAMERAS and Count.PROFILE_NUM_CAMERAS != "—" %}{"@type": "PropertyValue", "name": "Число камер рамы/створки", "unitText": "шт.", "value": "{{ Count.PROFILE_NUM_CAMERAS }}"},{% endif %}
|
||||
{% if Count.PROFILE_NUM_SEALS > 0 %}{"@type": "PropertyValue", "name": "Контуры уплотнения", "unitText": "шт.", "value": {{ Count.PROFILE_NUM_SEALS }}},{% endif %}
|
||||
{% if Count.PROFILE_THICKNESS > 5 %}{"@type": "PropertyValue", "name": "Монтажная ширина профиля", "unitCode": "MMT", "unitText": "мм", "value": {{ Count.PROFILE_THICKNESS }}},{% endif %}
|
||||
{% if Count.PROFILE_GLAZING_THICKNESS > 4 %}{"@type": "PropertyValue", "name": "Макс. толщина стеклопакета", "unitCode": "MMT", "unitText": "мм", "value": {{ Count.PROFILE_GLAZING_THICKNESS }}},{% endif %}
|
||||
{% if Count.PROFILE_HEAT_TRANSFER > 0.1 %}{"@type": "PropertyValue", "name": "Сопротивление теплопередаче (Ro)", "unitText": "м²·°C/Вт", "value": {{ Count.PROFILE_HEAT_TRANSFER|stringformat:".2f" }}},{% endif %}
|
||||
{% if Count.PROFILE_SOUND_PROOFING > 1 %}{"@type": "PropertyValue", "name": "Коэффициент звукоизоляции", "unitText": "дБ", "value": {{ Count.PROFILE_SOUND_PROOFING|stringformat:".1f" }}},{% endif %}
|
||||
{% if Count.PROFILE_HEIGHT > 15 %}{"@type": "PropertyValue", "name": "Высота в световом проеме (рама+створка)", "unitCode": "MMT", "unitText": "мм", "value": {{ Count.PROFILE_HEIGHT }}},{% endif %}
|
||||
{% if Count.PROFILE_RABBET > 1 %}{"@type": "PropertyValue", "name": "Фальц рамы", "unitCode": "MMT", "unitText": "мм", "value": {{ Count.PROFILE_RABBET }}},{% endif %}
|
||||
{% if Count.PROFILE_REINFORCEMENT %}{"@type": "PropertyValue", "name": "Армирование профиля", "value": "{{ Count.PROFILE_REINFORCEMENT|escapejs }}"},{% endif %}
|
||||
{% if Count.PROFILE_FILLET %}{"@type": "PropertyValue", "name": "Штапик", "value": "{{ Count.PROFILE_FILLET|escapejs }}"},{% endif %}
|
||||
{% if Count.PROFILE_SEAL_DESCRIPTION %}{"@type": "PropertyValue", "name": "Уплотнитель", "value": "{{ Count.PROFILE_SEAL_DESCRIPTION|escapejs }}"},{% endif %}
|
||||
{% if Count.PROFILE_OTHER %}{"@type": "PropertyValue", "name": "Прочие характеристики профиля", "value": "{{ Count.PROFILE_OTHER|escapejs }}"},{% endif %}
|
||||
{"@type": "PropertyValue", "name": "Источник данных", "value": "oknardia.ru"}
|
||||
]
|
||||
},
|
||||
{
|
||||
"@type": "Product",
|
||||
"name": "{{ Count.GLAZING_MARK|escapejs }}",
|
||||
"description": "{% if Count.GLAZING_BRIEF_DESCRIPTION %}{{ Count.GLAZING_BRIEF_DESCRIPTION|escapejs }}{% else %}Стеклопакет {{ Count.GLAZING_MARK|escapejs }}{% endif %}",
|
||||
{% if Count.GLAZING_MANUFACTURER and Count.GLAZING_MANUFACTURER != "—//—" %}"brand": {"@type": "Brand", "name": "{{ Count.GLAZING_MANUFACTURER|escapejs }}"},{% endif %}
|
||||
"additionalProperty": [
|
||||
{% if Count.GLAZING_CAMERAS_NUM >= 1 %}{"@type": "PropertyValue", "name": "Камер в стеклопакете", "unitText": "шт.", "value": {{ Count.GLAZING_CAMERAS_NUM }}},{% endif %}
|
||||
{% if Count.GLAZING_THICKNESS >= 3 %}{"@type": "PropertyValue", "name": "Толщина стеклопакета", "unitCode": "MMT", "unitText": "мм", "value": {{ Count.GLAZING_THICKNESS }}},{% endif %}
|
||||
{% if Count.GLAZING_HEAT_TRANSFER > 0.1 %}{"@type": "PropertyValue", "name": "Сопротивление теплопередаче (Ro)", "unitText": "м²·°C/Вт", "value": {{ Count.GLAZING_HEAT_TRANSFER|stringformat:".2f" }}},{% endif %}
|
||||
{% if Count.GLAZING_SOUNDPROOFING >= 10 %}{"@type": "PropertyValue", "name": "Коэффициент звукоизоляции", "unitText": "дБ", "value": {{ Count.GLAZING_SOUNDPROOFING|stringformat:".1f" }}},{% endif %}
|
||||
{% if Count.GLAZING_LIGHT_TRANSMISSION >= 1 %}{"@type": "PropertyValue", "name": "Коэффициент светопропускания", "unitCode": "P1", "unitText": "%", "value": {{ Count.GLAZING_LIGHT_TRANSMISSION|stringformat:".0f" }}},{% endif %}
|
||||
{% if Count.GLAZING_PASSING_SUN >= 1 %}{"@type": "PropertyValue", "name": "Коэффициент солнцепропускания", "unitCode": "P1", "unitText": "%", "value": {{ Count.GLAZING_PASSING_SUN|stringformat:".0f" }}},{% endif %}
|
||||
{% if Count.GLAZING_LIGHT_REFLECTION and Count.GLAZING_LIGHT_REFLECTION != "—/—" %}{"@type": "PropertyValue", "name": "Коэффициент светоотражения (внешний/внутренний)", "unitText": "%", "value": "{{ Count.GLAZING_LIGHT_REFLECTION|escapejs }}"},{% endif %}
|
||||
{% if Count.GLAZING_REFLECTION_AND_ABSORPTION and Count.GLAZING_REFLECTION_AND_ABSORPTION != "—/—" %}{"@type": "PropertyValue", "name": "Теплоотражение/теплопоглощение", "unitText": "%", "value": "{{ Count.GLAZING_REFLECTION_AND_ABSORPTION|escapejs }}"},{% endif %}
|
||||
{"@type": "PropertyValue", "name": "Тонирование", "value": "{{ Count.GLAZING_TONING|escapejs }}"},
|
||||
{"@type": "PropertyValue", "name": "Источник данных", "value": "oknardia.ru"}
|
||||
]
|
||||
}
|
||||
]
|
||||
}
|
||||
}{% if not forloop.last %},{% endif %}{% endfor %}
|
||||
]
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Top_JS3%}<script type="text/javascript">
|
||||
$(function () {
|
||||
$('[data-toggle="popover"]').popover({
|
||||
@@ -36,23 +193,31 @@
|
||||
})
|
||||
</script>{% endblock %}
|
||||
|
||||
|
||||
{% block Top_CSS1 %}{% endblock %}
|
||||
|
||||
{% block Main_Content %}<!--- ------------------------------------------------------------------------------------------------------------------------- --->
|
||||
<div class="row col-xs-12">
|
||||
<div class="col-md-9 col-xs-8">
|
||||
<h1>Сравнении оконных наборов:{% for Count in SET_LIST %}{% if forloop.first %} {% else %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ Count.SET_NAME }}{% if forloop.last %}.{% endif %}{% endfor %}</h1>
|
||||
{# Хлебные крошки: Главная → Каталог → Оконные наборы (ссылка) → текущее сравнение #}
|
||||
<ol class="breadcrumb">
|
||||
<li><a href="/">Главная</a></li>
|
||||
<li><a href="/catalog/">Каталог</a></li>
|
||||
<li><a href="/catalog/sets/">Оконные наборы</a></li>
|
||||
<li>Сравнение:{% for Count in SET_LIST %}{% if forloop.first %} {% else %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}«{{ Count.SET_NAME }}»{% endfor %}</li>
|
||||
</ol>
|
||||
{# Исправлена опечатка: «Сравнении» → «Сравнение» #}
|
||||
<h1>Сравнение оконных наборов:{% for Count in SET_LIST %}{% if forloop.first %} {% else %}{% if forloop.last %} и {% else %}, {% endif %}{% endif %}{{ Count.SET_NAME }}{% if forloop.last %}.{% endif %}{% endfor %}</h1>
|
||||
<p>Оконный набор — это комплект оконного профиля (рамы и сворки), стеклопакета, фурнитуры и уплотнителей — готовое окно в сборе для установки в проём. В набор может входить отлив, подоконник, откос, встраиваемые системы <nobr>климат-контроля</nobr>, оконная фурнитура открывания, запоры, уплотнители, москитная сетка… а также сопутствующие услуги: демонтаж старых и установка новых окон, доставка, гарантийное обслуживание, уборка, вынос и утилизация строительного мусора, защитное укрытие мебели на время монтажа и другое.</p>
|
||||
<h4>В таблицу сравнения услуг и условий, входящих в оконные наборы, добавлены следующие производители и поставщики:</h4>
|
||||
{# h4 → h2 для правильной иерархии заголовков (SEO); визуальный размер сохраняем через style #}
|
||||
<h2 style="font-size:1em;font-weight:bold;margin-top:1em;">В таблицу сравнения услуг и условий, входящих в оконные наборы, добавлены следующие производители и поставщики:</h2>
|
||||
<ul>{% for MERCANT in LIST_MERCHANT %}
|
||||
<li>{{ MERCANT }}.</li>{% endfor %}
|
||||
</ul>
|
||||
<h4>В средней части таблице вы сможете сравнить характеристики оконных профилей в наборах:</h4>
|
||||
<h2 style="font-size:1em;font-weight:bold;margin-top:1em;">В средней части таблицы вы сможете сравнить характеристики оконных профилей в наборах:</h2>
|
||||
<ul>{% for PROFILE in LIST_PROFILE %}
|
||||
<li>{{ PROFILE }}.</li>{% endfor %}
|
||||
</ul>
|
||||
<h4>Нижний блок таблицы посвящен характеристикам и сравнению стеклопакетов в наборах. Формулы выбранных стеклопакетов:</h4>
|
||||
<h2 style="font-size:1em;font-weight:bold;margin-top:1em;">Нижний блок таблицы посвящен характеристикам и сравнению стеклопакетов в наборах. Формулы выбранных стеклопакетов:</h2>
|
||||
<ul>{% for GLAZING in LIST_GLAZING %}
|
||||
<li>{{ GLAZING }}.</li>{% endfor %}
|
||||
</ul>
|
||||
@@ -66,14 +231,18 @@
|
||||
<thead>
|
||||
<tr style="background:white;">
|
||||
<th class="col-sm-2">Поставщик:<br /><small style="font-size:small;font-weight:100;">компания, предлагающая установку окон</small></th>{% for Count in SET_LIST %}
|
||||
<th class="col-xs-1" title="Установку окон предлагает компания «{{ Count.MERCHANT }}»"><h2 style="font-size:1em;margin:0;font-weight:bold;">{{ Count.MERCHANT }}</h2><br />
|
||||
<th class="col-xs-1" title="Установку окон предлагает компания «{{ Count.MERCHANT }}»">
|
||||
{# h2 в ячейке таблицы — семантический мусор; заменяем на strong #}
|
||||
<strong>{{ Count.MERCHANT }}</strong><br />
|
||||
<img src="http://oknardia.ru/media/{{ Count.MERCHANT_LOGO }}" style="height: 25px;width:auto;" alt="{{ Count.MERCHANT }}"><br />
|
||||
<div class="url"><nobr>{% if Count.IS_COMMERCIAL %}<a href="{{ Count.MERCHANT_URL }}" target="_blank" rel="nofollow">{{ Count.MERCHANT_URL_SHOT|truncatechars:30 }}</a>{% else %}{{ Count.MERCHANT_URL_SHOT|truncatechars:30 }}{% endif %}</nobr><br /><a href="/catalog/company/{{ Count.MERCHANT_ID }}-{{ Count.MERCHANT_T }}/">в каталоге</a></div>
|
||||
</th>{% endfor %}
|
||||
</tr><tr>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Название набора:</th>{% for Count in SET_LIST %}
|
||||
<th title="Название оконного набора: «{{ Count.SET_NAME }}»">{{ Count.SET_NAME|truncatechars:25 }}</th>{% endfor %}
|
||||
</tr><tr class="rating">
|
||||
</tr>
|
||||
<tr class="rating">
|
||||
<th><nobr>Рейтиг «Окнардии»:</nobr></th>{% for Count in SET_LIST %}
|
||||
<td{% if Count.RATING_SET_COLOR != "" %} style="background:{{ Count.RATING_SET_COLOR }};"{% endif %}>
|
||||
<nobr title="Рейтинг «Окнардии» для окон набора «{{ Count.SET_NAME }}» компании «{{ Count.MERCHANT }}» — {% if Count.RATING_SET_N > 0.1 %}{{ Count.RATING_SET_N|stringformat:".2f" }}{% else %}не присвоен{% endif %}"><!-- НАЧАЛО звездочки рейтинга -->{% for Star in Count.RATING_SET %}{% if Star == 0 %}<i class="glyphicon glyphicon-star-empty"></i>{% else %}<b class="glyphicon glyphicon-star"></b>{% endif %}{% endfor %}<!-- КОНЕЦ звездочки рейтинга НАЧАЛО бедж --> {% if Count.RATING_SET_N > 0.1 %}<tt class="badge">{{ Count.RATING_SET_N|stringformat:".2f" }}</tt>{% endif %}<!-- КОНЕЦ бедж --> <a
|
||||
@@ -174,7 +343,7 @@
|
||||
<td{% if Count.PROFILE_FILLET != "" %} title="Характеристики штапика: {{ Count.PROFILE_FILLET }}"{% endif %}>{% if Count.PROFILE_FILLET != "" %}{{ Count.PROFILE_FILLET }}{% else %}—{% endif %}</td>{% endfor %}
|
||||
</tr><tr>
|
||||
<th>Уплотнитель:</th>{% for Count in SET_LIST %}
|
||||
<td{% if Count.PROFILE_SEAL_DESCRIPTION != "" %} title="Хараектеристики уплотнитель стеклопакета и контуров рама-створка: {{ Count.PROFILE_SEAL_DESCRIPTION|lower }}"{% endif %}>{% if Count.PROFILE_SEAL_DESCRIPTION != "" %}{{ Count.PROFILE_SEAL_DESCRIPTION|capfirst }}{% else %}—{% endif %}</td>{% endfor %}
|
||||
<td{% if Count.PROFILE_SEAL_DESCRIPTION != "" %} title="Характеристики уплотнитель стеклопакета и контуров рама-створка: {{ Count.PROFILE_SEAL_DESCRIPTION|lower }}"{% endif %}>{% if Count.PROFILE_SEAL_DESCRIPTION != "" %}{{ Count.PROFILE_SEAL_DESCRIPTION|capfirst }}{% else %}—{% endif %}</td>{% endfor %}
|
||||
</tr><tr>
|
||||
<th>Прочие характеристики:</th>{% for Count in SET_LIST %}
|
||||
<td{% if Count.PROFILE_OTHER != "" %} title="Прочие характеристики рамы и створки: {{ Count.PROFILE_OTHER }}"{% endif %}><small>{% if Count.PROFILE_OTHER != "" %}{{ Count.PROFILE_OTHER }}{% else %}—{% endif %}</small></td>{% endfor %}
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
<!--- Информация об адресах просмотренных текущим пользователем --->{% load filters %}
|
||||
{% if LAST_VISIT and LAST_VISIT|length >= 1 %}<div class="col-xs-12">
|
||||
<div class="col-md-11 col-xs-12 last_user_visit"><h5>Цены на окна просмотренные вами:</h5>
|
||||
<ul>{% for ITEM in LAST_VISIT %}
|
||||
<li><a href="{{ ITEM.LastURL }}">Цены на окна для серии {{ ITEM.LastApart }} <small>({{ ITEM.LastAddress }})</small></a> <small>{{ ITEM.Time }}</small></li>{% endfor %}
|
||||
</ul>
|
||||
{% load static %}<!-- Информация об адресах, просмотренных текущим пользователем (читается из браузерных кук) -->
|
||||
<div class="col-xs-12">
|
||||
<div class="col-md-11 col-xs-12{% if background_color != "None" %} last_user_visit{% endif %}" id="last_user_visit_container" style="display:none;">
|
||||
<h5>Цены на окна просмотренные вами:</h5>
|
||||
<ul id="last_visits_list"></ul>
|
||||
</div>
|
||||
</div>{% endif %}
|
||||
</div>
|
||||
<script type="text/javascript" src="{% static 'js/last_user_visit.js' %}"></script>
|
||||
|
||||
@@ -1,13 +1,13 @@
|
||||
{# Отрисовка больших картинок с проемами и схамаи открывания #}{% load static %}{% if WIN_DIM %}
|
||||
{# Отрисовка больших картинок с проемами и схемами открывания #}{% load static %}{% if WIN_DIM %}
|
||||
{% for I_WIN_DIM in FLAP_DIM %}
|
||||
<div class="win_discr pull-left" id="flap{{ forloop.counter0 }}">
|
||||
<div><img src="{% static I_WIN_DIM.url2img %}" alt="{{ I_WIN_DIM.sDescription }}. Размер {{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0x{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0 (Ш х В, мм.). Типовая схема открывания." title="{{ I_WIN_DIM.sDescription }}. Размер {{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0x{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0 (Ш х В, мм.). Типовая схема открывания." itemprop="image" /></div>
|
||||
<div class="caption" style="width:{{ I_WIN_DIM.W }}px;min-width:13ex;">
|
||||
<nobr>{{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0×{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0 мм.</nobr><br />{% if not I_WIN_DIM.iQuantity == 0 %}
|
||||
<nobr><b>{{ I_WIN_DIM.iQuantity }} шт.</b>{% for I_II in I_WIN_DIM.qStr %}<span class="color-bullet" style="background-image:url('{% static 'img/svg/mark' %}{{ I_II }}.svg');"></span>{% endfor %}</nobr><br />{% endif %}
|
||||
{{ I_WIN_DIM.sDescription }}{% if not I_WIN_DIM.iQuantity == 0 %}<br />
|
||||
<a href="/tsena-odnogo-okna/{{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0x{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0mm/tip{{ I_WIN_DIM.id }}">цены только этого типового окна</a>{% endif %}
|
||||
</div>
|
||||
<div class="win_discr pull-left" id="flap{{ forloop.counter0 }}">
|
||||
<div><img src="{% static I_WIN_DIM.url2img %}" alt="{{ I_WIN_DIM.sDescription }}. Размер {{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0x{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0 (Ш х В, мм.). Типовая схема открывания." title="{{ I_WIN_DIM.sDescription }}. Размер {{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0x{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0 (Ш х В, мм.). Типовая схема открывания." itemprop="image" /></div>
|
||||
<div class="caption" style="width:{{ I_WIN_DIM.W }}px;min-width:13ex;">
|
||||
<nobr>{{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0×{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0 мм.</nobr><br />{% if not I_WIN_DIM.iQuantity == 0 %}
|
||||
<nobr><b>{{ I_WIN_DIM.iQuantity }} шт.</b>{% for I_II in I_WIN_DIM.qStr %}<span class="color-bullet" style="background-image:url('{% static 'img/svg/mark' %}{{ I_II }}.svg');"></span>{% endfor %}</nobr><br />{% endif %}
|
||||
{{ I_WIN_DIM.sDescription }}{% if not I_WIN_DIM.iQuantity == 0 %}<br />
|
||||
<a href="/catalog/standard_opening/price-{{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0x{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0mm-tip{{ I_WIN_DIM.id }}">цены только этого типового окна</a>{% endif %}
|
||||
</div>
|
||||
</div>{% endfor %}{% comment %}
|
||||
<script type="text/javascript">
|
||||
$(document).ready(function(){
|
||||
|
||||
@@ -7,9 +7,107 @@
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Description %}Дома серии {{ THIS_SERIA_NAME }}: детальная информация{% endblock %}
|
||||
{% block Description %}Серия {{ THIS_SERIA_NAME }}: типовые размеры оконных проёмов, схемы открывания, планировки квартир, карта зданий, статистика и цены на замену окон в домах этой серии.{% endblock %}
|
||||
|
||||
{% block Keywords %} {{ THIS_SERIA_NAME }}, серия {{ THIS_SERIA_NAME }}, проект {{ THIS_SERIA_NAME }}, года постройки зданий серии {{ THIS_SERIA_NAME }}, размеры окон в домах серии {{ THIS_SERIA_NAME }}, оконные проемы зданий серии {{ THIS_SERIA_NAME }}, дома серии {{ THIS_SERIA_NAME }} на карте, установка окон, цены на пластиковые окна{% endblock %}
|
||||
{% block Keywords %}{{ THIS_SERIA_NAME }}, серия {{ THIS_SERIA_NAME }}, проект {{ THIS_SERIA_NAME }}, года постройки зданий серии {{ THIS_SERIA_NAME }}, размеры окон в домах серии {{ THIS_SERIA_NAME }}, оконные проемы зданий серии {{ THIS_SERIA_NAME }}, дома серии {{ THIS_SERIA_NAME }} на карте, установка окон, цены на пластиковые окна{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}
|
||||
{# Canonical — предотвращает дубли при возможных GET-параметрах #}<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/catalog/seria/{{ THIS_SERIA_NAME_T }}/all{{ THIS_SERIA_ID }}/" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
<meta property="fb:app_id" content="258354027974262" />
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
<meta property="og:type" content="article" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/seria/{{ THIS_SERIA_NAME_T }}/all{{ THIS_SERIA_ID }}/" />
|
||||
<meta property="og:title" content="Серия {{ THIS_SERIA_NAME }}: размеры окон, планировки и карта домов | oknardia.ru" />
|
||||
<meta property="og:description" content="Типовые размеры и схемы открывания оконных проёмов в домах серии {{ THIS_SERIA_NAME }}. Карта зданий, статистика ввода в эксплуатацию по годам и цены на замену окон." />{% if THIS_SERIA_IMAGE_URL and THIS_SERIA_IMAGE_URL != "null.gif" %}
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/media/{{ THIS_SERIA_IMAGE_URL }}" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/media/{{ THIS_SERIA_IMAGE_URL }}" />{% else %}
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />{% endif %}
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta name="twitter:title" content="Серия {{ THIS_SERIA_NAME }}: размеры окон, планировки и карта домов | oknardia.ru" />
|
||||
<meta name="twitter:description" content="Типовые оконные проёмы, схемы открывания и цены на замену окон в домах серии {{ THIS_SERIA_NAME }}." />{% if THIS_SERIA_IMAGE_URL and THIS_SERIA_IMAGE_URL != "null.gif" %}
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/media/{{ THIS_SERIA_IMAGE_URL }}" />{% else %}
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />{% endif %}
|
||||
<meta property="twitter:url" content="{{ request.scheme }}://{{ request.get_host }}/catalog/seria/{{ THIS_SERIA_NAME_T }}/all{{ THIS_SERIA_ID }}/" />
|
||||
{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}{% comment %}
|
||||
JSON-LD для страницы серии типового строительства.
|
||||
BreadcrumbList: Google показывает хлебные крошки в сниппете вместо сырого URL — это важно,
|
||||
т.к. URL вида /catalog/seria/p-44/all7 выглядит некрасиво без расшифровки.
|
||||
TechArticle: описывает страницу как технический справочный материал по серии домов.
|
||||
{% endcomment %}
|
||||
{# JSON-LD: BreadcrumbList #}<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Каталог",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 3,
|
||||
"name": "Типовые серии домов",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/seria/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 4,
|
||||
"name": "Серия {{ THIS_SERIA_NAME|escapejs }}",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/catalog/seria/{{ THIS_SERIA_NAME_T }}/all{{ THIS_SERIA_ID }}/"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
{# JSON-LD: TechArticle — технический справочный материал о серии типового строительства #}<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "TechArticle",
|
||||
"inLanguage": "ru-RU",
|
||||
"headline": "Серия {{ THIS_SERIA_NAME|escapejs }}: типовые размеры оконных проёмов, схемы открывания и карта зданий",
|
||||
"description": "Технические характеристики оконных проёмов в домах типовой серии {{ THIS_SERIA_NAME|escapejs }}: размеры, схемы открывания, планировки квартир, карта зданий на территории России. Статистика ввода зданий в эксплуатацию по годам.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/catalog/seria/{{ THIS_SERIA_NAME_T }}/all{{ THIS_SERIA_ID }}/", {% if THIS_SERIA_IMAGE_URL and THIS_SERIA_IMAGE_URL != "null.gif" %}
|
||||
"image": "{{ request.scheme }}://{{ request.get_host }}/media/{{ THIS_SERIA_IMAGE_URL }}",{% else %}
|
||||
"image": "{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg",{% endif %}
|
||||
"about": {
|
||||
"@type": "Thing",
|
||||
"name": "Серия типового строительства {{ THIS_SERIA_NAME|escapejs }}",
|
||||
"description": "Типовой проект жилых зданий серии {{ THIS_SERIA_NAME|escapejs }} в России"
|
||||
},
|
||||
"publisher": {
|
||||
"@type": "Organization",
|
||||
"name": "Окнардия",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}",
|
||||
"logo": {
|
||||
"@type": "ImageObject",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg"
|
||||
}
|
||||
},
|
||||
"isPartOf": {
|
||||
"@type": "WebSite",
|
||||
"name": "Окнардия — агрегатор цен на замену окон",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}"
|
||||
}
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Top_JS1 %}
|
||||
<script type="text/javascript">
|
||||
@@ -34,44 +132,47 @@
|
||||
</div>
|
||||
</div>{# <!--- Хлебные крошки: КОНЕЦ ---> #}
|
||||
<div class="row">
|
||||
<div class="col-md-12">{{ THIS_SERIA_DESCRIPTION|safe }}</div>
|
||||
<div class="col-md-12">
|
||||
{# ВЕРХНЯЯ СТАТЬЯ: рендерится динамически (из БД), можно редактировать через админку #}
|
||||
<div>{{ THIS_SERIA_DESCRIPTION|safe }}</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="row">
|
||||
<div class="col-lg-10">
|
||||
<h2 class="header">Дома серии {{ THIS_SERIA_NAME }}: типовые размеры и схемы открывания</h2>
|
||||
<div class="col-lg-10">
|
||||
<h2 class="header">Дома серии {{ THIS_SERIA_NAME }}: типовые размеры и схемы открывания</h2>
|
||||
</div>
|
||||
<div class="col-lg-12" style="padding:1em 0 0 0;margin-left:-1em">
|
||||
{# СХЕМЫ ОТКРЫВАНИЯ: статическая часть, если используется кеш #}
|
||||
{% if PRE_RENDERED_STATIC_FLAPS_PATH %}
|
||||
{% include PRE_RENDERED_STATIC_FLAPS_PATH %}
|
||||
{% else %}
|
||||
{% include 'report/show_big_flap_pictures.html' %}
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-lg-12" style="padding:1em 0 0 0;margin-left:-1em">
|
||||
{% include 'report/show_big_flap_pictures.html' %}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="row">
|
||||
<div class="col-lg-8 col-xs-12 col-md-offset-1">
|
||||
<h3 class="header">Оконные проёмы в типовых квартирах <nobr>серии {{ THIS_SERIA_NAME }}</nobr></h3>
|
||||
<h3 class="header">Оконные проёмы в типовых квартирах <nobr>серии {{ THIS_SERIA_NAME }}</nobr></h3>
|
||||
</div>
|
||||
<div class="col-lg-8 col-xs-12 col-md-offset-1">
|
||||
<!--- прешаблон начало ---><table style="padding:2px;">{% templatetag openblock %} for row in TABLE_OF_WINDOWS {% templatetag closeblock %}
|
||||
<tr class="tr2">
|
||||
<td>{% templatetag openvariable %} row.APART_NAME|safe {% templatetag closevariable %}</td>{% templatetag openblock %} for col in row.WIN_IN_APART {% templatetag closeblock %}
|
||||
<td class="cntr">{% templatetag openblock %} if col.WIN_ID {% templatetag closeblock %}<nobr title="{% templatetag openvariable %} col.WIN_Q {% templatetag closevariable %} × {% templatetag openvariable %} col.WIN_DESCRIPTION {% templatetag closevariable %}: {% templatetag openvariable %} col.WIN_WIDTH {% templatetag closevariable %}шт.: {% templatetag openvariable %} col.WIN_HEIGHT {% templatetag closevariable %} (Ш×В, см.). Схема открывания: {% templatetag openvariable %} col.WIN_FLAPCFG {% templatetag closevariable %}">{% templatetag openblock %} for I_II in col.WIN_NUM {% templatetag closeblock %}<span style="background-image:url('{% static 'img/svg/mark' %}{% templatetag openvariable %} I_II {% templatetag closevariable %}.svg');"> </span>{% templatetag openblock %} endfor {% templatetag closeblock %}</nobr>{% templatetag openblock %} else {% templatetag closeblock %}—{% templatetag openblock %} endif {% templatetag closeblock %}</td>{% templatetag openblock %} endfor {% templatetag closeblock %}
|
||||
<td style="background:#f9f9f9;"><a href="#{% templatetag openvariable %} row.APART_ID {% templatetag closevariable %}" class="badge" title="Оконных предложений для квартиры: {% templatetag openvariable %} row.NUM_OFFERS {% templatetag closevariable %}"><small class="glyphicon glyphicon-tags" aria-hidden="true"></small> {% templatetag openvariable %} row.NUM_OFFERS {% templatetag closevariable %}</a></td>
|
||||
</tr>{% templatetag openblock %} endfor {% templatetag closeblock %}
|
||||
<tr class="trZ">
|
||||
<td style="font-size: xx-small;vertical-align:text-top">© 2015-{% now "Y" %}, данные: oknardia.ru</td>{% templatetag openblock %} for i in WIN_OFFER_AND_MERCHANT {% templatetag closeblock %}
|
||||
<td class="cntr" style="background:#f9f9f9;"><a href="/tsena-odnogo-okna/{% templatetag openvariable %} i.WIN_W|floatformat:0 {% templatetag closevariable %}0x{% templatetag openvariable %} i.WIN_H|floatformat:0 {% templatetag closevariable %}0mm/tip{% templatetag openvariable %} i.WIN_ID {% templatetag closevariable %}" class="badge" title="Ценовых предложений для окна: {% templatetag openvariable %} i.WIN_OFFER {% templatetag closevariable %}"><small class="glyphicon glyphicon-tags" aria-hidden="true"></small> {% templatetag openvariable %} i.WIN_OFFER {% templatetag closevariable %}</a></td>{% templatetag openblock %} endfor {% templatetag closeblock %}
|
||||
<td></td>
|
||||
</tr>
|
||||
</table>
|
||||
<!--- прешаблон конец ---></div>
|
||||
{# ТАБЛИЦА ОКОН: часть, которая считается при каждом запросе #}
|
||||
{% include "seria_info/all_seria_info_pre_light_dynamic_include.html" %}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="row">
|
||||
<div class="col-md-9"><a name="s_graph"></a>
|
||||
<h2 class="header">Здания серия {{ THIS_SERIA_NAME }}: ввод в эксплуатацию по годам</h2>
|
||||
</div>
|
||||
<div class="col-md-9 col-md-offset-1" style="height:300px;font-size:large;">
|
||||
{% include 'seria_info/yaer_graph.html' %}
|
||||
<div class="row">
|
||||
<div class="col-md-9"><a name="s_graph"></a>
|
||||
<h2 class="header">Здания серия {{ THIS_SERIA_NAME }}: ввод в эксплуатацию по годам</h2>
|
||||
</div>
|
||||
<div class="col-md-9 col-md-offset-1" style="height:300px;font-size:large;" id="graph">
|
||||
{# ГРАФИК: статическая часть, если используется кеш #}
|
||||
{% if PRE_RENDERED_STATIC_GRAPH_PATH %}
|
||||
{% include PRE_RENDERED_STATIC_GRAPH_PATH %}
|
||||
{% else %}
|
||||
{% include 'seria_info/all_seria_info_pre_light_static_graph.html' %}
|
||||
{% endif %}
|
||||
</div>
|
||||
<div class="col-md-9 col-md-offset-1">
|
||||
<div style="font-size: xx-small;float: right">© 2015-{% now "Y" %}, данные: oknardia.ru</div>
|
||||
@@ -82,36 +183,43 @@
|
||||
<div class="col-md-7"><a name="s_map"></a>
|
||||
<h2 class="header">Строения серии {{ THIS_SERIA_NAME }} на карте</h2>
|
||||
</div>
|
||||
<div class="col-md-7 col-lg-offset-1">
|
||||
<div class="col-md-7 col-lg-offset-1">
|
||||
<p><small>Чтобы посмотреть цены на установку и замену окон от партнёров «Окнардия» в своей квартире: найдите дом на карте; кликните на него; перейдите по ссылке «Смотреть коммерческие предложения». При необходимости смените типовую планировку квартиры (на странице ценовой выдачи, справа от изображения типовых проёмов и схем открывания).</small></p>
|
||||
<div style="height:350px;">{% include 'seria_info/geo_map.html' %}</div>
|
||||
<div style="height:350px;">
|
||||
<div id="SeriaMap" style="height: 100%;"></div>
|
||||
</div>
|
||||
<div style="font-size: xx-small;float: right">© 2015-{% now "Y" %}, данные: oknardia.ru</div>
|
||||
</div>
|
||||
|
||||
<diV class="col-md-4">
|
||||
<h3 class="header">Статистика <nobr>серии {{ THIS_SERIA_NAME }}</nobr></h3>
|
||||
<p>Совокупно во всех зданиях типового проекта:</p>
|
||||
<ul>
|
||||
<li><strong>{{ ACCOUNTS|price_format }}</strong> квартир.</li>
|
||||
<li>Проживает <strong>{{ APARTMENTS|price_format }}</strong> семей <small>(<strong>{{ RESIDENTS|price_format }}</strong> человек)</small>.</li>
|
||||
<li><strong>{{ RESIDENTIAL_M2|stringformat:".1f"|price_format }} м²</strong> жилых помещений.</li>
|
||||
<li><strong>{{ MUNICIPAL_M2|stringformat:".1f"|price_format }} м²</strong> — муниципальное жильё.</li>
|
||||
<li><strong>{{ GOVERNMENT_M2|stringformat:".1f"|price_format }} м²</strong> занимают государственные и городские службы, учреждения бытового обслуживания, магазины, офисы и тому подобное.</li>
|
||||
<li>Максимальный износ жилого фонда серии {{ THIS_SERIA_NAME }} — <strong>{{ CONDITION_MAX|stringformat:".2f" }} %</strong>. Минимальный — <strong>{{ CONDITION_MIN|stringformat:".2f" }} %</strong>. </li>
|
||||
</ul>
|
||||
</diV>
|
||||
{# КАРТА И СТАТИСТИКА: статическая часть, если используется кеш #}
|
||||
{% if PRE_RENDERED_STATIC_MAP_STATS_PATH %}
|
||||
{% include PRE_RENDERED_STATIC_MAP_STATS_PATH %}
|
||||
{% else %}
|
||||
<diV class="col-md-4">
|
||||
<h3 class="header">Статистика <nobr>серии {{ THIS_SERIA_NAME }}</nobr></h3>
|
||||
<p>Совокупно во всех зданиях типового проекта:</p>
|
||||
<ul>
|
||||
<li><strong>{{ ACCOUNTS|price_format }}</strong> квартир.</li>
|
||||
<li>Проживает <strong>{{ APARTMENTS|price_format }}</strong> семей <small>(<strong>{{ RESIDENTS|price_format }}</strong> человек)</small>.</li>
|
||||
<li><strong>{{ RESIDENTIAL_M2|stringformat:".1f"|price_format }} м²</strong> жилых помещений.</li>
|
||||
<li><strong>{{ MUNICIPAL_M2|stringformat:".1f"|price_format }} м²</strong> — муниципальное жильё.</li>
|
||||
<li><strong>{{ GOVERNMENT_M2|stringformat:".1f"|price_format }} м²</strong> занимают государственные и городские службы, учреждения бытового обслуживания, магазины, офисы и тому подобное.</li>
|
||||
<li>Максимальный износ жилого фонда серии {{ THIS_SERIA_NAME }} — <strong>{{ CONDITION_MAX|stringformat:".2f" }} %</strong>. Минимальный — <strong>{{ CONDITION_MIN|stringformat:".2f" }} %</strong>. </li>
|
||||
</ul>
|
||||
</diV>
|
||||
{% endif %}
|
||||
</div>
|
||||
|
||||
<div class="row">
|
||||
<div class="col-md-4 col-lg-offset-8" style="margin-top: 1em;">{# <div class="col-md-4"> #}
|
||||
<div class="col-md-4 col-lg-offset-8" style="margin-top: 1em;">
|
||||
<h5>Информация о других, отличных от {{ THIS_SERIA_NAME }}, типовых сериях в базе «Окнардия», типовых планировках квартир и оконных проёмах в них, а также рекомендации по замене окон:</h5>
|
||||
<div class="href_d">{% include 'seria_info/seria_nav.html' %}</div>
|
||||
<p style="text-align:right;padding-top:1em;padding-bottom:1em;"><a href="/stat_all/">Агрегированная информация<br />по типовым сериям домов<br />в базе «Окнардия»</a>.</p>
|
||||
<div class="href_d">{% include 'seria_info/seria_nav.html' with current_seria_id=THIS_SERIA_ID %}</div>
|
||||
<p style="text-align:left;padding: 1em 50% 2em 0;"><small>Смотри так же <a href="/stat_all/">агрегированную информацию</a> по типовым сериям типового строительства в базе «Окнардия».</small></p>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row">
|
||||
{% templatetag openblock %}include "report/report_last_user_visit.html" {% templatetag closeblock %}
|
||||
{% templatetag openblock %} include "report/report_log_user_visit.html" {% templatetag closeblock %}
|
||||
{% include "report/report_last_user_visit.html" %}
|
||||
{% include "report/report_log_user_visit.html" %}
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,43 @@
|
||||
{# ============================================================================ #}
|
||||
{# ДИНАМИЧЕСКИЕ ДАННЫЕ ДЛЯ СЕРИИ (НЕ кешируемая часть) #}
|
||||
{# Содержит: Таблица раскладки окон по квартирам + статистика предложений #}
|
||||
{# ЗАМЕЧАНИЕ: этот блок ЧАСТО меняется (при добавлении новых предложений) #}
|
||||
{# ============================================================================ #}
|
||||
|
||||
<div class="col-md-9 col-xs-12" style="padding:0;">
|
||||
<table style="padding:2px;">
|
||||
{% for row in TABLE_OF_WINDOWS %}
|
||||
<tr class="tr2">
|
||||
<td>{{ row.APART_NAME|safe }}</td>
|
||||
{% for col in row.WIN_IN_APART %}
|
||||
<td class="cntr">
|
||||
{% if col.WIN_ID %}
|
||||
<nobr title="{{ col.WIN_Q }} × {{ col.WIN_DESCRIPTION }}: {{ col.WIN_WIDTH }}шт.: {{ col.WIN_HEIGHT }} (Ш×В, см.). Схема открывания: {{ col.WIN_FLAPCFG }}">
|
||||
{% for I_II in col.WIN_NUM %}
|
||||
<span style="background-image:url('/static/img/svg/mark{{ I_II }}.svg');"> </span>
|
||||
{% endfor %}
|
||||
</nobr>
|
||||
{% else %}—{% endif %}
|
||||
</td>
|
||||
{% endfor %}
|
||||
<td style="background:#f9f9f9;">
|
||||
<a href="#{{ row.APART_ID }}" class="badge" title="Оконных предложений для квартиры: {{ row.NUM_OFFERS }}">
|
||||
<small class="glyphicon glyphicon-tags" aria-hidden="true"></small> {{ row.NUM_OFFERS }}
|
||||
</a>
|
||||
</td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
<tr class="trZ">
|
||||
<td style="font-size: xx-small;vertical-align:text-top">© 2015-2026, данные: oknardia.ru</td>
|
||||
{% for i in WIN_OFFER_AND_MERCHANT %}
|
||||
<td class="cntr" style="background:#f9f9f9;">
|
||||
<a href="/catalog/standard_opening/price-{{ i.WIN_W|floatformat:0 }}0x{{ i.WIN_H|floatformat:0 }}0mm-tip{{ i.WIN_ID }}" class="badge" title="Ценовых предложений для окна: {{ i.WIN_OFFER }}">
|
||||
<small class="glyphicon glyphicon-tags" aria-hidden="true"></small> {{ i.WIN_OFFER }}
|
||||
</a>
|
||||
</td>
|
||||
{% endfor %}
|
||||
<td></td>
|
||||
</tr>
|
||||
</table>
|
||||
</div>
|
||||
|
||||
@@ -0,0 +1,21 @@
|
||||
{# ============================================================================ #}
|
||||
{# СХЕМЫ ОТКРЫВАНИЯ И РАЗМЕРЫ (кешируемая статическая часть) #}
|
||||
{# ============================================================================ #}
|
||||
{% load static %}
|
||||
{% load filters %}
|
||||
|
||||
{% if WIN_DIM %}
|
||||
{% for I_WIN_DIM in FLAP_DIM %}
|
||||
<div class="win_discr pull-left" id="flap{{ forloop.counter0 }}">
|
||||
<div><img src="{% static I_WIN_DIM.url2img %}" alt="{{ I_WIN_DIM.sDescription }}. Размер {{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0x{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0 (Ш х В, мм.). Типовая схема открывания." title="{{ I_WIN_DIM.sDescription }}. Размер {{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0x{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0 (Ш х В, мм.). Типовая схема открывания." itemprop="image" /></div>
|
||||
<div class="caption" style="width:{{ I_WIN_DIM.W }}px;min-width:13ex;">
|
||||
<nobr>{{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0×{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0 мм.</nobr><br />{% if not I_WIN_DIM.iQuantity == 0 %}
|
||||
<nobr><b>{{ I_WIN_DIM.iQuantity }} шт.</b>{% for I_II in I_WIN_DIM.qStr %}<span class="color-bullet" style="background-image:url('{% static 'img/svg/mark' %}{{ I_II }}.svg');"></span>{% endfor %}</nobr><br />{% endif %}
|
||||
{{ I_WIN_DIM.sDescription }}{% if not I_WIN_DIM.iQuantity == 0 %}<br />
|
||||
<a href="/catalog/standard_opening/price-{{ I_WIN_DIM.iWinWidth|stringformat:".0f" }}0x{{ I_WIN_DIM.iWinHight|stringformat:".0f" }}0mm-tip{{ I_WIN_DIM.id }}">цены только этого типового окна</a>{% endif %}
|
||||
</div>
|
||||
</div>{% endfor %}
|
||||
{% else %}
|
||||
<h1>Нет данных о проемах и рекомендованных схемах открывания окон</h1>
|
||||
{% endif %}
|
||||
|
||||
@@ -0,0 +1,55 @@
|
||||
{# ============================================================================ #}
|
||||
{# ГРАФИК ВВОДА В ЭКСПЛУАТАЦИЮ (кешируемая статическая часть) #}
|
||||
{# ============================================================================ #}
|
||||
|
||||
<script type="text/javascript" src="https://www.google.com/jsapi" type="text/javascript"></script>
|
||||
<script type="text/javascript">
|
||||
google.load("visualization", "1", {packages:["corechart"]});
|
||||
google.setOnLoadCallback(drawChart);
|
||||
function drawChart() {
|
||||
let data = google.visualization.arrayToDataTable([
|
||||
["Год", "Введено в эксплуатацию", {role:'style'}],{% for row in DATA4GRAPH %}
|
||||
["{{ row.YEAR }}",{{ row.NUMS }},"color: #99{{ row.CLRS }}99"]{% if not forloop.last %},{% endif %}{% endfor %}
|
||||
]);
|
||||
let view = new google.visualization.DataView(data);
|
||||
view.setColumns([0, 1,
|
||||
{ calc: "stringify",
|
||||
sourceColumn: 1,
|
||||
type: "string",
|
||||
role: "annotation" },
|
||||
2
|
||||
]);
|
||||
let options = {
|
||||
animation:{
|
||||
duration: 1500,
|
||||
easing: 'in',
|
||||
startup: true
|
||||
},
|
||||
backgroundColor: "#EEEEEE",
|
||||
bar: {groupWidth: "76.4%"},
|
||||
chartArea: {left: "2%", top: "5%", width: '96%', height: '85%'},
|
||||
dataOpacity: 0.76,
|
||||
explorer:{
|
||||
maxZoomIn: 0.20,
|
||||
maxZoomOut: 32 },
|
||||
vAxis: {
|
||||
baselineColor:'grey',
|
||||
gridlines:{color: 'silver', count: 7},
|
||||
minorGridlines:{color: '#dddddd', count: 3},
|
||||
textPosition: 'in',
|
||||
textStyle: {fontSize: 10}
|
||||
},
|
||||
hAxis: { textStyle: {fontSize: 10} },
|
||||
isStacked: true,
|
||||
tooltip: {
|
||||
textStyle:{color: 'grey', fontSize: 10 },
|
||||
trigger: 'selection'
|
||||
},
|
||||
annotations: {textStyle: { fontSize: 8, bold: true, color: 'black', opacity: 0.8 }},
|
||||
legend: { position: "none" }
|
||||
};
|
||||
let chart = new google.visualization.ColumnChart(document.getElementById("graph"));
|
||||
chart.draw(data, options);
|
||||
}
|
||||
</script>
|
||||
|
||||
@@ -0,0 +1,93 @@
|
||||
{# ============================================================================ #}
|
||||
{# КАРТА И СТАТИСТИКА СЕРИИ (кешируемая статическая часть) #}
|
||||
{# ============================================================================ #}
|
||||
{% load humanize %}
|
||||
{% load filters %}
|
||||
|
||||
{# БЛОК КАРТА: левая часть (col-md-7) #}
|
||||
<script src="https://api-maps.yandex.ru/2.1/?lang=ru_RU" type="text/javascript"></script>
|
||||
<script type="text/javascript">
|
||||
|
||||
// Функция для декодирования Base64-обфускованных геоданных (защита координат)
|
||||
function decodeGeoData(b64str) {
|
||||
try {
|
||||
var json = atob(b64str);
|
||||
return JSON.parse(json);
|
||||
} catch(e) {
|
||||
console.error('Ошибка декодирования геоданных:', e);
|
||||
return [];
|
||||
}
|
||||
}
|
||||
|
||||
ymaps.ready(function () {
|
||||
let myMap = new ymaps.Map('SeriaMap', {
|
||||
center: [55.75, 37.57],
|
||||
zoom: 10,
|
||||
behaviors: ['default', 'scrollZoom'],
|
||||
controls: [ 'rulerControl', 'zoomControl', 'geolocationControl', 'fullscreenControl' ]
|
||||
});
|
||||
// Создадим кластеризатор, вызвав функцию-конструктор.
|
||||
clusterer = new ymaps.Clusterer({
|
||||
preset: 'islands#invertedGrayClusterIcons',
|
||||
groupByCoordinates: false,
|
||||
hasHint: false,
|
||||
viewportMargin: 0,
|
||||
zoomMargin: 16,
|
||||
clusterDisableClickZoom: false,
|
||||
gridSize: 80
|
||||
});
|
||||
geoObjects = [];
|
||||
|
||||
const linkText = 'Смотреть коммерческие предложения</a>';
|
||||
const hintText = '<b>Здание серии {{ THIS_SERIA_NAME }}</b>';
|
||||
const apartmentId = {{ first_apart_id|default:0 }};
|
||||
const seriaId = {{ THIS_SERIA_ID }};
|
||||
const seriaSlug = '{{ THIS_SERIA_NAME_T }}';
|
||||
|
||||
// Декодируем обфускованные геоданные: [lat, lon, addr_id, seria_id]
|
||||
var geoData = decodeGeoData('{{ DATA4GEO_B64 }}');
|
||||
|
||||
// Создаем метки для каждого здания серии
|
||||
for(var i = 0, len = geoData.length; i < len; i++) {
|
||||
const latitude = geoData[i][0];
|
||||
const longitude = geoData[i][1];
|
||||
const buildingId = geoData[i][2];
|
||||
// Формируем SEO-URL для каждой метки
|
||||
const balloonLink = `<a href="/price/seriaID${seriaId}--${seriaSlug}/appartID${apartmentId}/addressID${buildingId}--null">`;
|
||||
|
||||
geoObjects[i] = new ymaps.Placemark( [longitude, latitude],
|
||||
{ // Содержимое иконки, балуна и хинта.
|
||||
balloonContent: balloonLink + linkText,
|
||||
hintContent: hintText
|
||||
},
|
||||
{ preset:'islands#circleIcon',iconColor: 'silver'} );
|
||||
geoObjects[i].events
|
||||
.add('mouseenter', function (e) {
|
||||
e.get('target').options.set('preset', 'islands#yellowCircleIcon');
|
||||
})
|
||||
.add('mouseleave', function (e) {
|
||||
e.get('target').options.set('preset', 'islands#grayCircleIcon');
|
||||
});
|
||||
}
|
||||
// Добавляем метки в кластеризатор.
|
||||
clusterer.add(geoObjects);
|
||||
myMap.geoObjects.add(clusterer);
|
||||
// позиционирование карты так, чтобы на ней были видны все объекты кластера.
|
||||
myMap.setBounds(clusterer.getBounds(), { checkZoomRange: true });
|
||||
});
|
||||
</script>
|
||||
|
||||
{# БЛОК СТАТИСТИКА: правая часть (col-md-4) #}
|
||||
<diV class="col-md-4">
|
||||
<h3 class="header">Статистика <nobr>серии {{ THIS_SERIA_NAME }}</nobr></h3>
|
||||
<p>Совокупно во всех зданиях типового проекта:</p>
|
||||
<ul>
|
||||
<li><strong>{{ ACCOUNTS|price_format }}</strong> квартир.</li>
|
||||
<li>Проживает <strong>{{ APARTMENTS|price_format }}</strong> семей <small>(<strong>{{ RESIDENTS|price_format }}</strong> человек)</small>.</li>
|
||||
<li><strong>{{ RESIDENTIAL_M2|stringformat:".1f"|price_format }} м²</strong> жилых помещений.</li>
|
||||
<li><strong>{{ MUNICIPAL_M2|stringformat:".1f"|price_format }} м²</strong> — муниципальное жильё.</li>
|
||||
<li><strong>{{ GOVERNMENT_M2|stringformat:".1f"|price_format }} м²</strong> занимают государственные и городские службы, учреждения бытового обслуживания, магазины, офисы и тому подобное.</li>
|
||||
<li>Максимальный износ жилого фонда серии {{ THIS_SERIA_NAME }} — <strong>{{ CONDITION_MAX|stringformat:".2f" }} %</strong>. Минимальный — <strong>{{ CONDITION_MIN|stringformat:".2f" }} %</strong>. </li>
|
||||
</ul>
|
||||
</diV>
|
||||
|
||||
@@ -3,17 +3,17 @@
|
||||
{% load filters %}
|
||||
{% load humanize %}
|
||||
|
||||
{% block Title %} Статистика типового строительства СССР и России.{% endblock %}
|
||||
{% block Title %} Статистика типовых серий домов в России | Панельное строительство{% endblock %}
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{# block Date4Meta %}{{ META_DATA_PUBLISH|date:"c" }}{% endblock #}
|
||||
{% block Description %}Статистика типового строительства в России: анализ распределения панельных домов по сериям и регионам. Данные о 18228 зданиях, общей площади жилого фонда, графиках ввода в эксплуатацию. Здания серий: {% for CountSeria in SERIA_NAV_DIM %}{{ CountSeria.SERIA_R }}{% if not forloop.last %}, {% endif %}{% endfor %}.{% endblock %}
|
||||
|
||||
{# block Last4Meta %}{{ META_DATA_PUBLISH|date:"c" }}{% endblock #}
|
||||
{% block Keywords %}типовые серии домов, панельное строительство, статистика жилого фонда, каталог типовых зданий, распределение серий по регионам, статистика типовых домов, кирпичные дома, износ жилого фонда, площадь жилого фонда, количество зданий, годы возведения, анализ панельной застройки{% endblock %}
|
||||
|
||||
{% block Description %}Статистика типового строительства СССР и России. Географи, график ввода в эксплуатацтяю, метраж. Здания проектов серии: {% for CountSeria in SERIA_NAV_DIM %}{{ CountSeria.SERIA_R }}{% if not forloop.last %}, {% endif %}{% endfor %}.{% endblock %}
|
||||
{% block Author4Meta %}: Статистика типовых серий домов «Окнардия»{% endblock %}
|
||||
|
||||
{% block Keywords %}типовые проекты зданий, панельное строительство, {% for CountSeria in SERIA_NAV_DIM %}серия {{ CountSeria.SERIA_R }}, {{ CountSeria.SERIA_R }}, {% endfor %}, года простойки, регионы построки, распространенность{% endblock %}
|
||||
{% block CopyrightAuthor4Meta %}: Статистика типовых серий домов «Окнардия»{% endblock %}
|
||||
|
||||
{% block Top_JS1%}
|
||||
<script type="text/javascript">
|
||||
@@ -21,7 +21,95 @@ $(window).load(function(){let images = $('.half');images.each(function(i){$(this
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metatags --> #}
|
||||
{# Удалить: itemprop microdata, rel=standout, twitter:domain — устаревшие теги #}
|
||||
<meta name="news_keywords" content="типовые серии домов, панельное строительство, статистика жилого фонда, распределение серий по регионам" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/stat_all/" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
<meta property="fb:app_id" content="258354027974262" />
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/stat_all/" />
|
||||
<meta property="og:type" content="website" />
|
||||
<meta property="og:title" content="Статистика типового строительства - Окнардия" />
|
||||
<meta property="og:description" content="Статистика типового строительства в России. Анализ распределения серий домов по регионам, площадь жилого фонда, количество зданий." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="Статистика типового строительства - Окнардия" />
|
||||
<meta name="twitter:description" content="Статистика типового строительства в России. Анализ распределения серий домов по регионам." />
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta property="twitter:url" content="{{ request.scheme }}://{{ request.get_host }}/stat_all/" />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<meta name="relap-image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
{# <!-- END Дополнительные Metатags --> #}{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}
|
||||
{# JSON-LD: страница статистики типовых серий — CollectionPage + BreadcrumbList + DataCatalog #}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "CollectionPage",
|
||||
"name": "Статистика типовых серий домов",
|
||||
"description": "Анализ распределения типовых серий строительства в России: количество зданий, общая площадь, география и годы возведения.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/stat_all/",
|
||||
"isPartOf": {
|
||||
"@type": "WebSite",
|
||||
"name": "Окнардия",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
"mainEntity": {
|
||||
"@type": "DataCatalog",
|
||||
"name": "Статистика типовых серий",
|
||||
"description": "База данных типовых серий строительства в СССР и России",
|
||||
"dataset": [{% for CountSeria in SERIA_NAV_DIM %}
|
||||
{
|
||||
"@type": "Dataset",
|
||||
"name": "Данные серии {{ CountSeria.SERIA_R }}",
|
||||
"description": "Информация о типовой серии {{ CountSeria.SERIA_R }}"
|
||||
}{% if not forloop.last %},{% endif %}{% endfor %}
|
||||
]
|
||||
}
|
||||
}
|
||||
</script>
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Статистика",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/stat_all/"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Main_Content %}<div class="container-fluid">
|
||||
{# Хлебные крошки: НАЧАЛО #}
|
||||
<div class="row">
|
||||
<div class="col-md-11 col-xs-12">
|
||||
<ol class="breadcrumb">
|
||||
<li><a href="/">Главная</a></li>
|
||||
<li class="active">Статистика типового строительства России</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
{# Хлебные крошки: КОНЕЦ #}
|
||||
|
||||
<div class="row">
|
||||
<div class="col-md-9"><h1>Типовые серии домов в базе «Окнардия»</h1></div>
|
||||
</div>
|
||||
@@ -38,15 +126,6 @@ DimColor = [];
|
||||
for (i1=0; i1<=step; i1++ )
|
||||
for (i2=step; i2>=0; i2-- )
|
||||
for (i3=0; i3<=step; i3++ ) {
|
||||
//document.write(" <span style='color:#"
|
||||
// + ("00"+(i1*step_tone).toString(16)).substr(-2)
|
||||
// + ("00"+(i2*step_tone).toString(16)).substr(-2)
|
||||
// + ("00"+(i3*step_tone).toString(16)).substr(-2)
|
||||
// + ";'>█</span> -- ");
|
||||
//document.write( "#"
|
||||
// + ("00"+(i1*step_tone).toString(16)).substr(-2)
|
||||
// + ("00"+(i2*step_tone).toString(16)).substr(-2)
|
||||
// + ("00"+(i3*step_tone).toString(16)).substr(-2) + "<br>");
|
||||
DimColor.push("#"
|
||||
+ ("00"+(i1*step_tone).toString(16)).substr(-2)
|
||||
+ ("00"+(i2*step_tone).toString(16)).substr(-2)
|
||||
|
||||
@@ -2,8 +2,7 @@
|
||||
{% block Top_JS5 %}<script src="https://api-maps.yandex.ru/2.1/?lang=ru_RU" type="text/javascript"></script>
|
||||
{% if MAP_JS %}<script src="{% static '' %}{{ MAP_JS }}" charset="utf-8" type="text/javascript"></script>{% else %}<script type="text/javascript">
|
||||
let points = [{% for count in DATA4GEO %}{% if forloop.last %}[{{ count.LONGITUDE|stringformat:"f" }},{{ count.LATITUDE|stringformat:"f" }}]{% else %}[{{ count.LONGITUDE|stringformat:"f" }},{{ count.LATITUDE|stringformat:"f" }}],{% endif %}{% endfor %}];
|
||||
let forURL = [{% for count in DATA4GEO %}{{ count.ADDR_ID }}{# ,rus: '{{ count.ADDR_RUS }}',lat: '{{ count.ADDR_LAT }}' #}{% if not forloop.last %},{% endif %}{% endfor %}];
|
||||
|
||||
let forURL = [{% for count in DATA4GEO %}{ id: {{ count.ADDR_ID }} }{% if not forloop.last %},{% endif %}{% endfor %}];
|
||||
|
||||
ymaps.ready(function () {
|
||||
let myMap = new ymaps.Map('SeriaMap', {
|
||||
@@ -23,15 +22,23 @@ ymaps.ready(function () {
|
||||
gridSize: 80
|
||||
});
|
||||
geoObjects = [];
|
||||
add_str1 = '<a href="/';
|
||||
add_str2 = '/0/">Смотреть коммерческие предложения</a>';
|
||||
add_str3 = '<b>Здание серии {{ THIS_SERIA_NAME }}</b>';
|
||||
|
||||
const linkText = 'Смотреть коммерческие предложения</a>';
|
||||
const hintText = '<b>Здание серии {{ THIS_SERIA_NAME }}</b>';
|
||||
const apartmentId = {{ first_apart_id|default:0 }};
|
||||
const seriaId = {{ THIS_SERIA_ID }};
|
||||
const seriaSlug = '{{ THIS_SERIA_NAME_T }}';
|
||||
|
||||
// Данные передаются в конструктор метки.
|
||||
for(var i = 0, len = points.length; i < len; i++) {
|
||||
const buildingId = forURL[i].id;
|
||||
// Формируем SEO-URL для каждой метки
|
||||
const balloonLink = `<a href="/price/seriaID${seriaId}--${seriaSlug}/appartID${apartmentId}/addressID${buildingId}--null">`;
|
||||
|
||||
geoObjects[i] = new ymaps.Placemark( points[i],
|
||||
{ // Содержимое иконки, балуна и хинта.
|
||||
balloonContent: add_str1 + forURL[i] + add_str2,
|
||||
hintContent: add_str3
|
||||
balloonContent: balloonLink + linkText,
|
||||
hintContent: hintText
|
||||
},
|
||||
{ preset:'islands#circleIcon',iconColor: 'silver'} );
|
||||
geoObjects[i].events
|
||||
|
||||
13
oknardia/templates/seria_info/prepared/.gitignore
vendored
Normal file
13
oknardia/templates/seria_info/prepared/.gitignore
vendored
Normal file
@@ -0,0 +1,13 @@
|
||||
# Игнорируем все файлы в этой папке (генерируются динамически)
|
||||
# Содержит кеш-файлы со статик-шаблонами:
|
||||
# - *_id_static_flaps.html
|
||||
# - *_id_static_graph.html
|
||||
# - *_id_static_map_stats.html
|
||||
#
|
||||
# Эти файлы генерируются при запуске management-команды:
|
||||
# python manage.py regenerate_seria_prerender
|
||||
#
|
||||
# В продакшене они сохраняются на диск и включаются в шаблоны
|
||||
# за счет системы двухуровневого кеширования.
|
||||
*
|
||||
!.gitignore
|
||||
@@ -1 +1 @@
|
||||
{# Выводит навигацию по сериям домов #}{% for CountSeria in SERIA_NAV_DIM %}{% if CountSeria.SERIA_L == "" %}<span style="background-color:cornsilk;"><nobr>{{ CountSeria.SERIA_R }}</nobr></span>{% else %}<span><a href="/catalog/seria/{{ CountSeria.SERIA_L }}/all{{ CountSeria.ID2URL }}/"><nobr>{{ CountSeria.SERIA_R }}</nobr></a></span>{% endif %}{% endfor %}
|
||||
{# Выводит навигацию по сериям домов #}{% for CountSeria in SERIA_NAV_DIM %}{% if CountSeria.ID2URL == current_seria_id|default:0 %}<span style="background-color:cornsilk;"><nobr>{{ CountSeria.SERIA_R }}</nobr></span>{% else %}<span><a href="/catalog/seria/{{ CountSeria.SERIA_L }}/all{{ CountSeria.ID2URL }}/"><nobr>{{ CountSeria.SERIA_R }}</nobr></a></span>{% endif %}{% endfor %}
|
||||
@@ -1,27 +0,0 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="robots" content="noindex, nofollow" />
|
||||
<title>ОКНАРДИЯ :: Служебное</title>
|
||||
</head>
|
||||
<body>
|
||||
|
||||
<h1>Служебные ссылки</h1>
|
||||
|
||||
<ul>
|
||||
<li><b><a href="/service/tmp">Страница для тестирования верстки текста в блоге</a></b></li>
|
||||
</ul><ul>
|
||||
<li><b><a href="/service/make_sitemaps">Пересоздать файлы sitemaps.xml</a></b> (исполняется около 10 минут)</li>
|
||||
<li><a href="/service/make_JavaScripts4maps">Пересоздать JavaScript для карт на основе API Яндекс.Крат</a> (исполняется около 1 минуты)</li>
|
||||
<li><a href="/service/make_SeriaInfoRoot">Построить id корневых серий</a> (исполняется около 1 часа)</li>
|
||||
</ul><ul>
|
||||
<li><a href="/service/del_CachingTemplate">Удалить кеширующие шаблоны</a> (для страниц про серии домов)</li>
|
||||
</ul><ul>
|
||||
<li><a href="/service/make_FillGeoCode">Добавить GeoCode для адресов где его нет</a></li>
|
||||
</ul><ul>
|
||||
<li><a href="/service/make_Rating">Пересчитать рейтинги Профилей</a></li>
|
||||
</ul>
|
||||
|
||||
</body>
|
||||
</html>
|
||||
58
oknardia/templates/service/js_4all_seria_map_js.html
Executable file
58
oknardia/templates/service/js_4all_seria_map_js.html
Executable file
@@ -0,0 +1,58 @@
|
||||
step = Math.round( Math.pow({{ SERIA_NAV_DIM|length }}, 1./3.)-1);
|
||||
step_tone = Math.floor(0xF0/step);
|
||||
DimColor = [];
|
||||
for (i1=0; i1<=step; i1++ )
|
||||
for (i2=step; i2>=0; i2-- )
|
||||
for (i3=0; i3<=step; i3++ ) {
|
||||
DimColor.push("#"+("00"+(i1*step_tone).toString(16)).substr(-2)+("00"+(i2*step_tone).toString(16)).substr(-2)+("00"+(i3*step_tone).toString(16)).substr(-2));
|
||||
}
|
||||
// Объекты для хранения цветов и названий серий (вместо отдельных переменных)
|
||||
c = {};
|
||||
s = {};
|
||||
{% for CountSeria in SERIA_NAV_DIM %}c[{{ CountSeria.ID2URL }}] = DimColor[{{ forloop.counter0 }}]; s[{{ CountSeria.ID2URL }}] = "{{ CountSeria.SERIA_R }}"; {% endfor %}
|
||||
|
||||
b = '<a href="/';
|
||||
z = '/0/">Смотреть цены на установку окон</a>';
|
||||
w = '<b>Здание серии ';
|
||||
|
||||
// Функция-фабрика для создания маркеров (оптимизация размера JS)
|
||||
function m(coord, id, sId) {
|
||||
return new ymaps.Placemark(coord,
|
||||
{balloonContent: b + id + z, hintContent: w + (s[sId] || 'нет данных') + '</b>'},
|
||||
{preset: 'islands#circleIcon', iconColor: c[sId]}
|
||||
);
|
||||
}
|
||||
|
||||
// Функция для декодирования Base64-обфускованных координат (защита геоданных)
|
||||
function decodeGeoData(b64str) {
|
||||
try {
|
||||
var json = atob(b64str);
|
||||
return JSON.parse(json);
|
||||
} catch(e) {
|
||||
console.error('Ошибка декодирования геоданных:', e);
|
||||
return [];
|
||||
}
|
||||
}
|
||||
|
||||
ymaps.ready(function () {
|
||||
var myMap = new ymaps.Map('SeriaMap', {
|
||||
center: [55.75, 37.57],
|
||||
zoom: 10,
|
||||
behaviors: ['default', 'scrollZoom'],
|
||||
controls: ['rulerControl', 'zoomControl', 'geolocationControl', 'fullscreenControl']
|
||||
});
|
||||
myMap.behaviors.disable('scrollZoom');
|
||||
ymaps.modules.require(['PieChartClusterer'], function (PieChartClusterer) {
|
||||
var clusterer = new PieChartClusterer({margin: 10});
|
||||
// Декодируем обфускованные координаты: [lat, lon, addr_id, ser_id]
|
||||
var geoData = decodeGeoData('{{ DATA4GEO_B64 }}');
|
||||
var points = [];
|
||||
for (var i = 0; i < geoData.length; i++) {
|
||||
points.push(m([geoData[i][1], geoData[i][0]], String(geoData[i][2]), geoData[i][3]));
|
||||
}
|
||||
clusterer.add(points);
|
||||
myMap.geoObjects.add(clusterer);
|
||||
});
|
||||
// позиционирование карты так, чтобы на ней были видны все объекты клястера.
|
||||
// myMap.setBounds(clusterer.getBounds(), { checkZoomRange: true });
|
||||
});
|
||||
@@ -1,141 +0,0 @@
|
||||
<!DOCTYPE html>{% load static %}
|
||||
<html lang="ru-RU">
|
||||
<head>
|
||||
<meta charset="utf-8"/>
|
||||
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
|
||||
<meta http-equiv="content-language" content="ru"/>
|
||||
<meta http-equiv="Date" content="{% block Date4Meta %}{% now "c" %}{% endblock %}"/>
|
||||
<meta http-equiv="Last-Modified" content="{% block Last4Meta %}{% now "c" %}{% endblock %}"/>
|
||||
<meta http-equiv="Expires" content="{% block Expires4Meta %}{% now "c" %}{% endblock %}"/>
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1"/>
|
||||
|
||||
<meta name="description" content="
|
||||
{% block Description %}{{ META_DESCRIPTION|default:"" }}Служебный интерфейс «Окнардия» закрыт{% endblock %}"/>
|
||||
<meta name="keywords"
|
||||
content="{% block Keywords %}цены на пластиковые окна, агрегатор окон, доступ закрыт{{ META_KEYWORDS|default:"" }}{% endblock %}"/>
|
||||
<meta name="author" content="OKNARDIA.RU{% block Author4Meta %}{% endblock %}"/>
|
||||
<meta name="copyright" lang="ru" content="OKNARDIA.RU{% block CopyrightAuthor4Meta %}{% endblock %}"/>
|
||||
<meta name="robots" content="index,follow"/>
|
||||
<meta name="document-state" content="{{ META_DOCUMENT_STATE|default:"Static" }}"/>
|
||||
<meta name="generator" content="OKNARDIA 0.3β by Python/Django"/>
|
||||
<title>ОКНАРДИЯ: Нет доступа в служебный интерфейс</title>
|
||||
<link href="{% static 'css/bootstrap.min.css' %}" rel="stylesheet" type="text/css" />{# <link href="//maxcdn.bootstrapcdn.com/bootstrap/3.3.6/css/bootstrap.min.css" rel="stylesheet" type="text/css" />#}
|
||||
<script src="{% static 'js/jquery-2.1.1.min.js' %}" type="text/javascript"></script>{# <script src="//ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js" type="text/javascript"></script>#}
|
||||
<script src="{% static 'js/bootstrap.min.js' %}" type="text/javascript"></script>{# <script src="//maxcdn.bootstrapcdn.com/bootstrap/3.3.6/js/bootstrap.min.js" type="text/javascript"></script>#}{% block Top_JS1 %}{% endblock %}{% block Top_JS2 %}{% endblock %}{% block Top_JS3 %}{% endblock %}{% block Top_JS4 %}{% endblock %}{% block Top_JS5 %}{% endblock %}{% block Top_Meta1 %}{% endblock %}
|
||||
<link href="https://fonts.googleapis.com/css?family=Open+Sans+Condensed:300" rel="stylesheet">
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<div class="row">
|
||||
<div class="col-xs-8 col-xs-offset-4">
|
||||
<p> <br> <br></p>
|
||||
|
||||
<h1 style="font-family: 'Open Sans Condensed', sans-serif;
|
||||
font-size: 5em;
|
||||
font-weight: 900;
|
||||
text-shadow: 1px 1px 6px silver;">Служебный интерфейс закрыт</h1>
|
||||
|
||||
<p style="font-family: 'Open Sans Condensed', sans-serif;font-size: 3ex;">Доступ только для админов! Поисковикам и
|
||||
пользователям лезть сюда не за чем…</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div style="height: 100% !important; width: 100% !important; background-image: url('/static/img/cubex.png');
|
||||
background-repeat: no-repeat; background-position: 10% 85%; z-index:-2; position:absolute !important;
|
||||
bottom: 0; right: 0;">
|
||||
<div style="position: absolute; bottom: 0; right: 0;padding-bottom: 1ex;padding-right: 2ex;">
|
||||
<script type="text/javascript">
|
||||
{# <!-- Google Analylics --> #}(function (i, s, o, g, r, a, m) {
|
||||
i['GoogleAnalyticsObject'] = r;
|
||||
i[r] = i[r] || function () {
|
||||
(i[r].q = i[r].q || []).push(arguments)
|
||||
}, i[r].l = 1 * new Date();
|
||||
a = s.createElement(o), m = s.getElementsByTagName(o)[0];
|
||||
a.async = 1;
|
||||
a.src = g;
|
||||
m.parentNode.insertBefore(a, m)
|
||||
})(window, document, 'script', '//www.google-analytics.com/analytics.js', 'ga');
|
||||
ga('create', 'UA-9116991-5', 'auto');
|
||||
ga('send', 'pageview');
|
||||
{# <!-- Rating@Mail.ru counter --> #}var _tmr = _tmr || [];
|
||||
_tmr.push({id: "2018432", type: "pageView", start: (new Date()).getTime()});
|
||||
(function (d, w, id) {
|
||||
if (d.getElementById(id))return;
|
||||
var ts = d.createElement("script");
|
||||
ts.type = "text/javascript";
|
||||
ts.async = true;
|
||||
ts.id = id;
|
||||
ts.src = (d.location.protocol == "https:" ? "https:" : "http:") + "//top-fwz1.mail.ru/js/code.js";
|
||||
var f = function () {
|
||||
var s = d.getElementsByTagName("script")[0];
|
||||
s.parentNode.insertBefore(ts, s);
|
||||
};
|
||||
if (w.opera == "[object Opera]") {
|
||||
d.addEventListener("DOMContentLoaded", f, false);
|
||||
} else {
|
||||
f();
|
||||
}
|
||||
})(document, window, "topmailru-code");
|
||||
{# <!-- LiveInternet counter --> #}
|
||||
</script>
|
||||
<noscript>
|
||||
<div style="position:absolute;left:-10000px;">
|
||||
{# <!-- Rating@Mail.ru nosript --> #}<img src="//top-fwz1.mail.ru/counter?id=2018432;js=na"
|
||||
style="border:0;height:1px;width:1px" alt=""/>
|
||||
{# <!-- Yandex.Metrika counter --> #}<img src="//mc.yandex.ru/watch/32997984"
|
||||
style="border:0;height:1px;width:1px"
|
||||
alt=""/>{# <!-- /Yandex.Metrika counter --> #}
|
||||
</div>
|
||||
</noscript>
|
||||
{#<!-- Rating@Mail.ru logo -->#}<a target="_blank" href="http://top.mail.ru/jump?from=2018432"><img
|
||||
src="//top-fwz1.mail.ru/counter?id=2018432;t=216;l=1" style="border:0" rel="nofollow"
|
||||
alt="Рейтинг@Mail.ru"></a>{#<!-- //Rating@Mail.ru logo -->#}
|
||||
{# <!-- Yandex.Metrika informer --> #}<a href="https://metrika.yandex.ru/stat/?id=32997984&from=informer"
|
||||
target="_blank" rel="nofollow"><img
|
||||
src="https://informer.yandex.ru/informer/32997984/3_0_E0E0E0FF_C0C0C0FF_0_pageviews"
|
||||
style="width:88px; height:31px; border:0;" alt="Яндекс.Метрика"
|
||||
title="Яндекс.Метрика: данные за сегодня (просмотры, визиты и уникальные посетители)"
|
||||
onclick="try{Ya.Metrika.informer({i:this,id:32997984,lang:'ru'});return false}catch(e){}"/></a>{# <!-- /Yandex.Metrika informer --> #}
|
||||
{# <!-- begin of Top100 code --> #}<span id="rambler"><script id="top100Counter" type="text/javascript"
|
||||
src="//counter.rambler.ru/top100.jcn?3148853"></script><noscript>
|
||||
<a href="http://top100.rambler.ru/navi/3148853/"><img src="http://counter.rambler.ru/top100.cnt?3148853"
|
||||
alt="Rambler's Top100" border="0"/></a>
|
||||
</noscript></span>{# <!-- end of Top100 code --> #}
|
||||
<script type="text/javascript"><!--
|
||||
{#<!--LiveInternet counter-->#}document.write("<a href='//www.liveinternet.ru/click' target=_blank><img src='//counter.yadro.ru/hit?t50.2;r" + escape(document.referrer) + ((typeof(screen) == "undefined") ? "" : ";s" + screen.width + "*" + screen.height + "*" + (screen.colorDepth ? screen.colorDepth : screen.pixelDepth)) + ";u" + escape(document.URL) + ";" + Math.random() + "' alt='' title='LiveInternet' style='border:0;'><\/a>");
|
||||
{# <!-- Yandex.Metrika counter --> #}(function (d, w, c) {
|
||||
(w[c] = w[c] || []).push(function () {
|
||||
try {
|
||||
w.yaCounter32997984 = new Ya.Metrika({
|
||||
id: 32997984,
|
||||
clickmap: true,
|
||||
trackLinks: true,
|
||||
accurateTrackBounce: true,
|
||||
webvisor: true,
|
||||
trackHash: true
|
||||
});
|
||||
} catch (e) {
|
||||
}
|
||||
});
|
||||
var n = d.getElementsByTagName("script")[0], s = d.createElement("script"), f = function () {
|
||||
n.parentNode.insertBefore(s, n);
|
||||
};
|
||||
s.type = "text/javascript";
|
||||
s.async = true;
|
||||
s.src = "https://mc.yandex.ru/metrika/watch.js";
|
||||
if (w.opera == "[object Opera]") {
|
||||
d.addEventListener("DOMContentLoaded", f, false);
|
||||
} else {
|
||||
f();
|
||||
}
|
||||
})(document, window, "yandex_metrika_callbacks");
|
||||
//--></script>
|
||||
{# <!--/LiveInternet--> #}
|
||||
</div>
|
||||
<div style="position: absolute; bottom: 0; left: 0;padding-bottom: 1ex;padding-left: 2ex;font-size: x-small">
|
||||
© <a href="/">oknardia.ru</a>, 2015-{% now "Y" %}.
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</body>
|
||||
</html>
|
||||
@@ -57,7 +57,7 @@ $(window).load(function(){var images = $('.half');images.each(function(i){$(this
|
||||
<li>Размещено 22 оконных набора в предложениях. Размещаются как пластиковые, так и деревянные окна.</li>
|
||||
<li>С «Окнардией» работают уже семь <nobr>оконных-компании</nobr> партнёра.</li>
|
||||
<li>— Добавлен функционал сравнения характеристик оконных предложений и отдельных компонентов этих предложений.</li>
|
||||
<li>Создан <a href="/catalog/profile/">каталог профилей</a>, <a href="/catalog/standard_opening/">стандартных проёмов</a>, <a href="/catalog/seria/">типовых серий домов</a>, <a href="https://oknardia.ru/catalog/company/">оконных компаний</a>. В будущем будет каталог стеклопакетов и фурнитуры… планов много.</li>
|
||||
<li>Создан <a href="/catalog/profile/">каталог профилей</a>, <a href="/catalog/standard_opening/">стандартных проёмов</a>, <a href="/catalog/seria/">типовых серий домов</a>, <a href="{{ request.scheme }}://{{ request.get_host }}/catalog/company/">оконных компаний</a>. В будущем будет каталог стеклопакетов и фурнитуры… планов много.</li>
|
||||
<li>Построен <a href="/blogpost/17/Nagljadnoe-sravnenie-harakteristik-okonnyh-profilej/">алгоритм расчёта реальных рейтингов</a> предложений, профилей, стеклопакетов и сервиса компаний. Рейтинги не на базе «общественного голосования», «опросов» или измерением <nobr>«интернет-популярности»</nobr>, а на базе физических характеристик и измеримых параметров. Таким образом — это объектовые рейтинги.</li>
|
||||
<li>Разработан <a href="https://widget.oknardia.ru/">виджет</a>, который позволяет оконной компании реализовать функционал «Окнардии» на собственном сайте.</li>
|
||||
<li>Расширены <nobr><a href="/blogpost/16/Novye-media-vozmozhnosti-uchastnikam-Oknardii-/">медиа-возможности</a></nobr> проекта: баннеры, посты в блоге, специальное выделение.</li>
|
||||
|
||||
@@ -1,16 +1,20 @@
|
||||
{% extends "base.html" %}{% load static %}
|
||||
|
||||
{% block Title %}Тарифы и услуги{% endblock %}
|
||||
{% block Title %}Тарифы и услуги маркетплейса Окнардия | Цены на размещение предложений окон{% endblock %}
|
||||
|
||||
{% block Add_Body_Attribute %} style="padding-top:70px;"{% endblock %}
|
||||
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Date4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"Y-m-d" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"c" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
{% block Last4Meta %}{% if PUB_DAT %}{{ PUB_DAT|date:"Y-m-d" }}{% else %}{% now "c" %}{% endif %}{% endblock %}
|
||||
|
||||
{% block Description %}Тарифы и услуги маркетплейс-агрегатора Окнардия. Размещение предложений пластиковых и деревянных окон, обновление цен на окна, рекламные баннеры и виджеты на сайт оконной компании.{% endblock %}
|
||||
{% block Description %}Тарифы размещения предложений по установке пластиковых и деревянных окон на маркетплейсе Окнардия. Пять тарифных планов: альфа (бесплатно), бета, дельта, мю (медийный), омега (виджет). Обновление цен, баннеры, публикации в блог, электронные заявки.{% endblock %}
|
||||
|
||||
{% block Keywords %}типовые проекты зданий, панельное строительство, {% for CountSeria in SERIA_NAV_DIM %}серия {{ CountSeria.SERIA_R }}, {{ CountSeria.SERIA_R }}, {% endfor %}, года простойки, регионы построки, распространенность{% endblock %}
|
||||
{% block Keywords %}тарифы окнардия, размещение предложений окон, цены на окна, маркетплейс окон, услуги для оконных компаний, виджет окон, баннеры на сайт, каталог окон, установка окон, продажа пластиковых окон, медийное продвижение окон{% endblock %}
|
||||
|
||||
{% block Author4Meta %}: Тарифы и услуги маркетплейса «Окнардия»{% endblock %}
|
||||
|
||||
{% block CopyrightAuthor4Meta %}: Тарифы маркетплейса «Окнардия»{% endblock %}
|
||||
|
||||
{% block Top_JS1%}
|
||||
<script type="text/javascript">
|
||||
@@ -18,14 +22,288 @@ $(window).load(function(){let images = $('.half');images.each(function(i){$(this
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Top_Meta1 %}{# <!-- BEGIN Дополнительные Metatags --> #}
|
||||
{# Удалить: itemprop microdata, rel=standout, twitter:domain — устаревшие теги #}
|
||||
<meta name="news_keywords" content="тарифы окнардия, размещение окон, маркетплейс окон, услуги оконных компаний, медийное продвижение" />
|
||||
<link rel="canonical" href="{{ request.scheme }}://{{ request.get_host }}/tariff/" />
|
||||
<!-- Разметка для соц-сетей Facebook Open Graph -->
|
||||
<meta property="fb:admins" name="admins" content="100000084781830" />
|
||||
<meta property="fb:pages" content="276108456054163" />
|
||||
<meta property="fb:app_id" content="258354027974262" />
|
||||
<meta property="fb:profile_id" name="profile_id" content="https://www.facebook.com/oknardia/" />
|
||||
<meta property="og:locale" content="ru_RU" />
|
||||
<meta property="og:site_name" content="oknardia.ru" />
|
||||
<meta property="og:url" content="{{ request.scheme }}://{{ request.get_host }}/tariff/" />
|
||||
<meta property="og:type" content="website" />
|
||||
<meta property="og:title" content="Тарифы и услуги маркетплейса Окнардия" />
|
||||
<meta property="og:description" content="Пять тарифных планов для размещения предложений пластиковых и деревянных окон. От альфа (бесплатно) до омега (виджет)." />
|
||||
<meta property="og:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<link rel="image_src" href="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<!-- Разметка для соц-сетей Twitter Card -->
|
||||
<meta name="twitter:title" content="Тарифы и услуги маркетплейса Окнардия" />
|
||||
<meta name="twitter:description" content="Пять тарифных планов для размещения предложений окон на маркетплейсе." />
|
||||
<meta name="twitter:card" content="summary" />
|
||||
<meta name="twitter:site" content="@oknardia" />
|
||||
<meta property="twitter:url" content="{{ request.scheme }}://{{ request.get_host }}/tariff/" />
|
||||
<meta name="twitter:image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
<meta name="relap-image" content="{{ request.scheme }}://{{ request.get_host }}/static/img/MerDY3gpU0w.jpg" />
|
||||
{# <!-- END Дополнительные Metатags --> #}{% endblock %}
|
||||
|
||||
{% block ADD_TO_HEAD %}
|
||||
{# JSON-LD: страница тарифов — CollectionPage + BreadcrumbList + PriceSpecification #}
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "CollectionPage",
|
||||
"name": "Тарифы и услуги маркетплейса Окнардия",
|
||||
"description": "Пять тарифных планов для размещения предложений пластиковых и деревянных окон на маркетплейсе Окнардия с различными услугами и ценами.",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/tariff/",
|
||||
"isPartOf": {
|
||||
"@type": "WebSite",
|
||||
"name": "Окнардия",
|
||||
"url": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
"mainEntity": {
|
||||
"@type": "PriceSpecification",
|
||||
"priceCurrency": "RUB",
|
||||
"name": "Тарифные планы Окнардия",
|
||||
"description": "Пять вариантов сотрудничества: альфа (бесплатно), бета, дельта, мю и омега"
|
||||
},
|
||||
"hasPart": [
|
||||
{
|
||||
"@type": "Offer",
|
||||
"name": "{α} Альфа — Старт",
|
||||
"description": "Размещение цен на установку пластиковых окон. Расширенная информация о компании в каталоге Окнардии.",
|
||||
"price": "0",
|
||||
"priceCurrency": "RUB",
|
||||
"aggregateRating": {
|
||||
"@type": "AggregateRating",
|
||||
"ratingValue": "4",
|
||||
"bestRating": "5",
|
||||
"worstRating": "1",
|
||||
"ratingCount": "100"
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Offer",
|
||||
"name": "{β} Бета — Коммерческий",
|
||||
"description": "Основное размещение цен на установку окон. Два предложения для разных комплектаций, одно обновление в месяц, логотип компании, информация о вашей компании в каталоге.",
|
||||
"price": "300",
|
||||
"priceCurrency": "RUB",
|
||||
"billingIncrement": "P1M",
|
||||
"priceSpecification": {
|
||||
"@type": "PriceSpecification",
|
||||
"price": "1000",
|
||||
"priceCurrency": "RUB",
|
||||
"name": "Стартовая установка"
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Offer",
|
||||
"name": "{δ} Дельта — Продвинутый",
|
||||
"description": "Расширенное размещение цен на установку окон. Двенадцать предложений для полного ассортимента, шесть обновлений в месяц, всплытие в выдаче, логотип компании, публикации в блог.",
|
||||
"price": "1500",
|
||||
"priceCurrency": "RUB",
|
||||
"billingIncrement": "P1M",
|
||||
"priceSpecification": {
|
||||
"@type": "PriceSpecification",
|
||||
"price": "5000",
|
||||
"priceCurrency": "RUB",
|
||||
"name": "Стартовая установка"
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Offer",
|
||||
"name": "{μ} Мю — Медийный",
|
||||
"description": "Медийное продвижение услуг на установку окон. Баннеры в ценовой выдаче, публикации о ваших услугах в блог, ежемесячные аналитические отчёты.",
|
||||
"price": "500",
|
||||
"priceCurrency": "RUB",
|
||||
"billingIncrement": "P1M",
|
||||
"priceSpecification": {
|
||||
"@type": "PriceSpecification",
|
||||
"price": "500",
|
||||
"priceCurrency": "RUB",
|
||||
"name": "Стартовая установка"
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Offer",
|
||||
"name": "{ω} Омега — Виджет",
|
||||
"description": "Полный пакет для продвижения услуг на установку окон. Виджет на ваш сайт, восемнадцать предложений на установку, логотип компании, публикации в блог, все функции плана Дельта.",
|
||||
"price": "9000",
|
||||
"priceCurrency": "RUB",
|
||||
"billingIncrement": "P1M",
|
||||
"priceSpecification": {
|
||||
"@type": "PriceSpecification",
|
||||
"price": "45000",
|
||||
"priceCurrency": "RUB",
|
||||
"name": "Стартовая установка"
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "BreadcrumbList",
|
||||
"itemListElement": [
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 1,
|
||||
"name": "Главная",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/"
|
||||
},
|
||||
{
|
||||
"@type": "ListItem",
|
||||
"position": 2,
|
||||
"name": "Тарифы",
|
||||
"item": "{{ request.scheme }}://{{ request.get_host }}/tariff/"
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
<script type="application/ld+json">
|
||||
{
|
||||
"@context": "https://schema.org",
|
||||
"@type": "FAQPage",
|
||||
"name": "FAQ: Тарифы и услуги маркетплейса Окнардия",
|
||||
"description": "Ответы на часто задаваемые вопросы о тарифах, услугах и возможностях размещения предложений на маркетплейсе Окнардия",
|
||||
"mainEntity": [
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Что такое маркетплейс Окнардия?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Окнардия — это агрегатор (маркетплейс) для сравнения цен на установку пластиковых и деревянных окон в зданиях типового строительства в России. Пользователи указывают адрес дома, система распознаёт серию здания и выдаёт типовые размеры оконных проёмов, а затем показывает предложения от поставщиков на установку и замену окон."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Как выбрать подходящий тариф для моей компании?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Выбор тариф зависит от ваших целей: Альфа (бесплатно) — для тестирования; Бета (300₽/мес) — для базового размещения с логотипом; Дельта (1500₽/мес) — для расширенного каталога и публикаций; Мю (500₽/мес) — для медийного продвижения с баннерами; Омега (9000₽/мес) — полный пакет с виджетом на сайт рекламодателя."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Какой SEO-эффект даёт размещение на Окнардии?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Логотип вашей компании в ценовой выдаче содержит ссылку на ваш сайт — это мощный источник качественных внешних ссылок (backlinks), что положительно влияет на ранжирование вашего сайта в поисковых системах. Публикации в блог «Окнардии» также способствуют SEO за счёт релевантного контента и внутридомовых ссылок."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Что такое виджет Окнардия и зачем он нужен?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "«Виджет. ОКНАРДИЯ» — это встраиваемый фрейм-блок (iframe), который устанавливается на сайт поставщика. Виджет позволяет посетителям указать адрес дома и выбрать квартиру, после чего видят типовые размеры проёмов, схемы открывания и ваши предложения (наборы окон) с ценами — всё прямо на вашем сайте, без перехода на Окнардию."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Как работает система «всплытия» (поднятия) предложений?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "«Всплытие» — это возможность поднять ваши ценовые предложения в приоритетный блок выдачи. Гарантируется присутствие в блоке (позиция может варьироваться из-за сортировки по удаленности офиса от адреса клиента, но видимость гарантирована). Всплытие также способствует попаданию предложений в Rich Snippet'ы и виджеты поисковиков (Google и Яндекс). Частые обновления и всплытия заставляют поисковиков чаще переиндексировать ваши данные, что повышает вероятность появления в снипетах. На плане Дельта доступно 8 всплытий в месяц, на Омега — тоже 8."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Можно ли обновлять цены и описания в течение месяца?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Да, каждый тариф предусматривает определённое количество обновлений в месяц. Альфа — ⅓ обновления, Бета — 1 обновление, Дельта — 6 обновлений, Мю — нет обновлений (только баннеры и публикации), Омега — 8 обновлений. Обновления применяются ко всем проёмам и ценам выбранного набора."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Что входит в стартовую установку (запуск)?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Стартовая установка — это единовременный платёж за подготовку и размещение вашей первой ценовой информации, настройку профиля компании, загрузку логотипа и описаний. Стоимость варьируется от 0₽ (Альфа) до 45 000₽ (Омега). После включения тариф переходит на ежемесячное взимание."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Есть ли скидки при оплате на год вперёд?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Да, при оплате годовым авансом предусмотрены скидки: Бета -30%, Дельта -40%, Мю -20%, Омега -35%. Авансовый тариф за год включает запуск и установку, что позволяет сэкономить значительную сумму при планировании долгосрочного сотрудничества."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Как долго эффект от размещения логотипа с ссылкой?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Логотип с ссылкой на ваш сайт работает пока ваша подписка активна, плюс сохраняет SEO-эффект ещё примерно 6 месяцев после завершения подписки (кеширование поисковыми системами и остаточная ценность backlinks)."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Какой SEO-эффект от публикаций в блоге Окнардии?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Публикации в блог «Окнардии» — это постоянный источник SEO-трафика. Ссылки на ваш сайт остаются на сайте навсегда и продолжают работать даже после завершения подписки — это постоянная ценность в SEO. Публикации отлично подходят для продвижения акций и скидок на установку окон. На планах Дельта и Омега публикации включены; на плане Мю также доступны публикации. Каждая статья может содержать до 25 000 знаков, иллюстрации, видео и анимацию."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Какой эффект от баннеров на Окнардии?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Баннеры имеют мощный престижный эффект: ваша компания отображается между ценовыми предложениями и в каталоге, позиционируя вас как ведущего поставщика. Можно практически забрендировать весь сайт так, что он будет восприниматься как собственный проект компании. Плюс баннеры содержат прямые HTML-ссылки, которые поисковики индексируют — это качественные backlinks для SEO."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Что такое виджет Окнардия и как его установить?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Виджет — это встраиваемый фрейм-блок (iframe) для вашего сайта. Посетители указывают адрес дома и выбирают квартиру, видят размеры проёмов и ваши предложения прямо на вашем сайте. Это повышает конверсию и удержание клиента. Примеры и инструкции на widget.oknardia.ru"
|
||||
}
|
||||
},
|
||||
{
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Проект «Окнардия» гибкий и открыт к сотрудничеству. Если вас интересуют специальные условия, кастомные решения, расширенные возможности или нестандартные формы партнёрства, свяжитесь с командой через форму обратной связи. Мы обсуждаем любые предложения: универсальные калькуляторы окон, специализированные виджеты, интеграцию ваших систем, генераторы смет и прейскурантов, и многое другое."
|
||||
}
|
||||
},
|
||||
{
|
||||
"@type": "Question",
|
||||
"name": "Какие типовые размеры окон покрывает Окнардия?",
|
||||
"acceptedAnswer": {
|
||||
"@type": "Answer",
|
||||
"text": "Размеры окон в каталоге зависят от серии типового строительства. Система распознаёт серию по адресу и выдаёт все типовые раскладки проёмов и схемы открывания для данной серии (например, П-44, 5-этажка, кирпичный). На каждый набор вы можете разместить предложения под разные комплектации: профили, стеклопакеты, фурнитуру, варианты монтажа и отделки."
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block Main_Content %}<div class="container-fluid">
|
||||
{# Хлебные крошки: НАЧАЛО #}
|
||||
<div class="row">
|
||||
<div class="col-md-11 col-xs-12">
|
||||
<ol class="breadcrumb">
|
||||
<li><a href="/">Главная</a></li>
|
||||
<li class="active">Тарифы</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
{# Хлебные крошки: КОНЕЦ #}
|
||||
|
||||
<div class="row">
|
||||
<div class="col-md-9 col-md-offset-1 col-xs-12"><h1>Направления сотрудничества с «Окнардия» и тарифы</h1></div>
|
||||
</div>
|
||||
|
||||
<div class="row">
|
||||
<div class="col-md-10 col-md-offset-1 col-xs-12 tariff">
|
||||
{% if SENDER %}{% if SENDER == "Ok!" %}<p style="background: lightgreen;">Спасибо за ваше обращение. Мы обязательно свяжемся с вами.</p>{% elif SENDER == "Error!" %}<p style="background: lightsalmon;">Что-то пошло не так. Не удалось отправить e-mail. Попробуйте еще раз или используте для связи info@oknardia.ru </p>{% endif %}{% endif %}
|
||||
{% if SENDER %}{% if SENDER == "Ok!" %}<p style="background: lightgreen;">Спасибо за ваше обращение. Мы обязательно свяжемся с вами.</p>{% elif SENDER == "Error!" %}<p style="background: lightsalmon;">Что-то пошло не так. Не удалось отправить e-mail. Попробуйте еще раз или используете для связи info@oknardia.ru </p>{% endif %}{% endif %}
|
||||
<table>
|
||||
<thead>
|
||||
<tr>
|
||||
@@ -216,12 +494,318 @@ $(window).load(function(){let images = $('.half');images.each(function(i){$(this
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
<P> </P>
|
||||
<P>Мы открыты к сотрудничеству. Много идей в наших планах (в том числе универсальный калькулятор окон, виджет калькулятора, генератор смет и прейскурантов и еще много чего) и если у вас есть потребность в дополнительных разработках для рынка окон, будем рады обсудить возможные формы сотрудничества.</P>
|
||||
</div>
|
||||
<P> </P>
|
||||
<P>Мы открыты к сотрудничеству. Много идей в наших планах (в том числе универсальный калькулятор окон, виджет калькулятора, генератор смет и прейскурантов и еще много чего) и если у вас есть потребность в дополнительных разработках для рынка окон, будем рады обсудить возможные формы сотрудничества.</P>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Модальное окно обратной связи -->
|
||||
{# FAQ секция для SEO: НАЧАЛО #}
|
||||
<div class="row" style="margin-top: 60px; margin-bottom: 40px;">
|
||||
<div class="col-md-10 col-md-offset-1 col-xs-12">
|
||||
<h2 style="margin-bottom: 30px;">Часто задаваемые вопросы о тарифах и услугах</h2>
|
||||
|
||||
<div class="panel-group" id="accordion" role="tablist" aria-multiselectable="true" style="font-size: 120%;">
|
||||
|
||||
{# Вопрос 1: Что такое Окнардия #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" role="tab" id="headingOne">
|
||||
<h4 class="panel-title">
|
||||
<a role="button" data-toggle="collapse" data-parent="#accordion" href="#collapseOne" aria-expanded="true"
|
||||
aria-controls="collapseOne">
|
||||
Что такое маркетплейс Окнардия?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div id="collapseOne" class="panel-collapse collapse in" role="tabpanel" aria-labelledby="headingOne">
|
||||
<div class="panel-body">
|
||||
Окнардия — это агрегатор (маркетплейс) для сравнения цен на установку пластиковых и деревянных
|
||||
окон в зданиях типового строи­тельства России. Пользователи указывают адрес дома, система
|
||||
распознаёт серию здания и выдаёт типовые размеры оконных проёмов, а затем показывает предложения
|
||||
от поставщиков на установку и замену окон.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 2: Выбор тарифа #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingTwo" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseTwo" aria-expanded="true" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseTwo" role="button">
|
||||
Как выбрать подходящий тариф для моей компании?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="headingTwo" class="panel-collapse collapse in" id="collapseTwo" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
Выбор тарифа зависит от ваших целей:
|
||||
<ul style="margin-top: 10px;">
|
||||
<li><strong>Альфа (бесплатно)</strong> — для тестирования и ознакомления с платформой;
|
||||
</li>
|
||||
<li><strong>Бета (300₽/мес)</strong> — для базового размещения с логотипом и двумя
|
||||
предло­жениями;
|
||||
</li>
|
||||
<li><strong>Дельта (1500₽/мес)</strong> — для расширенного каталога с 12 наборами,
|
||||
публикациями и всплытиями;
|
||||
</li>
|
||||
<li><strong>Мю (500₽/мес)</strong> — для медийного продвижения с баннерами и публикациями
|
||||
в блог;
|
||||
</li>
|
||||
<li><strong>Омега (9000₽/мес)</strong> — полный пакет с виджетом на ваш сайт и максимум
|
||||
возможностей.
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 3: SEO эффект #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingThree" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseThree" aria-expanded="true" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseThree" role="button">
|
||||
Какой SEO-эффект даёт размещение на Окнардии?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="collapseThree" class="panel-collapse collapse in" id="collapseThree" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
Логотип вашей компании в ценовой выдаче содержит ссылку на ваш сайт — это мощный
|
||||
источник качественных внешних ссылок (backlinks), что положительно влияет на ранжирование вашего сайта
|
||||
в поисковых системах. Публикации в блог «Окнардии» также способствуют SEO за счёт
|
||||
релевантного контента и внутренних ссылок. SEO-эффект сохраняется пока ваша подписка активна, плюс ещё
|
||||
примерно шесть месяцев после завершения (кеширование и остаточная ценность ссылок).
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 4: Всплытия #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingFour" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseFour" aria-expanded="false" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseFour" role="button">
|
||||
Как работает система «всплытия» (поднятия) предложений в выдаче?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="collapseFour" class="panel-collapse collapse" id="collapseFour" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
«Всплытие» гарантирует присутствие ваших предложений в приоритетном блоке выдачи и в Rich
|
||||
Snippet'ах (виджеты, острова) поисковиков. Позиция может варьи­роваться из-за сортировки по удалённости
|
||||
офиса от адреса клиента, но видимость в блоке гаранти­рована. Важный бонус: поисковики
|
||||
интенсивнее переинде­ксируют предложения с частыми обновлениями, что повышает вероятность появления
|
||||
в поисковых сниппетах и медийных виджетах (Google и Яндекс). На плане Дельта доступно
|
||||
восемь всплытий в месяц, на Омега — тоже восемь.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 5: Обновления цен #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingSix" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseSix" aria-expanded="false" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseSix" role="button">
|
||||
Можно ли обновлять цены и описания в течение месяца?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="headingSix" class="panel-collapse collapse" id="collapseSix" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
Да, каждый тариф предусма­тривает определённое количество обновлений в месяц. Обновления
|
||||
применяются ко всем проёмам и ценам выбранного набора:
|
||||
<ul style="margin-top: 10px;">
|
||||
<li><strong>Альфа</strong> — ⅓ обновления (в среднем один раз в три месяца);</li>
|
||||
<li><strong>Бета</strong> — одно обновление в месяц;</li>
|
||||
<li><strong>Дельта</strong> — шесть обновлений в месяц;</li>
|
||||
<li><strong>Мю</strong> — нет обновлений (фокус на медийном продвижении и баннерах);
|
||||
</li>
|
||||
<li><strong>Омега</strong> — восемь обновлений в месяц.</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 7: Стартовая установка #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingSeven" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseSeven" aria-expanded="false" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseSeven" role="button">
|
||||
Что входит в стартовую установку (запуск)?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="headingSeven" class="panel-collapse collapse" id="collapseSeven" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
Стартовая установка — это единовре­менный платёж за подготовку и размещение вашей первой
|
||||
ценовой информации, настройку профиля компании, загрузку логотипа, описания компании и ваших
|
||||
наборов. Стоимость варьируется от 0₽ (Альфа) до 45 000₽ (Омега) в зависимости от сложности
|
||||
и объёма работ. После включения тариф переходит на ежемесячное взимание по устано­вленной
|
||||
цене.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 8: Скидки за год #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingEight" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseEight" aria-expanded="false" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseEight" role="button">
|
||||
Есть ли скидки при оплате на год вперёд?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="headingEight" class="panel-collapse collapse" id="collapseEight" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
Да, при оплате годовым авансом преду­смотрены щедрые скидки:
|
||||
<ul style="margin-top: 10px;">
|
||||
<li><strong>Бета</strong> — 30% скидка (2 520₽ вместо 3 600₽);</li>
|
||||
<li><strong>Дельта</strong> — 40% скидка (13 800₽ вместо 23 000₽);</li>
|
||||
<li><strong>Мю</strong> — 20% скидка (5 200₽ вместо 6 500₽);</li>
|
||||
<li><strong>Омега</strong> — 35% скидка (70 200₽ вместо 108 000₽).</li>
|
||||
</ul>
|
||||
Авансовый платёж включает стартовую установку и полный год взимания ежемесячного тарифа.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 9: Длительность эффекта #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingNine" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseNine" aria-expanded="false" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseNine" role="button">
|
||||
Как долго эффект от размещения логотипа с ссылкой сохраняется?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="headingNine" class="panel-collapse collapse" id="collapseNine" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
Логотип с ссылкой на ваш сайт работает пока ваша подписка активна. После завершения подписки
|
||||
SEO-эффект продолжает сохраняться примерно 6 месяцев благодаря кешированию поисковыми системами и остаточной
|
||||
ценности внешних ссылок. Это означает, что даже если вы временно прекратили размещение, ваш сайт
|
||||
получает положи­тельный эффект от ссылок ещё полгода.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 10: Публикации в блоге #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingNine" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseNine" aria-expanded="false" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseNine" role="button">
|
||||
Как долго эффект от размещения логотипа с ссылкой сохраняется?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="headingNine" class="panel-collapse collapse" id="collapseNine" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
Логотип с ссылкой на ваш сайт работает пока ваша подписка активна. После завершения подписки
|
||||
SEO-эффект продолжает сохраняться примерно шесть месяцев благодаря кешированию поисковыми системами
|
||||
и остаточной ценности внешних ссылок. Это означает, что даже если вы временно прекратили
|
||||
размещение, ваш сайт получает положи­тельный эффект от ссылок ещё полгода.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 11: Баннеры и SEO #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingEleven" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseEleven" aria-expanded="false" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseEleven" role="button">
|
||||
Какой SEO-эффект от баннеров на Окнардии? Можно ли ими брендировать?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="collapseEleven" class="panel-collapse collapse" id="collapseEleven" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
Баннеры имеют мощный престижный эффект: ваша компания отображается между блоками ценовых предложений и на страницах
|
||||
каталога, украшая платформу и позиционируя вас как ведущего поставщика. Благодаря баннерам можно
|
||||
практически «забренди­ровать» весь сайт так, что он будет воспри­ниматься почти как собственный
|
||||
проект вашей компании. Баннеры размером 100% × 175px могут быть графическими, видео или HTML с CSS/JS
|
||||
анимацией. Доступны на плане Мю (медийный). Кроме того, баннеры на Окнардии содержат прямые
|
||||
HTML-ссылки (без редиректов и JavaScript), что позволяет поисковикам их полностью индекси­ровать.
|
||||
Каждый баннер — это полноценный backlink на ваш сайт, дающий SEO-эффект.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 12: Виджет #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingTwelve" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseTwelve" aria-expanded="false" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseTwelve" role="button">
|
||||
Что такое виджет Окнардия и как его установить на сайт?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="headingTwelve" class="panel-collapse collapse" id="collapseTwelve" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
«Виджет. ОКНАРДИЯ» — это встраиваемый фрейм-блок (iframe), который устана­вливается на сайт
|
||||
поставщика окон. Виджет позволяет посетителям указать адрес дома и выбрать квартиру, после чего видят
|
||||
типовые размеры проёмов, схемы открывания и ваши предложения (наборы окон) с ценами — всё
|
||||
прямо на вашем сайте. Это повышает конверсию и удержание клиента. Пример и инструкции по установке
|
||||
доступны на <a href="https://widget.oknardia.ru/" target="_blank">widget.oknardia.ru</a> (примечание:
|
||||
сайт может быть временно недоступен).
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 13: Специальные пожелания и гибкость #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingThirteen" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseThirteen" aria-expanded="false" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseThirteen" role="button">
|
||||
Что делать, если есть специальные пожелания или неста­ндартные условия?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="collapseThirteen" class="panel-collapse collapse" id="collapseThirteen" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
«Окнардия» — гибкий и открытый к сотрудни­честву проект. Если вас интересуют специальные условия,
|
||||
кастомные решения, расширенные возможности или неста­ндартные формы партнёрства, <a href="/contact/">свяжитесь
|
||||
с командой проекта</a>. Мы обсуждаем любые предложения: униве­рсальные калькуляторы окон,
|
||||
специали­зированные виджеты, интеграция ваших систем, генераторы смет и прейску­рантов, анали­тические
|
||||
отчёты и многое другое. Ваши идеи и пожелания — важная часть развития платформы.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# Вопрос 14: Типовые размеры #}
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading" id="headingFourteen" role="tab">
|
||||
<h4 class="panel-title">
|
||||
<a aria-controls="collapseFourteen" aria-expanded="false" data-parent="#accordion" data-toggle="collapse"
|
||||
href="#collapseFourteen" role="button">
|
||||
Какие типовые размеры окон покрывает Окнардия?
|
||||
</a>
|
||||
</h4>
|
||||
</div>
|
||||
<div aria-labelledby="headingFourteen" class="panel-collapse collapse" id="collapseFourteen" role="tabpanel">
|
||||
<div class="panel-body">
|
||||
Размеры окон в каталоге зависят от серии типового строи­тельства. Система распознаёт серию по адресу
|
||||
и выдаёт все типовые раскладки проёмов и схемы открывания для данной серии (например, П-44,
|
||||
5-этажка, кирпичный и т. п.). На каждый набор наборов окон вы можете разместить
|
||||
предложения под разные комплектации: разные профили, стеклопакеты, фурнитуру, варианты монтажа и отделки.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</div>
|
||||
{# FAQ секция для SEO: КОНЕЦ #}
|
||||
|
||||
|
||||
<div class="modal fade" id="Form_for_Feedback" tabindex="-1" role="dialog" aria-labelledby="myModalLabel">
|
||||
<div class="modal-dialog" role="document">
|
||||
<div class="modal-content">
|
||||
|
||||
@@ -11,8 +11,8 @@
|
||||
LOGIN-LOGOUT на отдельный сервер.
|
||||
|
||||
Даннеа Google reCAPTCHA: https://www.google.com/recaptcha/admin#site/319090428?setup
|
||||
Публичный Ключ: 6Lf87gQTAAAAALmkG5ZsO0eJSvdSXcRvkxoPJCDB
|
||||
Секретный ключ: 6Lf87gQTAAAAADlqsJQToiWqg7urOWPrbfG_9zJB
|
||||
Публичный Ключ: cм. `.env`
|
||||
Секретный ключ: cм. `.env`
|
||||
|
||||
{% endcomment %}
|
||||
<script type="text/javascript">
|
||||
|
||||
@@ -3,73 +3,117 @@ __author__ = 'Sergei Erjemin'
|
||||
|
||||
from PIL import Image, ImageDraw
|
||||
from oknardia.settings import *
|
||||
import django.utils.dateformat
|
||||
import django.utils.timezone
|
||||
from pytils.translit import slugify, translify
|
||||
import os
|
||||
import math
|
||||
import re
|
||||
import html
|
||||
import urllib3
|
||||
import xml.dom.minidom
|
||||
|
||||
|
||||
def safe_html_spec_symbols(s: str) -> str:
|
||||
""" Очистка строки от HTML-разметки типографа
|
||||
""" Очистка строки от HTML-разметки и получение чистого текста.
|
||||
|
||||
:param s: str -- строка которую надо очистить
|
||||
:return: str: str -- очищенная строка
|
||||
Функция удаляет HTML-теги, содержимое исключённых тегов (script, style, object, embed, applet,
|
||||
iframe, svg, canvas, code, kbd, pre, var, samp, output, noscript, link, meta, form, input,
|
||||
button, textarea, select, base, title, head, body, track, source, picture), заменяет HTML-мнемоники
|
||||
на Unicode-символы и убирает лишние пробелы.
|
||||
|
||||
:param s: str -- строка которую надо очистить
|
||||
:return: str -- очищенная строка с чистым текстом
|
||||
"""
|
||||
# очистка строки от некоторых спец-символов HTML
|
||||
result = s.replace('­', '')
|
||||
result = result.replace('<span class="laquo">', '')
|
||||
result = result.replace('<span style="margin-right:0.44em;">', '')
|
||||
result = result.replace('<span style="margin-left:-0.44em;">', '')
|
||||
result = result.replace('<span class="raquo">', '')
|
||||
result = result.replace('<span class="point">', '')
|
||||
result = result.replace('<span class="thinsp">', ' ')
|
||||
result = result.replace('<span class="ensp">', '')
|
||||
result = result.replace('</span>', '')
|
||||
result = result.replace(' ', ' ')
|
||||
result = result.replace('«', '«')
|
||||
result = result.replace('»', '»')
|
||||
result = result.replace('…', '…')
|
||||
result = result.replace('<nobr>', '')
|
||||
result = result.replace('</nobr>', '')
|
||||
result = result.replace('—', '—')
|
||||
result = result.replace('№', '№')
|
||||
result = result.replace('<br />', ' ')
|
||||
result = result.replace('<br>', ' ')
|
||||
# Шаг 1: Удаляем содержимое "опасных" и невидимых тегов
|
||||
# Опасные: script, object, embed, applet, iframe, svg, canvas
|
||||
# Техническое содержимое: style, code, kbd, pre, var, samp, output, noscript
|
||||
# Формы: form, input, button, textarea, select
|
||||
# Служебные: meta, link, base, title, head, body, track, source, picture
|
||||
# Используем флаг IGNORECASE и DOTALL для работы с многострочным контентом
|
||||
result = re.sub(
|
||||
r'<(script|style|code|kbd|pre|var|samp|output|noscript|link|meta|iframe|object|embed|applet|form|input|button|textarea|select|svg|canvas|base|title|head|body|track|source|picture)(?:\s[^>]*)?>.*?</\1>',
|
||||
'',
|
||||
s,
|
||||
flags=re.IGNORECASE | re.DOTALL
|
||||
)
|
||||
|
||||
# Удаляем самозакрывающиеся теги (что-то типа <input/>, <embed/>, и т.д.)
|
||||
result = re.sub(
|
||||
r'<(input|embed|meta|link|base|track|source|img)(?:\s[^>]*)?/>',
|
||||
'',
|
||||
result,
|
||||
flags=re.IGNORECASE
|
||||
)
|
||||
|
||||
# Шаг 2: Удаляем все остальные HTML-теги (в т.ч. самозакрывающиеся)
|
||||
result = re.sub(r'<[^>]+>', '', result)
|
||||
|
||||
# Шаг 3: Заменяем HTML-мнемоники на Unicode-символы (включая числовые и именованные)
|
||||
# html.unescape() обрабатывает: , <, №, € и т.д.
|
||||
result = html.unescape(result)
|
||||
|
||||
# Шаг 4: Очищаем множественные пробелы (в т.ч. табуляцию и переводы строк)
|
||||
result = re.sub(r'\s+', ' ', result)
|
||||
|
||||
# Шаг 5: Убираем пробелы в начале и конце строки
|
||||
result = result.strip()
|
||||
|
||||
return result
|
||||
|
||||
|
||||
# def Rus2Lat(RusString):
|
||||
# return translit(re.sub(
|
||||
# r'<[\s\S]*?>', '', re.sub(r'&[\S]*?;', '-', RusString)
|
||||
# ), "ru", reversed=True).replace(u" ", u"-").replace(u"'", u"").replace(u"/", u"~").replace(u"\\", u"~").replace(u"--", u"-")
|
||||
def sanitize_slug(text: str, separator: str = '-', max_length: int = 200) -> str:
|
||||
""" Преобразует текст в URL-безопасный слаг (slug).
|
||||
|
||||
Функция очищает текст от HTML-разметки, выполняет транслитерацию русского текста в
|
||||
латиницу, заменяет пробелы и недопустимые символы на разделитель (по умолчанию дефис),
|
||||
и возвращает готовый к использованию в URL слаг.
|
||||
|
||||
Этапы обработки:
|
||||
1. Очистка от HTML-разметки через safe_html_spec_symbols()
|
||||
2. Транслитерация русского текста в латиницу через pytils.translit.slugify()
|
||||
3. Замена множественных разделителей на один
|
||||
4. Удаление разделителя в начале и конце
|
||||
5. Прерывание на max_length символов
|
||||
|
||||
:param text: str -- исходный текст, может содержать HTML и русский текст
|
||||
:param separator: str -- разделитель для слага (по умолчанию дефис '-')
|
||||
pytils.slugify() всегда использует дефис, этот параметр
|
||||
конвертирует результат в нужный разделитель
|
||||
:param max_length: int -- максимальная длина слага в символах (по умолчанию 200)
|
||||
:return: str -- очищенный и готовый к использованию слаг
|
||||
|
||||
Примеры:
|
||||
>>> sanitize_slug(' Тест — HTML <b>текст</b> ')
|
||||
'test-html-tekst'
|
||||
>>> sanitize_slug('Привет мир!!! @#$')
|
||||
'privet-mir'
|
||||
>>> sanitize_slug('<p>Русский текст в слаге</p>')
|
||||
'russkii-tekst-v-slage'
|
||||
>>> sanitize_slug('Проверка_слага', separator='_')
|
||||
'proverka_slaga'
|
||||
"""
|
||||
# Шаг 1: Очищаем от HTML и мнемоник, убираем лишние пробелы
|
||||
cleaned = safe_html_spec_symbols(text)
|
||||
|
||||
# Шаг 2: Транслитерируем русский текст в латиницу (pytils.slugify использует дефис)
|
||||
slug = slugify(cleaned)
|
||||
|
||||
# Шаг 3: Конвертируем разделитель если нужен другой (не дефис)
|
||||
if separator != '-':
|
||||
slug = slug.replace('-', separator)
|
||||
|
||||
# Шаг 4: Убираем множественные разделители (например, '---' -> '-')
|
||||
slug = re.sub(f'{re.escape(separator)}+', separator, slug)
|
||||
|
||||
# Шаг 5: Убираем разделитель в начале и конце если он есть
|
||||
slug = slug.strip(separator)
|
||||
|
||||
# Шаг 6: Обрезаем по max_length если нужно (и убираем разделитель в конце)
|
||||
if max_length and len(slug) > max_length:
|
||||
slug = slug[:max_length].rstrip(separator)
|
||||
|
||||
return slug.lower()
|
||||
|
||||
|
||||
# def Rus2Url (RusString):
|
||||
# return re.sub(r'^-|-$', '',
|
||||
# re.sub(r'-{1,}', '-',
|
||||
# re.sub(r'<[\s\S]*?>|&[\S]*?;|[\W]', '-',
|
||||
# re.sub(r'\+', '-plus', translit(RusString, "ru", reversed=True))
|
||||
# )
|
||||
# )
|
||||
# ).lower()
|
||||
#
|
||||
#
|
||||
# # Суммирует все цифры в строке через произвольные (не цифровые) разделители
|
||||
# def sum_through(string_w_slash):
|
||||
# string_w_slash = re.sub( r"[^0-9]", u",", string_w_slash)
|
||||
# ListTerms = string_w_slash.split(u',')
|
||||
# Summ = 0
|
||||
# for Count in ListTerms:
|
||||
# try:
|
||||
# Summ += int(Count)
|
||||
# except:
|
||||
# pass
|
||||
# return Summ
|
||||
#
|
||||
#
|
||||
def get_rating_set_for_stars(rating: float = 0.) -> list:
|
||||
""" Возвращает массив 1 и 0 для отрисовки звёздочек.
|
||||
|
||||
@@ -86,24 +130,13 @@ def get_rating_set_for_stars(rating: float = 0.) -> list:
|
||||
rating_set.append(0)
|
||||
return rating_set
|
||||
|
||||
|
||||
#
|
||||
#
|
||||
# # рассчитывает дистанцию в км. между двумя геокоординатами
|
||||
# def get_geo_distance(lon1, lat1, lat2, lon2):
|
||||
# lonA, latA, latB, lonB = map(math.radians, [lon1, lat1, lat2, lon2])
|
||||
# distance = 2 * math.asin(math.sqrt(math.sin((latB - latA) / 2) ** 2 + math.cos(latA) * math.cos(latB) * math.sin(
|
||||
# (lonB - lonA) / 2) ** 2)) * 6371.032 # РАДИУС ЗЕМЛИ 6371.032 КМ.
|
||||
# return distance
|
||||
|
||||
|
||||
def normalize(val: float, val_max: int = 5, val_min: int = 0) -> float:
|
||||
def normalize(val: float, val_max: float = 5.0, val_min: float = 0.0) -> float:
|
||||
""" Нормализация значения
|
||||
|
||||
:param val: float -- значение которое надо нормализовать
|
||||
:param val_max: int -- максимальное значение в нормализуемом диапазоне
|
||||
:param val_min: int -- минимальное значение в нормализуемом диапазоне
|
||||
:return: float: float -- нормализованное значение
|
||||
:param val_max: float -- максимальное значение в нормализуемом диапазоне
|
||||
:param val_min: float -- минимальное значение в нормализуемом диапазоне
|
||||
:return: float -- нормализованное значение
|
||||
"""
|
||||
return float(val - val_min) / float(val_max - val_min)
|
||||
|
||||
@@ -201,7 +234,7 @@ def make_big_img_win_flap(img_file_name_with_path: str, width: int, height: int,
|
||||
# height_door = int(height_door)
|
||||
# создаем картинку с нужными размерами
|
||||
img = Image.new("RGBA", (int(width * PICT_H / height_max), PICT_H), (255, 255, 255, 0))
|
||||
print(img_file_name_with_path)
|
||||
# print(img_file_name_with_path)
|
||||
# находим крайние точки периметра (если окно -- выравнено вверх; если дверь -- вниз)
|
||||
top = 0
|
||||
left = 0
|
||||
@@ -586,11 +619,4 @@ def sum_through(string_w_slash: str) -> int:
|
||||
return sum_result
|
||||
|
||||
|
||||
def touch_reload_wsgi(s: str = ''):
|
||||
""" Функция перезагружает WSGI-сервер.
|
||||
|
||||
:return: None
|
||||
"""
|
||||
with open(TOUCH_RELOAD, 'a', encoding="utf-8") as f:
|
||||
f.write(f'\nreload wsgi by cash-template {s}'
|
||||
f' {django.utils.dateformat.format(django.utils.timezone.now(), "Y-m-d H:i:s")}')
|
||||
# Удалить: touch_reload_wsgi() — серверный reload теперь оркестрируется внешним процесс-менеджером.
|
||||
|
||||
@@ -1,44 +1,57 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
__author__ = 'Sergei Erjemin'
|
||||
|
||||
from django.http import HttpResponse
|
||||
from django.shortcuts import HttpResponseRedirect
|
||||
from django.http import HttpRequest, HttpResponse
|
||||
from django.http import HttpResponse, HttpResponseRedirect
|
||||
from django.http import HttpRequest
|
||||
from oknardia.models import Building_Info
|
||||
# from time import clock
|
||||
import json
|
||||
import re
|
||||
import urllib
|
||||
|
||||
|
||||
def autocomplete_addr(request: HttpRequest) -> HttpResponse:
|
||||
""" Функция для автозаполнения формы выбора адреса. Получает методом GET переменную "term" и по ее образцу
|
||||
ищет доступные адреса в базе адреса из таблицы Building_Info
|
||||
"""Функция для автозаполнения формы выбора адреса.
|
||||
|
||||
Получает методом GET переменную "term" и по её образцу ищет доступные адреса
|
||||
в таблице Building_Info. Результаты возвращаются в JSON формате для jQuery UI.
|
||||
|
||||
:param request: входящий http-запрос
|
||||
:return response: исходящий http-ответ
|
||||
:return: JSON ответ с массивом адресов или редирект на главную
|
||||
"""
|
||||
# Для автозаполнения используется JQuery_UI: http://jqueryui.com/
|
||||
# Пример и инструкции по использованию: http://professorweb.ru/my/javascript/jquery/level4/4_5.php
|
||||
#
|
||||
# ВНИМАНИЕ ТЕХНИЧЕСКИЙ ДОЛГ,: Более навороченный, по описанию лучше подходящий компонент автозаполнения
|
||||
# https://www.devbridge.com/sourcery/components/jquery-autocomplete/ не заработал. Ну и хрен с ним!
|
||||
#
|
||||
# ВНИМАНИЕ ТЕХНИЧЕСКИЙ ДОЛГ: возможен "перегрев" при частом обращении -- [Errno 10053]
|
||||
# Предположительно из-за отсутсвия csrfmiddlewaretoken-серилизации Django. Проблема пофикусена(?) 2014-11-14
|
||||
# tStart = clock()
|
||||
if request.method == 'GET' and 'term' in request.GET:
|
||||
part_blocks = re.split(r"[,/;\s.\\:]+", str(request.GET['term']))
|
||||
if request.GET['use_filter'] == "only_known":
|
||||
q_autocomplete = Building_Info.objects.filter(kSeria_Link__kRoot_id__isnull=False)
|
||||
else:
|
||||
q_autocomplete = Building_Info.objects
|
||||
for i in part_blocks:
|
||||
q_autocomplete = q_autocomplete.filter(sAddress__icontains=i)
|
||||
q_autocomplete = q_autocomplete.all().order_by('sAddress')
|
||||
to_response = ""
|
||||
for i in q_autocomplete[:10]:
|
||||
to_response += '"' + i.sAddress + u'",'
|
||||
to_response = '[' + to_response[0:-1] + ']' # Убираем последнюю запятую
|
||||
return HttpResponse(to_response)
|
||||
else:
|
||||
if request.method != 'GET' or 'term' not in request.GET:
|
||||
return HttpResponseRedirect("/")
|
||||
|
||||
# Получаем поисковый термин и очищаем его
|
||||
search_term = str(request.GET.get('term', '')).strip()
|
||||
if not search_term:
|
||||
return HttpResponse('[]', content_type='application/json')
|
||||
|
||||
# Разбиваем на части для поиска по компонентам адреса (город, улица, номер)
|
||||
part_blocks = re.split(r"[,/;\s.\\:]+", search_term)
|
||||
part_blocks = [p.strip().lower() for p in part_blocks if p.strip()] # Приводим к нижнему регистру
|
||||
|
||||
# Начинаем с базового набора или фильтруем только по известным сериям
|
||||
if request.GET.get('use_filter') == "only_known":
|
||||
q_autocomplete = Building_Info.objects.filter(
|
||||
kSeria_Link__kRoot_id__isnull=False
|
||||
)
|
||||
else:
|
||||
q_autocomplete = Building_Info.objects
|
||||
|
||||
# Получаем адреса и фильтруем на уровне Python для гарантированной регистронезависимости
|
||||
# (особенно важно для русского текста в SQLite и, возможно, других БД)
|
||||
all_addresses = q_autocomplete.values_list('sAddress', flat=True).distinct()
|
||||
|
||||
filtered_addresses = []
|
||||
for address in all_addresses:
|
||||
address_lower = address.lower()
|
||||
# Проверяем, содержатся ли все части поиска в адресе (без учета регистра)
|
||||
if all(part in address_lower for part in part_blocks):
|
||||
filtered_addresses.append(address)
|
||||
|
||||
# Сортируем и ограничиваем до 10 результатов
|
||||
addresses = sorted(filtered_addresses)[:10]
|
||||
|
||||
# Конвертируем в JSON (безопаснее, чем ручная конкатенация)
|
||||
result = json.dumps(list(addresses), ensure_ascii=False)
|
||||
|
||||
return HttpResponse(result, content_type='application/json; charset=utf-8')
|
||||
|
||||
@@ -6,10 +6,9 @@ from django.core.exceptions import ObjectDoesNotExist
|
||||
from oknardia.models import BlogPosts
|
||||
from oknardia.settings import *
|
||||
from django.utils import timezone
|
||||
from web.add_func import safe_html_spec_symbols
|
||||
from web.add_func import safe_html_spec_symbols, sanitize_slug
|
||||
from time import time
|
||||
import re
|
||||
import pytils
|
||||
from oknardia.settings import *
|
||||
|
||||
|
||||
@@ -38,7 +37,7 @@ def blog_list_posts(request: HttpRequest, page: str = "0") -> HttpResponse:
|
||||
except ValueError:
|
||||
page = 0
|
||||
dim_blogposts = [] # массив блог-постов для формирования списка
|
||||
to_template = {} # словарь, для передачи шаблону
|
||||
to_template: dict[str, object] = {} # словарь, для передачи шаблону
|
||||
template = "blog/blog_list.html" # шаблон
|
||||
in_list = NUM_BLOG_TIZER_IN_PAGE # длина списка блогов в выдачe
|
||||
# проверяем нужно ли ставить кнопку BACK и куда она ссылается
|
||||
@@ -86,15 +85,19 @@ def blog_list_posts(request: HttpRequest, page: str = "0") -> HttpResponse:
|
||||
'NAME1': post.kBlogAuthorUser.kDjangoUser.first_name,
|
||||
'NAME2': post.kBlogAuthorUser.kDjangoUser.last_name,
|
||||
'PUB_DAT': post.dPostDataBegin,
|
||||
'MOD_DAT': post.dPostDataModify,
|
||||
'HEADER': post.sPostHeader,
|
||||
'HEADER_D': safe_html_spec_symbols(post.sPostHeader),
|
||||
'HEADER_T': pytils.translit.slugify(safe_html_spec_symbols(post.sPostHeader)).lower(),
|
||||
'HEADER_T': sanitize_slug(post.sPostHeader),
|
||||
'POST_ID': post.id,
|
||||
'USER_STATUS': post.kBlogAuthorUser.get_sUserStatus_display(),
|
||||
'USER_AVATAR': post.kBlogAuthorUser.sUserAvatarImg,
|
||||
'USER_TITLE': post.kBlogAuthorUser.sUserJobTitle,
|
||||
'USER_FROM_ID_OFFICE': post.kBlogAuthorUser.kMerchantOffice,
|
||||
'CONTENT_CUT': post.sPostContent})
|
||||
'CONTENT_CUT': post.sPostContent,
|
||||
'META_DESC': post.sMetaDescription,
|
||||
'META_KW': post.sMetaKeywords,
|
||||
'IMG_BLOG': post.sImgForBlogSocial})
|
||||
# ищем CUT в тексте блога
|
||||
i_cut1 = post.sPostContent.lower().find(u"<cut")
|
||||
if i_cut1 != -1:
|
||||
@@ -108,14 +111,37 @@ def blog_list_posts(request: HttpRequest, page: str = "0") -> HttpResponse:
|
||||
dim_blogposts[i].update({'CUT_TEXT': u"Читать дальше →"})
|
||||
else:
|
||||
# Проверка на случай если нет "cut" и текст не длинный... нужна ли кнопка "читать дальше"?
|
||||
if len(post.sPostContent) < 4096:
|
||||
if len(post.sPostContent) < 2048:
|
||||
dim_blogposts[i].update({'CUT_TEXT': u"NONE"})
|
||||
else:
|
||||
dim_blogposts[i].update({'CUT_TEXT': u"Читать дальше →"})
|
||||
i += 1
|
||||
|
||||
# Формируем SEO-данные для мета-тегов страницы
|
||||
# Ключевые слова для B2B блога (компании-поставщик и их клиенты)
|
||||
combined_keywords = u"oknardia, окнардия, блог, поставщики окон, производители, установщики, компании"
|
||||
first_post_image = ""
|
||||
|
||||
if dim_blogposts:
|
||||
# Объединяем META_KW из нескольких первых постов
|
||||
collected_keywords = []
|
||||
for post_dict in dim_blogposts[:3]: # из первых 3 постов
|
||||
if post_dict.get('META_KW'):
|
||||
# Берем только часть keywords без фиксированного префикса (чтобы не повторять)
|
||||
kw_parts = post_dict['META_KW'].split(", ")
|
||||
if len(kw_parts) > 4: # пропускаем первые 4 (фиксированный префикс)
|
||||
collected_keywords.extend(kw_parts[4:])
|
||||
if collected_keywords:
|
||||
combined_keywords = u"oknardia, окнардия, блог, поставщики окон, производители, установщики, " + ", ".join(collected_keywords[:5])
|
||||
# Берем изображение первого поста для og:image
|
||||
if dim_blogposts[0].get('IMG_BLOG'):
|
||||
first_post_image = f"/media/{dim_blogposts[0]['IMG_BLOG']}"
|
||||
|
||||
to_template.update({'DIM_BLOGPOST': dim_blogposts,
|
||||
'META_DATA_PUB': q[0].dPostDataBegin,
|
||||
'META_DATA_MODIFY': q[0].dPostDataModify,
|
||||
'META_KEYWORDS': combined_keywords,
|
||||
'META_IMAGE': first_post_image,
|
||||
'PAGE_BACK': page,
|
||||
'ticks': float(time()-time_start)})
|
||||
return render(request, template, to_template)
|
||||
@@ -141,7 +167,7 @@ def blog_post(request: HttpRequest, post_id: str = "0", page_back: str = None) -
|
||||
back_page = int(request.GET["page-back"])
|
||||
except (TypeError, KeyError):
|
||||
back_page = 0
|
||||
to_template = {} # словарь, для передачи шаблону
|
||||
to_template: dict[str, object] = {} # словарь, для передачи шаблону
|
||||
template = "blog/blog_post.html" # шаблон
|
||||
|
||||
q = BlogPosts.objects.get(id=post_id)
|
||||
@@ -157,23 +183,27 @@ def blog_post(request: HttpRequest, post_id: str = "0", page_back: str = None) -
|
||||
'ID': q.id})
|
||||
if PATH_FOR_IMG_BLOG in q.sImgForBlogSocial.name:
|
||||
to_template.update({'IMG_FOR_BLOG': q.sImgForBlogSocial})
|
||||
to_template.update({'PUB_DAT': q.dPostDataBegin,
|
||||
'PUB_MODIFY': q.dPostDataModify,
|
||||
'HEADER': q.sPostHeader,
|
||||
'HEADER_T': pytils.translit.slugify(safe_html_spec_symbols(q.sPostHeader)).lower(),
|
||||
'USER_STATUS': q.kBlogAuthorUser.get_sUserStatus_display(),
|
||||
'USER_AVATAR': q.kBlogAuthorUser.sUserAvatarImg,
|
||||
'USER_TITLE': q.kBlogAuthorUser.sUserJobTitle,
|
||||
'USER_FROM_ID_OFFICE': q.kBlogAuthorUser.kMerchantOffice,
|
||||
'CONTENT': re.sub(r'<cut[\s\S]*?>', '', q.sPostContent, 0, re.IGNORECASE)})
|
||||
to_template.update({'TIZER': safe_html_spec_symbols(
|
||||
re.sub('<script[\s\S]*?</script>|<style[\s\S]*?</style>|<iframe[\s\S]*?</iframe>',
|
||||
'', to_template["CONTENT"], 0, re.IGNORECASE))})
|
||||
to_template.update({
|
||||
'PUB_DAT': q.dPostDataBegin,
|
||||
'PUB_MODIFY': q.dPostDataModify,
|
||||
'HEADER': safe_html_spec_symbols(q.sPostHeader),
|
||||
'HEADER_T': sanitize_slug(q.sPostHeader),
|
||||
'USER_STATUS': q.kBlogAuthorUser.get_sUserStatus_display(),
|
||||
'USER_AVATAR': q.kBlogAuthorUser.sUserAvatarImg,
|
||||
'USER_TITLE': q.kBlogAuthorUser.sUserJobTitle,
|
||||
'USER_FROM_ID_OFFICE': q.kBlogAuthorUser.kMerchantOffice,
|
||||
'CONTENT': re.sub(r'<cut[\s\S]*?>', '', q.sPostContent, 0, re.IGNORECASE),
|
||||
'MOD_DAT': q.dPostDataModify,
|
||||
'META_DESC': q.sMetaDescription,
|
||||
'META_KW': q.sMetaKeywords
|
||||
})
|
||||
content = to_template.get('CONTENT', '')
|
||||
to_template.update({'TIZER': safe_html_spec_symbols(str(content))})
|
||||
# получаем следующую по дате запись
|
||||
try:
|
||||
q1 = BlogPosts.objects.filter(dPostDataBegin__gt=q.dPostDataBegin, dPostDataBegin__lt=timezone.now(),
|
||||
bPublished=True, bArchive=False).order_by('dPostDataBegin')[0]
|
||||
to_template.update({'FORW_HEADER_T': pytils.translit.slugify(safe_html_spec_symbols(q1.sPostHeader)).lower(),
|
||||
to_template.update({'FORW_HEADER_T': sanitize_slug(q1.sPostHeader),
|
||||
'FORW_ID': q1.id})
|
||||
except(IndexError, ObjectDoesNotExist, BlogPosts.DoesNotExist):
|
||||
to_template.update({'FORW_DISABLE': True})
|
||||
@@ -181,7 +211,7 @@ def blog_post(request: HttpRequest, post_id: str = "0", page_back: str = None) -
|
||||
try:
|
||||
q1 = BlogPosts.objects.filter(dPostDataBegin__lt=q.dPostDataBegin, bPublished=True,
|
||||
bArchive=False).order_by('-dPostDataBegin')[0]
|
||||
to_template.update({'BACK_HEADER_T': pytils.translit.slugify(safe_html_spec_symbols(q1.sPostHeader)).lower(),
|
||||
to_template.update({'BACK_HEADER_T': sanitize_slug(q1.sPostHeader),
|
||||
'BACK_ID': q1.id})
|
||||
except(IndexError, ObjectDoesNotExist, BlogPosts.DoesNotExist):
|
||||
to_template.update({'BACK_DISABLE': True})
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
511
oknardia/web/catalog_companies.py
Normal file
511
oknardia/web/catalog_companies.py
Normal file
@@ -0,0 +1,511 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
Каталог производителей и компаний.
|
||||
|
||||
Модуль предоставляет views для отображения:
|
||||
1. Списка всех производителей с их ключевыми показателями (рейтинг, количество
|
||||
предложений, среднюю цену и т.п.)
|
||||
2. Детальную информацию о конкретном производителе со всеми его оконными наборами
|
||||
|
||||
Все запросы переведены на Django ORM для лучшей производительности и чистоты кода.
|
||||
"""
|
||||
from django.shortcuts import render, redirect
|
||||
from django.http import HttpRequest, HttpResponse, Http404
|
||||
from django.db.models import Count, Avg, Max, Min, DecimalField
|
||||
from oknardia.models import (
|
||||
MerchantBrand,
|
||||
SetKit,
|
||||
PriceOffer,
|
||||
)
|
||||
from web.report1 import get_last_all_user_visit_list
|
||||
from web.add_func import get_rating_set_for_stars, sanitize_slug
|
||||
import django.utils.dateformat
|
||||
import time
|
||||
import random
|
||||
import re
|
||||
import pytils
|
||||
|
||||
|
||||
def _get_company_statistics() -> list:
|
||||
"""
|
||||
Получает список компаний (MerchantBrand) с агрегированной статистикой.
|
||||
|
||||
Статистика включает:
|
||||
- Количество оконных наборов от компании
|
||||
- Средний рейтинг наборов
|
||||
- Количество ценовых предложений
|
||||
- Среднюю цену предложений
|
||||
- Дату последнего обновления цены
|
||||
|
||||
Оптимизировано для минимизации запросов к БД.
|
||||
|
||||
Returns:
|
||||
list: Список словарей с данными компаний
|
||||
"""
|
||||
# 1. Статистика по наборам (SetKit) для каждой компании
|
||||
set_stats = (
|
||||
SetKit.objects
|
||||
.filter(kSet2User__kMerchantOffice__kMerchantName__isnull=False)
|
||||
.values('kSet2User__kMerchantOffice__kMerchantName_id')
|
||||
.annotate(
|
||||
num_sets=Count('id', distinct=True),
|
||||
avg_rating=Avg('fSetRating')
|
||||
)
|
||||
)
|
||||
set_stats_dict = {
|
||||
stat['kSet2User__kMerchantOffice__kMerchantName_id']: {
|
||||
'num_sets': stat['num_sets'],
|
||||
'avg_rating': stat['avg_rating'] or 0
|
||||
}
|
||||
for stat in set_stats
|
||||
}
|
||||
|
||||
# 2. Статистика по ценовым предложениям (PriceOffer)
|
||||
companies_data = (
|
||||
PriceOffer.objects
|
||||
.filter(
|
||||
sOfferActive=True,
|
||||
kOfferFromUser__kMerchantOffice__kMerchantName__isnull=False
|
||||
)
|
||||
.values('kOfferFromUser__kMerchantOffice__kMerchantName_id')
|
||||
.annotate(
|
||||
num_offers=Count('id', distinct=True),
|
||||
price_avg=Avg('fOfferPrice', output_field=DecimalField()),
|
||||
last_update=Max('dOfferModify')
|
||||
)
|
||||
.order_by('-last_update')
|
||||
)
|
||||
|
||||
# 3. Получаем все объекты MerchantBrand одним запросом (решение проблемы N+1)
|
||||
company_ids = [
|
||||
offer['kOfferFromUser__kMerchantOffice__kMerchantName_id']
|
||||
for offer in companies_data
|
||||
]
|
||||
merchants = MerchantBrand.objects.in_bulk(company_ids)
|
||||
|
||||
# 4. Собираем финальный результат
|
||||
result = []
|
||||
for offer in companies_data:
|
||||
company_id = offer['kOfferFromUser__kMerchantOffice__kMerchantName_id']
|
||||
merchant = merchants.get(company_id)
|
||||
|
||||
if not merchant:
|
||||
continue
|
||||
|
||||
set_stat = set_stats_dict.get(company_id, {
|
||||
'num_sets': 0,
|
||||
'avg_rating': 0
|
||||
})
|
||||
|
||||
result.append({
|
||||
'id': merchant.id,
|
||||
'sMerchantName': merchant.sMerchantName,
|
||||
'pMerchantLogo': merchant.pMerchantLogo,
|
||||
'NumSets': set_stat['num_sets'],
|
||||
'RatingAVG': set_stat['avg_rating'],
|
||||
'NumOffers': offer['num_offers'],
|
||||
'PriceAVG': offer['price_avg'],
|
||||
'lastUpdate': offer['last_update']
|
||||
})
|
||||
|
||||
# Сортируем по среднему рейтингу (убывание)
|
||||
result.sort(key=lambda x: x['RatingAVG'], reverse=True)
|
||||
|
||||
return result
|
||||
|
||||
|
||||
def _format_company_for_template(company_data: dict) -> dict:
|
||||
"""
|
||||
Форматирует данные компании для вывода в шаблон.
|
||||
|
||||
Применяет:
|
||||
- Конвертацию времени в читаемый формат (e.g., "3 дня назад")
|
||||
- Склонение существительных (plural forms)
|
||||
- Вычисление звёзд рейтинга
|
||||
- Скатывание имени в slug для URL
|
||||
|
||||
Args:
|
||||
company_data (dict): Словарь с данными компании
|
||||
|
||||
Returns:
|
||||
dict: Отформатированные данные компании
|
||||
"""
|
||||
formatted = company_data.copy()
|
||||
# Вычисляем звёзды на основе рейтинга
|
||||
formatted['STARS'] = get_rating_set_for_stars(
|
||||
formatted['RatingAVG']
|
||||
)
|
||||
# Применяем правильные формы множественного числа
|
||||
formatted['NumSets'] = pytils.numeral.get_plural(
|
||||
formatted['NumSets'],
|
||||
"оконный набор, оконных набора, оконных наборов"
|
||||
)
|
||||
formatted['NumOffers'] = pytils.numeral.get_plural(
|
||||
formatted['NumOffers'],
|
||||
"вариант, варианта, вариантов"
|
||||
)
|
||||
# Конвертируем время последнего обновления в читаемый формат
|
||||
if formatted['lastUpdate']:
|
||||
timestamp = int(
|
||||
django.utils.dateformat.format(
|
||||
formatted['lastUpdate'],
|
||||
'U'
|
||||
)
|
||||
)
|
||||
formatted['lastUpdate'] = pytils.dt.distance_of_time_in_words(
|
||||
timestamp
|
||||
)
|
||||
# Генерируем slug из имени компании для URL
|
||||
formatted['sMerchantMainURL'] = sanitize_slug(formatted['sMerchantName'])
|
||||
return formatted
|
||||
|
||||
|
||||
def catalog_company(request: HttpRequest) -> HttpResponse:
|
||||
"""
|
||||
Показывает список всех производителей с ключевыми показателями.
|
||||
|
||||
GET параметры: опционально могут использоваться для фильтрации
|
||||
|
||||
Контекст шаблона:
|
||||
- COMPANIES (list): Список компаний с статистикой
|
||||
- LOG_VISIT (list): Последние визиты всех пользователей
|
||||
|
||||
Args:
|
||||
request (HttpRequest): HTTP запрос от клиента
|
||||
|
||||
Returns:
|
||||
HttpResponse: Отрендеренная HTML страница со списком компаний
|
||||
"""
|
||||
# Получаем статистику по компаниям с использованием ORM
|
||||
companies_list = _get_company_statistics()
|
||||
|
||||
# Форматируем каждую компанию для вывода в шаблон
|
||||
formatted_companies = [
|
||||
_format_company_for_template(company)
|
||||
for company in companies_list
|
||||
]
|
||||
|
||||
# Получаем информацию о посещениях для персонализации
|
||||
to_template: dict[str, object] = {
|
||||
'COMPANIES': formatted_companies,
|
||||
'LOG_VISIT': get_last_all_user_visit_list(),
|
||||
}
|
||||
|
||||
return render(request, "catalog/catalog_company.html", to_template)
|
||||
|
||||
|
||||
def _lowercase_first_char(text: str) -> str:
|
||||
"""
|
||||
Преобразует первый символ строки в нижний регистр.
|
||||
|
||||
Args:
|
||||
text (str): Исходная строка
|
||||
|
||||
Returns:
|
||||
str: Строка с строчным первым символом (если длина > 0)
|
||||
"""
|
||||
return text[0].lower() + text[1:] if len(text) > 0 else text
|
||||
|
||||
|
||||
def _clean_text_field(text: str, empty_values: list) -> str:
|
||||
"""
|
||||
Очищает текстовое поле, удаляя типичные маркеры "пусто" и преобразуя
|
||||
первый символ в нижний регистр.
|
||||
|
||||
Args:
|
||||
text (str): Исходный текст
|
||||
empty_values (list): Список значений, которые считаются "пустыми"
|
||||
|
||||
Returns:
|
||||
str: Очищенный текст или пустая строка если значение в empty_values
|
||||
"""
|
||||
if text.lower() in empty_values:
|
||||
return ""
|
||||
return _lowercase_first_char(text)
|
||||
|
||||
|
||||
def _get_company_sets_detail(company_id: int) -> list:
|
||||
"""
|
||||
Получает все оконные наборы для компании с полной статистикой по ценам.
|
||||
|
||||
Использует оптимизированные select_related и prefetch_related для минимизации
|
||||
запросов к БД. Группирует данные по наборам (SetKit) с уникальностью.
|
||||
|
||||
Args:
|
||||
company_id (int): ID компании (MerchantBrand)
|
||||
|
||||
Returns:
|
||||
list: Список словарей с данными наборов, отсортированные по рейтингу
|
||||
"""
|
||||
# Получаем активные ценовые предложения для компаний с агрегацией по наборам
|
||||
price_stats = (
|
||||
PriceOffer.objects
|
||||
.filter(
|
||||
sOfferActive=True,
|
||||
kOfferFromUser__kMerchantOffice__kMerchantName_id=company_id
|
||||
)
|
||||
.values('kOffer2SetKit_id')
|
||||
.annotate(
|
||||
num_offers=Count('id'),
|
||||
price_avg=Avg('fOfferPrice', output_field=DecimalField()),
|
||||
last_update=Max('dOfferModify'),
|
||||
early_creation=Min('dOfferCreate')
|
||||
)
|
||||
)
|
||||
|
||||
# Преобразуем в словарь для быстрого доступа по ID набора
|
||||
price_stats_dict = {
|
||||
stat['kOffer2SetKit_id']: {
|
||||
'num_offers': stat['num_offers'],
|
||||
'price_avg': stat['price_avg'],
|
||||
'last_update': stat['last_update'],
|
||||
'early_creation': stat['early_creation']
|
||||
}
|
||||
for stat in price_stats
|
||||
}
|
||||
|
||||
# Получаем все наборы компании с их зависимостями
|
||||
# select_related оптимизирует ForeignKey запросы (профиль, стеклопакет)
|
||||
sets_queryset = (
|
||||
SetKit.objects
|
||||
.filter(
|
||||
kSet2User__kMerchantOffice__kMerchantName_id=company_id
|
||||
)
|
||||
.select_related(
|
||||
'kSet2User',
|
||||
'kSet2User__kMerchantOffice',
|
||||
'kSet2User__kMerchantOffice__kMerchantName',
|
||||
'kSet2PVCprofiles',
|
||||
'kSet2Glazing'
|
||||
)
|
||||
.order_by('-fSetRating')
|
||||
)
|
||||
|
||||
# Собираем результат, комбинируя данные SetKit с агрегированной статистикой
|
||||
result = []
|
||||
seen_set_ids = set()
|
||||
|
||||
for setkit in sets_queryset:
|
||||
# Пропускаем дубликаты наборов (может быть несколько ценовых предложений
|
||||
# для одного набора)
|
||||
if setkit.id in seen_set_ids:
|
||||
continue
|
||||
seen_set_ids.add(setkit.id)
|
||||
|
||||
# Получаем статистику по ценам для этого набора
|
||||
price_stat = price_stats_dict.get(setkit.id, {
|
||||
'num_offers': 0,
|
||||
'price_avg': None,
|
||||
'last_update': None,
|
||||
'early_creation': None
|
||||
})
|
||||
|
||||
# Собираем все данные в один объект
|
||||
result.append({
|
||||
'setkit': setkit,
|
||||
'num_offers': price_stat['num_offers'],
|
||||
'price_avg': price_stat['price_avg'],
|
||||
'last_update': price_stat['last_update'],
|
||||
'early_creation': price_stat['early_creation'],
|
||||
'merchant_office': setkit.kSet2User.kMerchantOffice,
|
||||
'merchant_brand': setkit.kSet2User.kMerchantOffice.kMerchantName,
|
||||
'profile': setkit.kSet2PVCprofiles,
|
||||
'glazing': setkit.kSet2Glazing
|
||||
})
|
||||
|
||||
return result
|
||||
|
||||
|
||||
def _format_set_for_template(set_data: dict, empty_values: list) -> dict:
|
||||
"""
|
||||
Форматирует данные оконного набора для вывода в шаблон.
|
||||
|
||||
Применяет:
|
||||
- Преобразование URL в удобный для отображения формат
|
||||
- Разделение email адресов на части (для обфускации)
|
||||
- Вычисление звёзд рейтинга
|
||||
- Конвертация времени в читаемый формат
|
||||
- Создание slugs для названий и производителей
|
||||
- Склонение числительных(контуры, швы и т.п.)
|
||||
- Очистку пустых полей от стандартных маркеров ("нет", "—" и т.п.)
|
||||
|
||||
Args:
|
||||
set_data (dict): Данные набора с объектами моделей
|
||||
empty_values (list): Список значений, считаемых "пустыми"
|
||||
|
||||
Returns:
|
||||
dict: Отформатированные данные для шаблона
|
||||
"""
|
||||
set_kit = set_data['setkit']
|
||||
merchant_office = set_data['merchant_office']
|
||||
merchant_brand = set_data['merchant_brand']
|
||||
profile = set_data['profile']
|
||||
glazing = set_data['glazing']
|
||||
|
||||
formatted = {
|
||||
# Ключи ниже оставлены в legacy-формате, т.к. шаблон использует именно их имена.
|
||||
'idSetKit': set_kit.id,
|
||||
'sSetName': set_kit.sSetName,
|
||||
'sMerchantName': merchant_brand.sMerchantName,
|
||||
'sMerchantDescription': merchant_brand.sMerchantDescription,
|
||||
'fSetRating': {
|
||||
'RATING': set_kit.fSetRating,
|
||||
'STARS': get_rating_set_for_stars(set_kit.fSetRating)
|
||||
},
|
||||
'num_offers': set_data['num_offers'],
|
||||
'price_avg': set_data['price_avg'],
|
||||
'bSetDelivery': set_kit.bSetDelivery,
|
||||
'bSetUninstallInstall': set_kit.bSetUninstallInstall,
|
||||
'sSetImplementAll': set_kit.sSetImplementAll,
|
||||
'sSetImplementHandles': set_kit.sSetImplementHandles,
|
||||
'sMerchantMainURL': {
|
||||
'URL': merchant_office.kMerchantName.sMerchantMainURL,
|
||||
'URL_VIEW': re.sub(
|
||||
r"^https?://|/$|www\.",
|
||||
"",
|
||||
merchant_office.kMerchantName.sMerchantMainURL
|
||||
)
|
||||
},
|
||||
'sOfficePhones': merchant_office.sOfficePhones,
|
||||
'sOfficeDescription': merchant_office.sOfficeDescription,
|
||||
'sOfficeEmails': merchant_office.sOfficeEmails,
|
||||
'sOfficeName': merchant_office.sOfficeName,
|
||||
'sOfficeAddress': merchant_office.sOfficeAddress,
|
||||
'fOfficeGeoCode_Latitude': merchant_office.fOfficeGeoCode_Latitude,
|
||||
'fOfficeGeoCode_Longitude': merchant_office.fOfficeGeoCode_Longitude,
|
||||
'sOfficeDiscountMetaFormula': merchant_office.sOfficeDiscountMetaFormula,
|
||||
'pMerchantLogo': merchant_office.kMerchantName.pMerchantLogo,
|
||||
'idPVC': profile.id,
|
||||
'sProfileBriefDescription': profile.sProfileBriefDescription,
|
||||
'iProfileCameras': profile.iProfileCameras,
|
||||
'sProfileName': {
|
||||
'NAME': profile.sProfileName,
|
||||
'NAME_T': sanitize_slug(profile.sProfileName)
|
||||
},
|
||||
'sProfileManufacturer': {
|
||||
'NAME': profile.sProfileManufacturer,
|
||||
'NAME_T': sanitize_slug(profile.sProfileManufacturer)
|
||||
},
|
||||
'sProfileColor': profile.sProfileColor,
|
||||
'sProfileSealDescription': profile.sProfileSealDescription,
|
||||
'fProfileSeals': pytils.numeral.sum_string(
|
||||
profile.fProfileSeals,
|
||||
pytils.numeral.MALE,
|
||||
"контур, контура, контуров"
|
||||
),
|
||||
'sGlazingBriefDescription': glazing.sGlazingBriefDescription,
|
||||
'sGlazingManufacturer': glazing.sGlazingManufacturer,
|
||||
'sGlazingMark': glazing.sGlazingMark,
|
||||
'sGlazingToning': glazing.sGlazingToning,
|
||||
'sSetImplementCatch': _clean_text_field(set_kit.sSetImplementCatch, empty_values),
|
||||
'sSetClimateControl': _clean_text_field(set_kit.sSetClimateControl, empty_values),
|
||||
'sProfileReinforcement': _lowercase_first_char(profile.sProfileReinforcement),
|
||||
'sSetSill': _lowercase_first_char(set_kit.sSetSill),
|
||||
'sSetPanes': _lowercase_first_char(set_kit.sSetPanes),
|
||||
'sSetSlope': _lowercase_first_char(set_kit.sSetSlope),
|
||||
'sSetUninstallInstall': _lowercase_first_char(set_kit.sSetUninstallInstall),
|
||||
'sSetDelivery': _lowercase_first_char(set_kit.sSetDelivery),
|
||||
'sSetOtherConditions': _lowercase_first_char(set_kit.sSetOtherConditions),
|
||||
}
|
||||
|
||||
# Конвертируем даты в читаемый формат
|
||||
if set_data['last_update']:
|
||||
timestamp = int(django.utils.dateformat.format(set_data['last_update'], 'U'))
|
||||
formatted['lastUpdate'] = pytils.dt.distance_of_time_in_words(timestamp)
|
||||
|
||||
if set_data['early_creation']:
|
||||
timestamp = int(django.utils.dateformat.format(set_data['early_creation'],'U'))
|
||||
formatted['earlyCreation'] = pytils.dt.distance_of_time_in_words(timestamp)
|
||||
|
||||
# Разделяем email на части для обфускации (показываем середину отдельно)
|
||||
# На фронтенде JS собирает все обратно в валидный e-mail
|
||||
if formatted['sOfficeEmails']:
|
||||
try:
|
||||
email_len = len(formatted['sOfficeEmails'])
|
||||
k = random.randint(1, max(1, int(email_len / 2) - 1))
|
||||
formatted['sOfficeEmails'] = [
|
||||
formatted['sOfficeEmails'][0:k],
|
||||
formatted['sOfficeEmails'][k:-k],
|
||||
formatted['sOfficeEmails'][-k:]
|
||||
]
|
||||
except (ValueError, ZeroDivisionError):
|
||||
# Если ошибка при случайном разделении, оставляем как есть
|
||||
pass
|
||||
|
||||
return formatted
|
||||
|
||||
|
||||
def catalog_company_detail(
|
||||
request: HttpRequest,
|
||||
company_id: str,
|
||||
company_name_slug: str
|
||||
) -> HttpResponse:
|
||||
"""
|
||||
Показывает детальную информацию о компании и все её оконные наборы.
|
||||
|
||||
Производит редирект если slug в URL не совпадает с актуальным.
|
||||
|
||||
GET параметры: опционально могут использоваться для фильтрации
|
||||
|
||||
Контекст шаблона:
|
||||
- COMPANY (str): Название компании
|
||||
- COMPANY_ID (int): ID компании
|
||||
- COMPANY_T (str): Slug компании
|
||||
- SETS (list): Список оконных наборов с их полной информацией
|
||||
- IMG_FOR_BLOG (str): Логотип компании
|
||||
- LIST_NOT (list): Стандартные маркеры "пусто"
|
||||
- LOG_VISIT (list): Последние визиты всех пользователей
|
||||
- ticks (float): Время выполнения представления (в секундах)
|
||||
|
||||
Args:
|
||||
request (HttpRequest): HTTP запрос от клиента
|
||||
company_id (str): ID компании в виде строки
|
||||
company_name_slug (str): Slug названия компании из URL
|
||||
|
||||
Returns:
|
||||
HttpResponse: Отрендеренная HTML страница с деталью компании или редирект
|
||||
"""
|
||||
time_start = time.perf_counter()
|
||||
company_id_int = int(company_id)
|
||||
|
||||
# Получаем компанию или возвращаем 404
|
||||
try:
|
||||
company = MerchantBrand.objects.get(id=company_id_int)
|
||||
except MerchantBrand.DoesNotExist:
|
||||
raise Http404("Компания не найдена")
|
||||
|
||||
# Проверяем что slug совпадает (для SEO и красивых URL)
|
||||
actual_slug = sanitize_slug(company.sMerchantName)
|
||||
if actual_slug != company_name_slug:
|
||||
return redirect(
|
||||
f'/catalog/company/{company_id_int}-{actual_slug}'
|
||||
)
|
||||
|
||||
# Типичные маркеры, которые означают что поле пусто
|
||||
empty_values = ["нет", "—", ""]
|
||||
|
||||
# Получаем все наборы компании с ценовой статистикой
|
||||
sets_list = _get_company_sets_detail(company_id_int)
|
||||
|
||||
# Форматируем каждый набор для вывода в шаблон
|
||||
formatted_sets = [
|
||||
_format_set_for_template(set_data, empty_values)
|
||||
for set_data in sets_list
|
||||
]
|
||||
|
||||
to_template: dict[str, object] = {
|
||||
'COMPANY': company.sMerchantName,
|
||||
'COMPANY_ID': company_id_int,
|
||||
'COMPANY_T': company_name_slug,
|
||||
'SETS': formatted_sets,
|
||||
'HEADER': f'Изготовитель окон «{company.sMerchantName}»',
|
||||
'META_KEYWORDS': company.sMerchantName,
|
||||
'IMG_FOR_BLOG': company.pMerchantLogo,
|
||||
'LIST_NOT': empty_values,
|
||||
'LOG_VISIT': get_last_all_user_visit_list(),
|
||||
}
|
||||
|
||||
# Добавляем метрику выполнения представления
|
||||
to_template['ticks'] = float(time.perf_counter() - time_start)
|
||||
|
||||
return render(request, "catalog/catalog_company_detail.html", to_template)
|
||||
91
oknardia/web/catalog_openings.py
Normal file
91
oknardia/web/catalog_openings.py
Normal file
@@ -0,0 +1,91 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
from django.db.models import F
|
||||
from django.shortcuts import render
|
||||
from django.http import HttpRequest, HttpResponse
|
||||
from oknardia.models import MountDim2Apartment
|
||||
from web.report1 import get_last_all_user_visit_list
|
||||
from web.add_func import get_flaps_for_mini_pictures, sanitize_slug
|
||||
import time
|
||||
from typing import Any
|
||||
from itertools import groupby
|
||||
from operator import itemgetter
|
||||
|
||||
|
||||
def _append_visit_context(to_template: dict, request: HttpRequest, time_start: float) -> None:
|
||||
"""Дописывает в контекст стандартный хвост: визиты и время выполнения."""
|
||||
to_template.update({
|
||||
# получаем последние визиты всех посетителей из базы
|
||||
'LOG_VISIT': get_last_all_user_visit_list(),
|
||||
'ticks': float(time.perf_counter() - time_start),
|
||||
})
|
||||
|
||||
|
||||
def standard_opening(request: HttpRequest) -> HttpResponse:
|
||||
"""
|
||||
Каталог стандартных оконных проёмов и балконных блоков.
|
||||
|
||||
Что делает вьюха:
|
||||
- Собирает уникальные пары «проём ↔ серия» через ORM.
|
||||
- Агрегирует данные для шаблона в структуру LIST_WIN_OPENING с помощью groupby.
|
||||
- Добавляет в контекст последние визиты и время выполнения.
|
||||
"""
|
||||
time_start = time.perf_counter()
|
||||
|
||||
q_win_opening = (
|
||||
MountDim2Apartment.objects.filter(kApartment__kSeria_id=F('kApartment__kSeria__kRoot_id'))
|
||||
.values(
|
||||
'kMountDim_id',
|
||||
'kMountDim__sFlapConfig',
|
||||
'kMountDim__sDescripion',
|
||||
'kMountDim__bIsDoor',
|
||||
'kMountDim__bIsNearDoor',
|
||||
'kMountDim__iWinHight',
|
||||
'kMountDim__iWinWidth',
|
||||
'kApartment__kSeria_id',
|
||||
'kApartment__kSeria__sName',
|
||||
)
|
||||
.distinct()
|
||||
.order_by(
|
||||
'-kMountDim__iWinWidth',
|
||||
'-kMountDim__iWinHight',
|
||||
'kMountDim__bIsNearDoor',
|
||||
'kMountDim__bIsDoor',
|
||||
'kMountDim_id',
|
||||
'kApartment__kSeria__sName',
|
||||
)
|
||||
)
|
||||
|
||||
list_windows_opening: list[dict[str, Any]] = []
|
||||
# Группируем результаты по ID проёма, чтобы собрать все серии, в которые он входит.
|
||||
# `order_by` в запросе гарантирует, что все записи для одного проёма идут подряд.
|
||||
for mount_dim_id, group in groupby(q_win_opening, key=itemgetter('kMountDim_id')):
|
||||
rows_for_opening = list(group)
|
||||
first_row = rows_for_opening[0]
|
||||
description_full = first_row['kMountDim__sDescripion'] or ''
|
||||
|
||||
# Собираем список серий для текущего проёма.
|
||||
serias_for_opening = [
|
||||
{
|
||||
'ID': row['kApartment__kSeria_id'],
|
||||
'NAME_T': sanitize_slug(row['kApartment__kSeria__sName']),
|
||||
'NAME': row['kApartment__kSeria__sName'],
|
||||
}
|
||||
for row in rows_for_opening
|
||||
]
|
||||
# Формируем данные для строки таблиц (типовой проем)
|
||||
list_windows_opening.append({
|
||||
'ID': mount_dim_id,
|
||||
'INCLUDING_IN_SERIA': serias_for_opening,
|
||||
'URL2IMG': get_flaps_for_mini_pictures(first_row['kMountDim__sFlapConfig']),
|
||||
'FLAP_CONFIG': first_row['kMountDim__sFlapConfig'],
|
||||
'DESCRIPTION': description_full.split(' для')[0].split(' (')[0],
|
||||
'DESCRIPTION_L': description_full,
|
||||
'IS_DOOR': first_row['kMountDim__bIsDoor'],
|
||||
'IS_NEAR_DOOR': first_row['kMountDim__bIsNearDoor'],
|
||||
'H': first_row['kMountDim__iWinHight'] * 10, # см -> мм
|
||||
'W': first_row['kMountDim__iWinWidth'] * 10, # см -> мм
|
||||
})
|
||||
|
||||
to_template = {'LIST_WIN_OPENING': list_windows_opening}
|
||||
_append_visit_context(to_template, request, time_start)
|
||||
return render(request, 'catalog/catalog_standard_opening.html', to_template)
|
||||
353
oknardia/web/catalog_profiles.py
Normal file
353
oknardia/web/catalog_profiles.py
Normal file
@@ -0,0 +1,353 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
from datetime import datetime
|
||||
from django.core.exceptions import ObjectDoesNotExist
|
||||
from django.db.models import Count
|
||||
from django.shortcuts import render, redirect
|
||||
from django.http import HttpRequest, HttpResponse
|
||||
from oknardia.settings import *
|
||||
from oknardia.models import Catalog2Profile, PVCprofiles, PriceOffer
|
||||
from web.report1 import get_last_all_user_visit_list
|
||||
from web.add_func import normalize, get_rating_set_for_stars, sanitize_slug
|
||||
import time
|
||||
import json
|
||||
import re
|
||||
import pytils
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Модульные хелперы, общие для всех вьюх этого файла
|
||||
# ---------------------------------------------------------------------------
|
||||
def _merchant_row_to_dict(row: dict) -> dict:
|
||||
"""Преобразует ORM-строку с данными партнёра в словарь для шаблона."""
|
||||
merchant_name = row["kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__sMerchantName"]
|
||||
return {
|
||||
"MERCHANT_ID": row["kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__id"],
|
||||
"MERCHANT_NAME": merchant_name,
|
||||
"MERCHANT_NAME_T": sanitize_slug(merchant_name),
|
||||
"MERCHANT_LOGO_URL": row["kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__pMerchantLogo"],
|
||||
"MERCHANT_OFFERS": row["offers_by_merchant"],
|
||||
}
|
||||
|
||||
|
||||
def _profile_row_to_dict(profile: dict) -> dict:
|
||||
"""Преобразует ORM-строку профиля в словарь для шаблона."""
|
||||
return {
|
||||
"PROFILE_NAME": profile["sProfileBriefDescription"],
|
||||
"PROFILE_ID": profile["id"],
|
||||
"PROFILE_URL": sanitize_slug(profile["sProfileName"]),
|
||||
"PROFILE_RATING": profile["fProfileRating"],
|
||||
"PROFILE_RATING_STARS": get_rating_set_for_stars(profile["fProfileRating"]),
|
||||
}
|
||||
|
||||
|
||||
def _append_visit_context(to_template: dict, request: HttpRequest, time_start: float) -> None:
|
||||
"""Дописывает в контекст стандартный хвост: визиты и время выполнения."""
|
||||
to_template.update({
|
||||
'LOG_VISIT': get_last_all_user_visit_list(),
|
||||
'ticks': float(time.perf_counter() - time_start),
|
||||
})
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
|
||||
def catalog_profile(request: HttpRequest) -> HttpResponse:
|
||||
"""
|
||||
КАТАЛОГ ПРОФИЛЕЙ: страница со списком производителей и моделей (марками) профилей
|
||||
|
||||
:param request: HttpRequest -- входящий http-запрос
|
||||
:return response: HttpResponse -- исходящий http-ответ
|
||||
"""
|
||||
time_start = time.perf_counter()
|
||||
# Берём только те поля, которые реально нужны для построения страницы каталога.
|
||||
# Это позволяет не тащить лишние данные из БД и сразу работать с простыми словарями.
|
||||
profile_rows = list(
|
||||
PVCprofiles.objects.values(
|
||||
"id",
|
||||
"sProfileName",
|
||||
"sProfileBriefDescription",
|
||||
"sProfileManufacturer",
|
||||
).order_by("sProfileManufacturer", "sProfileBriefDescription")
|
||||
)
|
||||
profile_count = len(profile_rows)
|
||||
# В этом контексте смешанные типы значений (str/int/list/float), поэтому задаём общий тип явно.
|
||||
to_template: dict[str, object] = {
|
||||
'CATALOG_PROFILE_NUM': pytils.numeral.get_plural(profile_count, "профиль,профиля,профилей")
|
||||
}
|
||||
|
||||
list_profile_manufactures = []
|
||||
tmp_profile_manufacture = ""
|
||||
for profile in profile_rows:
|
||||
if profile["sProfileManufacturer"] == "":
|
||||
# Пустой производитель в каталоге только мешает: не создаём для него отдельную группу.
|
||||
continue
|
||||
|
||||
if tmp_profile_manufacture != profile["sProfileManufacturer"]:
|
||||
# Новый производитель — открываем новую группу карточек.
|
||||
tmp_profile_manufacture = profile["sProfileManufacturer"]
|
||||
list_profile_manufactures.append({
|
||||
"PROF_MAN_ID": profile["id"],
|
||||
"PROF_MAN": profile["sProfileManufacturer"],
|
||||
"PROF_MAN_T": sanitize_slug(profile["sProfileManufacturer"]),
|
||||
"PROF_MAN_LIST": [{
|
||||
"PROF_NAME_ID": profile["id"],
|
||||
"PROF_NAME": profile["sProfileBriefDescription"],
|
||||
"PROF_NAME_T": sanitize_slug(profile["sProfileName"]),
|
||||
}]
|
||||
})
|
||||
else:
|
||||
# Если производитель уже встречался, просто дописываем новую модель в его список.
|
||||
list_profile_manufactures[-1]["PROF_MAN_LIST"].append({
|
||||
"PROF_NAME_ID": profile["id"],
|
||||
"PROF_NAME": profile["sProfileBriefDescription"],
|
||||
"PROF_NAME_T": sanitize_slug(profile["sProfileName"]),
|
||||
})
|
||||
|
||||
to_template.update({
|
||||
'CATALOG_PROFILE_MAN1_NAME2': list_profile_manufactures,
|
||||
'CATALOG_MANUFACT_NUM': len(list_profile_manufactures),
|
||||
'CATALOG_MANUFACT_NUM_W':
|
||||
pytils.numeral.sum_string(len(list_profile_manufactures), pytils.numeral.MALE, ("производитель",
|
||||
"производителя",
|
||||
"производителей")),
|
||||
})
|
||||
_append_visit_context(to_template, request, time_start)
|
||||
return render(request, "catalog/catalog_of_profiles.html", to_template)
|
||||
|
||||
|
||||
def catalog_profile_model(request: HttpRequest, manufacture_id: int, manufacture_name: str,
|
||||
model_id: int, model_name: str) -> HttpResponse:
|
||||
"""
|
||||
КАТАЛОГ ПРОФИЛЕЙ: страница с описанием марки профиля
|
||||
|
||||
:param request: HttpRequest -- входящий http-запрос
|
||||
:param manufacture_id: id профиля. Предполагается, что это первый id при сортировке по sProfileBriefDescription
|
||||
:param manufacture_name: название производителя (транслитерированное sanitize_slug())
|
||||
:param model_id: id модели (марки) профиля
|
||||
:param model_name: модель (марка) профиля (транслитерированное sanitize_slug(sProfileName))
|
||||
:return response: HttpResponse -- исходящий http-ответ
|
||||
"""
|
||||
time_start = time.perf_counter()
|
||||
manufacture_id = int(manufacture_id)
|
||||
model_id = int(model_id)
|
||||
q_pvc_by_id = PVCprofiles.objects.get(id=model_id)
|
||||
manufacturer_slug = sanitize_slug(q_pvc_by_id.sProfileManufacturer)
|
||||
model_slug = sanitize_slug(q_pvc_by_id.sProfileName)
|
||||
if manufacturer_slug != manufacture_name \
|
||||
or model_slug != model_name \
|
||||
or manufacture_id != model_id:
|
||||
return redirect(f"/catalog/profile/{model_id}-{manufacturer_slug}/"
|
||||
f"{model_id}-{model_slug}")
|
||||
|
||||
# Локальные помощники держат вьюху короче и не размазывают однотипную логику по коду.
|
||||
def build_other_list(value: str) -> list[str]:
|
||||
# Убираем пустые куски, чтобы не плодить «пустые» характеристики в шаблоне.
|
||||
result = []
|
||||
for chunk in (part.strip() for part in value.split(";")):
|
||||
if not chunk:
|
||||
continue
|
||||
if ":" in chunk:
|
||||
head, tail = chunk.split(":", 1)
|
||||
result.append(f"<b>{head.strip()}:</b>{tail.strip()}")
|
||||
else:
|
||||
result.append(f"<b>{chunk}</b>")
|
||||
return result
|
||||
|
||||
def update_pub_dat(current_pub_dat: datetime | None, candidate_pub_dat: datetime | None) -> datetime | None:
|
||||
# На странице оставляем дату публикации/обновления только если она реально новее карточки профиля.
|
||||
if candidate_pub_dat is None:
|
||||
return current_pub_dat
|
||||
if current_pub_dat is None or candidate_pub_dat.replace(tzinfo=None) > current_pub_dat.replace(tzinfo=None):
|
||||
return candidate_pub_dat
|
||||
return current_pub_dat
|
||||
|
||||
def apply_rating_colors(rating: dict, rating_pairs: tuple[tuple[str, str], ...], multiplier: int,
|
||||
gray: bool = False) -> None:
|
||||
# Один маленький helper вместо россыпи почти одинаковых строк: меняется только множитель и формат RGB.
|
||||
for rating_key, template_key in rating_pairs:
|
||||
color = int(255 - rating[rating_key] * multiplier)
|
||||
if gray:
|
||||
to_template[template_key] = f"{color},{color},{color}"
|
||||
else:
|
||||
to_template[template_key] = f"{color},255,{color}"
|
||||
|
||||
to_template: dict[str, object] = {"CATALOG_MODEL": q_pvc_by_id,
|
||||
"CATALOG_MAN2URL": manufacture_name,
|
||||
"CATALOG_URL": f"{manufacture_id}-{manufacture_name}",
|
||||
"CATALOG_URL2": f"{manufacture_id}-{manufacture_name}/{model_id}-{model_name}",
|
||||
"PROFILE_RATING_STARS": get_rating_set_for_stars(q_pvc_by_id.fProfileRating)}
|
||||
# Размер выборки для алгоритмического рейтинга: количество моделей профилей в каталоге.
|
||||
# Используется в JSON-LD (ratingCount) и поясняющем тексте на странице.
|
||||
to_template["PROFILE_RATING_SAMPLE_SIZE"] = PVCprofiles.objects.count()
|
||||
try:
|
||||
got_json = json.loads(q_pvc_by_id.sProfileDescription)
|
||||
# раскрашиваем кружочки рейтинга напротив характеристик профиля
|
||||
rating_pairs = (
|
||||
(RANK_PVCP_CAMERAS_NUM_NAME, "RANK_PVCP_CAMERAS_COLOR"),
|
||||
(RANK_PVCP_SEALS_NAME, "RANK_PVCP_SEALS_COLOR"),
|
||||
(RANK_PVCP_THICKNESS_NAME, "RANK_PVCP_THICKNESS_COLOR"),
|
||||
(RANK_PVCP_G_THICKNESS_NAME, "RANK_PVCP_G_THICKNESS_COLOR"),
|
||||
(RANK_PVCP_RABBET_NAME, "RANK_PVCP_RABBET_COLOR"),
|
||||
(RANK_PVCP_HEAT_TRANSFER_NAME, "RANK_PVCP_HEAT_TRANSFER_COLOR"),
|
||||
(RANK_PVCP_SOUNDPROOFING_NAME, "RANK_PVCP_SOUNDPROOFING_COLOR"),
|
||||
(RANK_PVCP_HEIGHT_NAME, "RANK_PVCP_HEIGHT_COLOR"),
|
||||
)
|
||||
if KEY_RATING in got_json:
|
||||
# кружочки зелёные
|
||||
apply_rating_colors(got_json[KEY_RATING], rating_pairs, 255)
|
||||
elif KEY_RATING_VIRTUAL in got_json:
|
||||
# кружочки серые
|
||||
apply_rating_colors(got_json[KEY_RATING_VIRTUAL], rating_pairs, 64, gray=True)
|
||||
else:
|
||||
pass
|
||||
if KEY_HTML in got_json:
|
||||
to_template.update({"EXTRA_INFO": got_json[KEY_HTML]})
|
||||
except (TypeError, ValueError, KeyError):
|
||||
pass
|
||||
to_template.update({"LIST_OTHER": build_other_list(q_pvc_by_id.sProfileOther)})
|
||||
# Партнёров считаем через ORM: так код проще читать и легче переносить между СУБД.
|
||||
q_merchant = (
|
||||
PriceOffer.objects.filter(
|
||||
kOffer2SetKit__kSet2PVCprofiles_id=model_id,
|
||||
sOfferActive=True,
|
||||
)
|
||||
.values(
|
||||
"kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__id",
|
||||
"kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__sMerchantName",
|
||||
"kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__pMerchantLogo",
|
||||
)
|
||||
.annotate(offers_by_merchant=Count("id"))
|
||||
.order_by("-offers_by_merchant", "kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__sMerchantName")
|
||||
)
|
||||
to_template.update({'MERCHANTS': [_merchant_row_to_dict(row) for row in q_merchant]})
|
||||
# Близкие профили этого же производителя нужны для быстрых переходов по карточкам.
|
||||
q_profiles = (
|
||||
PVCprofiles.objects.filter(sProfileManufacturer=q_pvc_by_id.sProfileManufacturer)
|
||||
.exclude(id=model_id)
|
||||
.values("id", "fProfileRating", "sProfileBriefDescription", "sProfileName")
|
||||
.order_by("fProfileRating")
|
||||
)
|
||||
to_template.update({'PROFILES': [_profile_row_to_dict(profile) for profile in q_profiles]})
|
||||
# Описание профиля берём через связку каталог -> блог: это один ORM-запрос вместо сырого SQL.
|
||||
q_profiles_detail = (
|
||||
Catalog2Profile.objects.filter(
|
||||
kProfile_id=model_id,
|
||||
sCatalogCardType=CATALOG_RECORD_FOR_PROFILE_MODEL,
|
||||
kBlogCatalog__isnull=False,
|
||||
)
|
||||
.select_related("kBlogCatalog")
|
||||
.order_by("kBlogCatalog__iCatalogSort")
|
||||
)
|
||||
profile_blog_posts = [row.kBlogCatalog for row in q_profiles_detail if row.kBlogCatalog is not None]
|
||||
to_template.update({'PROFILE_DETAIL': profile_blog_posts})
|
||||
# Картинка и дата публикации для meta-тегов берутся из связанного блога, если он есть.
|
||||
if profile_blog_posts:
|
||||
for blog_post in profile_blog_posts:
|
||||
if blog_post.sImgForBlogSocial:
|
||||
to_template['IMG_FOR_BLOG'] = blog_post.sImgForBlogSocial
|
||||
break
|
||||
|
||||
pub_dat: datetime = q_pvc_by_id.dProfileModify
|
||||
if profile_blog_posts:
|
||||
profile_blog_dat: datetime | None = max((post.dPostDataModify for post in profile_blog_posts), default=pub_dat)
|
||||
pub_dat = update_pub_dat(pub_dat, profile_blog_dat) or pub_dat
|
||||
to_template['PUB_DAT'] = pub_dat
|
||||
_append_visit_context(to_template, request, time_start)
|
||||
return render(request, "catalog/catalog_of_profiles_model.html", to_template)
|
||||
|
||||
|
||||
def catalog_profile_manufacture(request: HttpRequest, manufacture_id: int, manufacture_name: str) -> HttpResponse:
|
||||
"""
|
||||
КАТАЛОГ ПРОФИЛЕЙ: страница с описанием производителя профилей и списком марки производимых им профилей
|
||||
|
||||
:param request: HttpRequest -- входящий http-запрос
|
||||
:param manufacture_id: id профиля. Предполагается, что это первый id при сортировке по sProfileBriefDescription
|
||||
:param manufacture_name: название производителя (транслитерированное sanitize_slug())
|
||||
:return response: HttpResponse -- исходящий http-ответ
|
||||
"""
|
||||
time_start = time.perf_counter()
|
||||
manufacture_id = int(manufacture_id)
|
||||
q_pvc_by_id = PVCprofiles.objects.get(id=manufacture_id)
|
||||
if sanitize_slug(q_pvc_by_id.sProfileManufacturer) != manufacture_name:
|
||||
return redirect(f'/catalog/profile/{manufacture_id}-'
|
||||
f'{sanitize_slug(q_pvc_by_id.sProfileManufacturer)}')
|
||||
else:
|
||||
q_pvc_by_id = PVCprofiles.objects.order_by('id') \
|
||||
.filter(sProfileManufacturer=q_pvc_by_id.sProfileManufacturer).first()
|
||||
if q_pvc_by_id.id != manufacture_id:
|
||||
return redirect(f'/catalog/profile/{q_pvc_by_id.id}-'
|
||||
f'{sanitize_slug(q_pvc_by_id.sProfileManufacturer)}')
|
||||
to_template: dict[str, object] = {'CATALOG_MANUFACT': q_pvc_by_id.sProfileManufacturer,
|
||||
'CATALOG_MAN2URL': manufacture_name,
|
||||
'CATALOG_URL': f"{manufacture_id}-{manufacture_name}"}
|
||||
try:
|
||||
# Получаем статью-описание производителя через Catalog2Profile → BlogPosts.
|
||||
# GROUP BY из оригинального SQL здесь не нужен: нас устраивает любая первая запись.
|
||||
catalog_entry = (
|
||||
Catalog2Profile.objects.filter(
|
||||
kProfile__sProfileManufacturer=q_pvc_by_id.sProfileManufacturer,
|
||||
sCatalogCardType=CATALOG_RECORD_FOR_PROFILE_MANUFACTURER,
|
||||
kBlogCatalog__bCatalog=True,
|
||||
)
|
||||
.select_related("kBlogCatalog")
|
||||
.first()
|
||||
)
|
||||
if catalog_entry is None or catalog_entry.kBlogCatalog is None:
|
||||
raise ObjectDoesNotExist
|
||||
manufacture_description = catalog_entry.kBlogCatalog
|
||||
# PUB_DAT убран: на странице производителя дата меняется и от рейтинга, и от статьи,
|
||||
# поэтому Date4Meta/Last4Meta удалены из шаблона — base.html использует {% now %} по умолчанию.
|
||||
if PATH_FOR_IMG_BLOG in (manufacture_description.sImgForBlogSocial or ""):
|
||||
to_template.update({'IMG_FOR_BLOG': manufacture_description.sImgForBlogSocial})
|
||||
content = re.sub(r'<cut[\s\S]*>', '', manufacture_description.sPostContent, 0, re.IGNORECASE)
|
||||
to_template.update({'HEADER': manufacture_description.sPostHeader, 'CONTENT': content})
|
||||
to_template.update({'TIZER': re.sub(
|
||||
r'<script[\s\S]*?</script>|<style[\s\S]*?</style>|<iframe[\s\S]*?</iframe>',
|
||||
'', content, 0, re.IGNORECASE,
|
||||
)})
|
||||
except (ObjectDoesNotExist, IndexError, TypeError, KeyError):
|
||||
pass
|
||||
|
||||
# Список всех профилей этого производителя для навигации по карточкам.
|
||||
q_profiles = (
|
||||
PVCprofiles.objects.filter(sProfileManufacturer=q_pvc_by_id.sProfileManufacturer)
|
||||
.values("id", "fProfileRating", "sProfileBriefDescription", "sProfileName")
|
||||
.order_by("fProfileRating")
|
||||
)
|
||||
to_template.update({'PROFILES': [_profile_row_to_dict(p) for p in q_profiles]})
|
||||
|
||||
try:
|
||||
# Доля предложений этого производителя относительно всех предложений в базе.
|
||||
offers_by_manufacture = PriceOffer.objects.filter(
|
||||
kOffer2SetKit__kSet2PVCprofiles__sProfileManufacturer=q_pvc_by_id.sProfileManufacturer,
|
||||
).count()
|
||||
total_offers = PriceOffer.objects.count()
|
||||
offers_other = total_offers - offers_by_manufacture
|
||||
to_template.update({
|
||||
'OFFERS_BY_MAUFACTURE': offers_by_manufacture,
|
||||
'OFFERS_OTHER': offers_other,
|
||||
'OFFERS_ANGLE': 90 + 180 * normalize(offers_by_manufacture, total_offers),
|
||||
})
|
||||
if offers_by_manufacture > 0:
|
||||
# Партнёры, у которых есть предложения с профилями этого производителя.
|
||||
q_merchant = (
|
||||
PriceOffer.objects.filter(
|
||||
kOffer2SetKit__kSet2PVCprofiles__sProfileManufacturer=q_pvc_by_id.sProfileManufacturer,
|
||||
)
|
||||
.values(
|
||||
"kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__id",
|
||||
"kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__sMerchantName",
|
||||
"kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__pMerchantLogo",
|
||||
)
|
||||
.annotate(offers_by_merchant=Count("id"))
|
||||
.order_by(
|
||||
"-offers_by_merchant",
|
||||
"kOffer2SetKit__kSet2User__kMerchantOffice__kMerchantName__sMerchantName",
|
||||
)
|
||||
)
|
||||
to_template.update({'MERCHANTS': [_merchant_row_to_dict(row) for row in q_merchant]})
|
||||
except (ObjectDoesNotExist, IndexError, TypeError):
|
||||
pass
|
||||
_append_visit_context(to_template, request, time_start)
|
||||
return render(request, "catalog/catalog_of_profiles_manufacture.html", to_template)
|
||||
|
||||
515
oknardia/web/catalog_series.py
Normal file
515
oknardia/web/catalog_series.py
Normal file
@@ -0,0 +1,515 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
from pathlib import Path
|
||||
from django.core.exceptions import ObjectDoesNotExist
|
||||
from django.db.models import Count, F, IntegerField, Value
|
||||
from django.shortcuts import render, redirect
|
||||
from django.http import HttpRequest, HttpResponse
|
||||
from django.template.loader import render_to_string
|
||||
from oknardia.settings import *
|
||||
from oknardia.models import (
|
||||
Apartment_Type,
|
||||
MountDim2Apartment,
|
||||
PriceOffer,
|
||||
Seria_Info,
|
||||
Win_MountDim,
|
||||
Building_Info,
|
||||
)
|
||||
from web.report1 import get_last_all_user_visit_list
|
||||
from web.add_func import get_flaps_for_big_pictures, sanitize_slug
|
||||
import time
|
||||
import os
|
||||
import math
|
||||
import base64
|
||||
import json
|
||||
|
||||
|
||||
def _append_visit_context(to_template: dict, request: HttpRequest, time_start: float) -> None:
|
||||
"""Дописывает в контекст стандартный хвост: визиты и время выполнения."""
|
||||
to_template.update({
|
||||
'LOG_VISIT': get_last_all_user_visit_list(),
|
||||
'ticks': float(time.perf_counter() - time_start),
|
||||
})
|
||||
|
||||
# Каталог типовых серий зданий.
|
||||
def catalog_seria(request: HttpRequest) -> HttpResponse:
|
||||
"""
|
||||
КАТАЛОГ ТИПОВЫХ СЕРИЙ: выводит список корневых серий из каталога.
|
||||
|
||||
:param request: HttpRequest -- входящий http-запрос
|
||||
:return response: HttpResponse -- исходящий http-ответ
|
||||
"""
|
||||
time_start = time.perf_counter()
|
||||
# Только корневые серии (id == kRoot_id), сортировка как в старом SQL.
|
||||
q_seria = (
|
||||
Seria_Info.objects.filter(id=F('kRoot_id'))
|
||||
.values('id', 'sURL2IMG', 'sName')
|
||||
.order_by('sName')
|
||||
)
|
||||
to_template: dict[str, object] = {
|
||||
'SERIAS': [
|
||||
{
|
||||
'ID': row['id'],
|
||||
'URL': row['sURL2IMG'],
|
||||
'NAME': row['sName'],
|
||||
'NAME_T': sanitize_slug(row['sName']),
|
||||
}
|
||||
for row in q_seria
|
||||
]
|
||||
}
|
||||
_append_visit_context(to_template, request, time_start)
|
||||
return render(request, "catalog/catalog_seria.html", to_template)
|
||||
|
||||
|
||||
def catalog_seria_info(
|
||||
request: HttpRequest,
|
||||
seria_name_translit: str | None,
|
||||
seria_id: int = DEFAULT_SERIA_ID_FOR_CATALOG,
|
||||
) -> HttpResponse:
|
||||
"""
|
||||
КАТАЛОГ ТИПОВОЙ СЕРИИ: детальная страница по серии домов.
|
||||
|
||||
Что делает вьюха:
|
||||
- канонизирует URL (root-id серии + корректный slug),
|
||||
- собирает таблицу окон по типам квартир,
|
||||
- для "тяжелого" режима дополнительно готовит навигацию/график/гео-данные
|
||||
и сохраняет pre-render include-шаблон для последующих быстрых ответов.
|
||||
|
||||
:param request: HttpRequest -- входящий http-запрос
|
||||
:param seria_name_translit: str -- имя серии здания (транслитерированное через pytils)
|
||||
:param seria_id: int -- id серии
|
||||
:return response: HttpResponse -- исходящий http-ответ
|
||||
"""
|
||||
time_start = time.perf_counter()
|
||||
# Канонизируем URL: страница серии должна открываться только по корневой серии и правильному slug.
|
||||
try:
|
||||
seria_id = int(seria_id)
|
||||
q_seria = Seria_Info.objects.only("id", "kRoot_id", "sName").get(id=seria_id)
|
||||
if q_seria.id != q_seria.kRoot_id or seria_name_translit != sanitize_slug(q_seria.sName):
|
||||
return redirect(f"/catalog/seria/{sanitize_slug(q_seria.sName)}/all{seria_id}")
|
||||
except (ObjectDoesNotExist, ValueError):
|
||||
return redirect("/catalog/")
|
||||
|
||||
# В DEV отключаем pre-render cache: всегда рендерим «тяжелый» шаблон напрямую,
|
||||
# чтобы тестировать актуальную серверную логику, а не сохраненный html-файл.
|
||||
if DEBUG:
|
||||
light_template = "seria_info/all_seria_info_pre_light.html"
|
||||
static_include_path = "" # в DEV не используем кеш
|
||||
is_hard_template = True
|
||||
else:
|
||||
# В PROD используем существующий pre-render include для статических данных (если есть).
|
||||
light_template = "seria_info/all_seria_info_pre_light.html"
|
||||
static_template_filename = f"seria_info/prepared/{seria_id}_id_static.html"
|
||||
static_template_w_path = f"{TEMPLATES[0]['DIRS'][0]}/{static_template_filename}"
|
||||
is_hard_template = not os.path.isfile(static_template_w_path)
|
||||
static_include_path = static_template_filename if not is_hard_template else ""
|
||||
|
||||
to_template: dict[str, object] = {}
|
||||
# Получаем все уникальные проемы серии и сразу добавляем iQuantity=1
|
||||
# для совместимости с get_flaps_for_big_pictures().
|
||||
list_win_in_seria = list(
|
||||
Win_MountDim.objects.filter(kApartment__kSeria_id=seria_id)
|
||||
.annotate(iQuantity=Value(1, output_field=IntegerField()))
|
||||
.only(
|
||||
"id",
|
||||
"iWinWidth",
|
||||
"iWinHight",
|
||||
"sDescripion",
|
||||
"bIsDoor",
|
||||
"bIsNearDoor",
|
||||
"sFlapConfig",
|
||||
"iWinDepth",
|
||||
)
|
||||
.order_by("-bIsNearDoor", "-bIsDoor", "iWinWidth", "-iWinHight", "id")
|
||||
.distinct()
|
||||
)
|
||||
|
||||
window_ids = [win.id for win in list_win_in_seria]
|
||||
apartments_in_seria = list(
|
||||
Apartment_Type.objects.filter(kSeria_id=seria_id)
|
||||
.values("id", "sNameApartment")
|
||||
.order_by("iSort", "id")
|
||||
)
|
||||
apartment_ids = [apartment["id"] for apartment in apartments_in_seria]
|
||||
|
||||
# Кэшируем количество проемов по паре (квартира, проем), чтобы не делать N*M обращений к БД.
|
||||
quantities_by_pair = {
|
||||
(row["kApartment_id"], row["kMountDim_id"]): row["iQuantity"]
|
||||
for row in MountDim2Apartment.objects.filter(
|
||||
kApartment_id__in=apartment_ids,
|
||||
kMountDim_id__in=window_ids,
|
||||
).values("kApartment_id", "kMountDim_id", "iQuantity")
|
||||
}
|
||||
# Число офферов считаем один раз по каждому проему и переиспользуем при сборке таблицы.
|
||||
offers_by_window = {
|
||||
row["kOffer2MountDim_id"]: row["num_offers"]
|
||||
for row in PriceOffer.objects.filter(kOffer2MountDim_id__in=window_ids)
|
||||
.values("kOffer2MountDim_id")
|
||||
.annotate(num_offers=Count("id"))
|
||||
}
|
||||
|
||||
total_column = len(list_win_in_seria) - 1
|
||||
table_of_win_in_seria_by_apartmment = []
|
||||
offer_and_merchant_per_win = [
|
||||
{
|
||||
"WIN_OFFER": offers_by_window.get(list_win_in_seria[i].id, 0),
|
||||
"WIN_MERCHANT": 0,
|
||||
"WIN_W": list_win_in_seria[i].iWinWidth,
|
||||
"WIN_H": list_win_in_seria[i].iWinHight,
|
||||
"WIN_ID": list_win_in_seria[i].id,
|
||||
}
|
||||
for i in range(total_column + 1)
|
||||
]
|
||||
|
||||
for apartment in apartments_in_seria:
|
||||
row_for_table = []
|
||||
# None = в строке квартиры еще не встретилось ни одного окна.
|
||||
min_offer_in_row = None
|
||||
for count_column, window in enumerate(list_win_in_seria):
|
||||
quantity = quantities_by_pair.get((apartment["id"], window.id), 0)
|
||||
if quantity != 0:
|
||||
num_offers = offers_by_window.get(window.id, 0)
|
||||
row_for_table.append(
|
||||
{
|
||||
"WIN_NUM": [chr(65 + count_column)],
|
||||
"WIN_Q": quantity,
|
||||
"WIN_ID": window.id,
|
||||
"WIN_WIDTH": window.iWinWidth,
|
||||
"WIN_HEIGHT": window.iWinHight,
|
||||
"WIN_DESCRIPTION": window.sDescripion,
|
||||
"WIN_FLAPCFG": window.sFlapConfig,
|
||||
}
|
||||
)
|
||||
if min_offer_in_row is None or min_offer_in_row > num_offers:
|
||||
min_offer_in_row = num_offers
|
||||
else:
|
||||
row_for_table.append({"WIN_NUM": "—"})
|
||||
|
||||
table_of_win_in_seria_by_apartmment.append(
|
||||
{
|
||||
"WIN_IN_APART": row_for_table,
|
||||
"APART_NAME": apartment["sNameApartment"],
|
||||
"APART_ID": apartment["id"],
|
||||
# Если у серии нет ни одного окна, показываем 0 вместо служебного sentinel.
|
||||
"NUM_OFFERS": 0 if min_offer_in_row is None else min_offer_in_row,
|
||||
}
|
||||
)
|
||||
|
||||
to_template.update(
|
||||
{
|
||||
"WIN_OFFER_AND_MERCHANT": offer_and_merchant_per_win,
|
||||
"TABLE_OF_WINDOWS": table_of_win_in_seria_by_apartmment,
|
||||
# Первая квартира из таблицы (нужна для картоки в пре-рендер шаблоне)
|
||||
"first_apart_id": table_of_win_in_seria_by_apartmment[0]["APART_ID"] if table_of_win_in_seria_by_apartmment else 0,
|
||||
}
|
||||
)
|
||||
|
||||
# Для "тяжелого" шаблона получаем навигацию, карту и график.
|
||||
# ВАЖНО: таблица окон (TABLE_OF_WINDOWS) считается ВСЕГДА — она не кешируется!
|
||||
if is_hard_template:
|
||||
to_template.update(get_flaps_for_big_pictures(list_win_in_seria))
|
||||
seria_id, for_seria_nav = seria_nav(seria_id)
|
||||
to_template.update(for_seria_nav)
|
||||
to_template.update(seria_info_year(seria_id))
|
||||
to_template.update(seria_info_geo_code(seria_id))
|
||||
|
||||
if not DEBUG:
|
||||
# Пре-рендер ТРЁХ отдельных файлов для статических данных.
|
||||
# Верхняя статья НЕ кешируется — она рендерится динамически, чтобы изменения
|
||||
# через админку были видны сразу без перезагрузки контейнера.
|
||||
prepared_dir = Path(TEMPLATES[0]["DIRS"][0]) / PATH_FOR_SERIA_INFO_HTML_INCLUDE
|
||||
prepared_dir.mkdir(parents=True, exist_ok=True)
|
||||
|
||||
# 1. Схемы открывания и размеры
|
||||
string_flaps = render_to_string(
|
||||
"seria_info/all_seria_info_pre_light_static_flaps.html",
|
||||
to_template
|
||||
)
|
||||
file_flaps = prepared_dir / f"{seria_id}_id_static_flaps.html"
|
||||
with open(file_flaps, "w", encoding="utf-8") as f:
|
||||
f.write(string_flaps)
|
||||
|
||||
# 2. График ввода в эксплуатацию
|
||||
string_graph = render_to_string(
|
||||
"seria_info/all_seria_info_pre_light_static_graph.html",
|
||||
to_template
|
||||
)
|
||||
file_graph = prepared_dir / f"{seria_id}_id_static_graph.html"
|
||||
with open(file_graph, "w", encoding="utf-8") as f:
|
||||
f.write(string_graph)
|
||||
|
||||
# 3. Карта и статистика
|
||||
string_map_stats = render_to_string(
|
||||
"seria_info/all_seria_info_pre_light_static_map_stats.html",
|
||||
to_template
|
||||
)
|
||||
file_map_stats = prepared_dir / f"{seria_id}_id_static_map_stats.html"
|
||||
with open(file_map_stats, "w", encoding="utf-8") as f:
|
||||
f.write(string_map_stats)
|
||||
|
||||
# Добавляем в контекст пути к кешируемым файлам (верхняя статья всегда динамична)
|
||||
pre_rendered_flaps_path = ""
|
||||
pre_rendered_graph_path = ""
|
||||
pre_rendered_map_stats_path = ""
|
||||
|
||||
if not DEBUG:
|
||||
# В production используем кеширующие файлы, если они существую<D183><D18E>
|
||||
prepared_dir = Path(TEMPLATES[0]["DIRS"][0]) / PATH_FOR_SERIA_INFO_HTML_INCLUDE
|
||||
|
||||
file_flaps = prepared_dir / f"{seria_id}_id_static_flaps.html"
|
||||
file_graph = prepared_dir / f"{seria_id}_id_static_graph.html"
|
||||
file_map_stats = prepared_dir / f"{seria_id}_id_static_map_stats.html"
|
||||
|
||||
if file_flaps.exists():
|
||||
pre_rendered_flaps_path = f"seria_info/prepared/{seria_id}_id_static_flaps.html"
|
||||
if file_graph.exists():
|
||||
pre_rendered_graph_path = f"seria_info/prepared/{seria_id}_id_static_graph.html"
|
||||
if file_map_stats.exists():
|
||||
pre_rendered_map_stats_path = f"seria_info/prepared/{seria_id}_id_static_map_stats.html"
|
||||
|
||||
to_template.update({
|
||||
"THIS_SERIA_NAME": q_seria.sName,
|
||||
"PRE_RENDERED_STATIC_FLAPS_PATH": pre_rendered_flaps_path,
|
||||
"PRE_RENDERED_STATIC_GRAPH_PATH": pre_rendered_graph_path,
|
||||
"PRE_RENDERED_STATIC_MAP_STATS_PATH": pre_rendered_map_stats_path,
|
||||
})
|
||||
|
||||
|
||||
_append_visit_context(to_template, request, time_start)
|
||||
return render(request, light_template, to_template)
|
||||
|
||||
|
||||
def seria_nav(seria_id: int = DEFAULT_SERIA_ID_FOR_CATALOG) -> tuple[int, dict]:
|
||||
"""
|
||||
Возвращает корректный seria_id и данные навигации по корневым сериям.
|
||||
|
||||
Если переданный seria_id невалиден, подбирает ближайший допустимый root-id.
|
||||
|
||||
:param seria_id: id серии
|
||||
:return: tuple[int, dict] -- (seria_id, {"SERIA_NAV_DIM": ..., "THIS_SERIA_*": ...})
|
||||
"""
|
||||
q_seria = list(
|
||||
Seria_Info.objects.filter(id=F("kRoot_id"))
|
||||
# sURL2IMG нужен для OG-image в шаблоне seria_info
|
||||
.only("id", "sName", "sSeriaDescription", "kRoot_id", "kParent_id", "sURL2IMG")
|
||||
.order_by("sName")
|
||||
)
|
||||
if not q_seria:
|
||||
return seria_id, {"SERIA_NAV_DIM": []}
|
||||
error_seria = True
|
||||
for count_seria in q_seria:
|
||||
if count_seria.id == int(seria_id):
|
||||
error_seria = False
|
||||
break
|
||||
if error_seria:
|
||||
# Ошибочный seria_id. Такой базовой серии нет и надо ее найти.
|
||||
try:
|
||||
query = Seria_Info.objects.get(id=int(seria_id))
|
||||
if query.kRoot_id is not None:
|
||||
# базовая серия прописана в kRoot_id
|
||||
seria_id = query.kRoot_id
|
||||
else:
|
||||
# Корневой серии нет.
|
||||
# Ищем методом наименьших расстояний
|
||||
min_min = 100_000_000
|
||||
min_id = seria_id
|
||||
for count_seria in q_seria:
|
||||
if math.fabs(int(seria_id) - count_seria.id) < min_min:
|
||||
min_min = math.fabs(int(seria_id) - count_seria.id)
|
||||
min_id = count_seria.id
|
||||
seria_id = min_id
|
||||
except ObjectDoesNotExist:
|
||||
seria_id = q_seria[0].id
|
||||
return all_seria_nav(seria_id, q_seria)
|
||||
|
||||
|
||||
def all_seria_nav(seria_id: int, q_seria) -> tuple[int, dict]:
|
||||
"""
|
||||
Формирует структуру навигации по сериям для шаблонов.
|
||||
|
||||
:param seria_id: активный id серии
|
||||
:param q_seria: коллекция серий (ORM-объекты или dict из values())
|
||||
:return: tuple[int, dict] -- (seria_id, словарь с SERIA_NAV_DIM и данными активной серии)
|
||||
"""
|
||||
seria_nav_dim = []
|
||||
this_return = {}
|
||||
# Поддерживаем оба формата входных элементов: ORM-объекты и dict из values().
|
||||
for count_seria in q_seria:
|
||||
seria_name = count_seria["sName"] if isinstance(count_seria, dict) else count_seria.sName
|
||||
seria_id_value = count_seria["id"] if isinstance(count_seria, dict) else count_seria.id
|
||||
seria_description = (
|
||||
count_seria.get("sSeriaDescription")
|
||||
if isinstance(count_seria, dict)
|
||||
else count_seria.sSeriaDescription
|
||||
)
|
||||
one_seria = {
|
||||
"SERIA_R": seria_name,
|
||||
"ID2URL": seria_id_value,
|
||||
"SERIA_L": sanitize_slug(seria_name),
|
||||
}
|
||||
if seria_id_value == seria_id:
|
||||
# Изображение серии: используется в OG-image в шаблоне seria_info
|
||||
seria_image = (
|
||||
count_seria.get("sURL2IMG")
|
||||
if isinstance(count_seria, dict)
|
||||
else count_seria.sURL2IMG
|
||||
)
|
||||
this_return.update({
|
||||
"THIS_SERIA_NAME": seria_name,
|
||||
"THIS_SERIA_DESCRIPTION": seria_description,
|
||||
# ID и slug серии нужны для canonical URL и JSON-LD в шаблоне
|
||||
"THIS_SERIA_ID": seria_id_value,
|
||||
"THIS_SERIA_NAME_T": sanitize_slug(seria_name),
|
||||
# URL изображения серии для OG-тегов (путь относительно /media/)
|
||||
"THIS_SERIA_IMAGE_URL": str(seria_image) if seria_image else "",
|
||||
})
|
||||
seria_nav_dim.append(one_seria)
|
||||
this_return.update({"SERIA_NAV_DIM": seria_nav_dim})
|
||||
return seria_id, this_return
|
||||
|
||||
|
||||
def seria_info_year(seria_id: int = DEFAULT_SERIA_ID_FOR_CATALOG) -> dict:
|
||||
"""Возвращает данные для графика ввода домов серии в эксплуатацию.
|
||||
|
||||
:param seria_id: int -- id корневой серии
|
||||
:return: dict -- данные для графика по годам вида:
|
||||
{"DATA4GRAPH": [{'YEAR': 1997, 'NUMS': 1, 'CLRS': '99'},
|
||||
{'YEAR': 1998, 'NUMS': 15, 'CLRS': 'сс'},
|
||||
{'YEAR': 1998, 'NUMS': 10, 'CLRS': 'a9'}
|
||||
]
|
||||
}
|
||||
"""
|
||||
seria_in_years = []
|
||||
query = list(
|
||||
Building_Info.objects.filter(kSeria_Link__kRoot_id=seria_id)
|
||||
.values("iCommissioning_year")
|
||||
.annotate(NumInYear=Count("iCommissioning_year"))
|
||||
.order_by("iCommissioning_year")
|
||||
)
|
||||
max_per_year = 0
|
||||
graph_color_light = 0xCC # самый светлый цвет на графике (максимальное значение)
|
||||
graph_color_dark = 0x99 # самый темный цвет на графике (минимальное значение)
|
||||
for year_count in query:
|
||||
if int(year_count["NumInYear"]) > max_per_year:
|
||||
max_per_year = int(year_count["NumInYear"])
|
||||
for year_count in query:
|
||||
data_of_year = {}
|
||||
try:
|
||||
data_of_year.update({
|
||||
"YEAR": int(year_count["iCommissioning_year"]),
|
||||
"NUMS": year_count["NumInYear"],
|
||||
"CLRS": str(hex(int(graph_color_dark + year_count["NumInYear"] * (
|
||||
graph_color_light - graph_color_dark) / max_per_year)))[2:]
|
||||
})
|
||||
except ValueError:
|
||||
continue
|
||||
seria_in_years.append(data_of_year)
|
||||
return {"DATA4GRAPH": seria_in_years}
|
||||
|
||||
|
||||
def seria_info_geo_code(seria_id: int | str = DEFAULT_SERIA_ID_FOR_CATALOG) -> dict:
|
||||
"""Возвращает гео-точки и агрегированную статистику по серии.
|
||||
|
||||
Кроме массива координат, функция считает суммарные показатели серии:
|
||||
жилые/муниципальные/государственные площади, число жителей, квартир,
|
||||
лицевых счетов и диапазон показателя состояния домов.
|
||||
|
||||
:param seria_id: int | str -- id серии, для которой нужно получить данные.
|
||||
:return: dict -- {
|
||||
"DATA4GEO": [...],
|
||||
"MUNICIPAL_M2": ...,
|
||||
"RESIDENTIAL_M2": ...,
|
||||
"GOVERNMENT_M2": ...,
|
||||
"RESIDENTS": ...,
|
||||
"APARTMENTS": ...,
|
||||
"ACCOUNTS": ...,
|
||||
"CONDITION_MAX": ...,
|
||||
"CONDITION_MIN": ...,
|
||||
}
|
||||
"""
|
||||
data_return = {}
|
||||
seria_to_geo = []
|
||||
municipal_m2 = 0 # муниципальный фонд (кв.м)
|
||||
residential_m2 = 0 # жилой фонд (кв.м)
|
||||
government_m2 = 0 # государственные учреждения занимают (кв.м.)
|
||||
residents = 0 # количество жильцов
|
||||
apartments = 0 # число квартир
|
||||
accounts = 0 # количество лицевых счетов
|
||||
condition_max = 0 # максимальное значение показателя состояния здания
|
||||
condition_min = 1_000_000 # минимальное значение показателя состояния здания
|
||||
query = Building_Info.objects.filter(kSeria_Link__kRoot_id=int(seria_id)).values(
|
||||
"id",
|
||||
"kSeria_Link__kRoot_id",
|
||||
"sAddress",
|
||||
"fResidential_Area",
|
||||
"fMunicipal_Area",
|
||||
"fGovernment_Area",
|
||||
"iNum_Residents",
|
||||
"iNum_Apartments",
|
||||
"iNum_Accounts",
|
||||
"fCondition_House",
|
||||
"fGeoCode_Latitude",
|
||||
"fGeoCode_Longitude",
|
||||
)
|
||||
# iterator() уменьшает пиковое потребление памяти на больших сериях домов.
|
||||
for count in query.iterator(chunk_size=500):
|
||||
latitude = count["fGeoCode_Latitude"] or 0
|
||||
longitude = count["fGeoCode_Longitude"] or 0
|
||||
municipal_area = count["fMunicipal_Area"] or 0
|
||||
residential_area = count["fResidential_Area"] or 0
|
||||
government_area = count["fGovernment_Area"] or 0
|
||||
num_residents = count["iNum_Residents"] or 0
|
||||
num_apartments = count["iNum_Apartments"] or 0
|
||||
num_accounts = count["iNum_Accounts"] or 0
|
||||
house_condition = count["fCondition_House"] or 0
|
||||
|
||||
if int(latitude) != 0 and int(longitude) != 0:
|
||||
seria_to_geo.append({"LATITUDE": latitude,
|
||||
"LONGITUDE": longitude,
|
||||
"ADDR_ID": count["id"],
|
||||
"ADDR_LAT": sanitize_slug(count["sAddress"]),
|
||||
"ADDR_RUS": count["sAddress"],
|
||||
"SER_ID": count["kSeria_Link__kRoot_id"]
|
||||
})
|
||||
if municipal_area > 0:
|
||||
municipal_m2 += municipal_area
|
||||
if residential_area > 0:
|
||||
residential_m2 += residential_area
|
||||
if government_area > 0:
|
||||
government_m2 += government_area
|
||||
if num_residents > 0:
|
||||
residents += num_residents
|
||||
if num_apartments > 0:
|
||||
apartments += num_apartments
|
||||
if num_accounts > 0:
|
||||
accounts += num_accounts
|
||||
if house_condition > 0:
|
||||
if house_condition > condition_max:
|
||||
condition_max = house_condition
|
||||
if house_condition < condition_min:
|
||||
condition_min = house_condition
|
||||
data_return.update({"DATA4GEO": seria_to_geo,
|
||||
"MUNICIPAL_M2": municipal_m2,
|
||||
"RESIDENTIAL_M2": residential_m2,
|
||||
"GOVERNMENT_M2": government_m2,
|
||||
"RESIDENTS": residents,
|
||||
"APARTMENTS": apartments,
|
||||
"ACCOUNTS": accounts,
|
||||
"CONDITION_MAX": condition_max,
|
||||
"CONDITION_MIN": condition_min})
|
||||
|
||||
# Кодируем геоданные в Base64 для защиты (используется в статик-шаблонах)
|
||||
# Формат: [latitude, longitude, addr_id, seria_id] для каждого здания
|
||||
geo_for_encoding = []
|
||||
for geo_point in seria_to_geo:
|
||||
geo_for_encoding.append([
|
||||
float(geo_point["LATITUDE"]),
|
||||
float(geo_point["LONGITUDE"]),
|
||||
geo_point["ADDR_ID"],
|
||||
geo_point["SER_ID"]
|
||||
])
|
||||
|
||||
geo_json = json.dumps(geo_for_encoding, separators=(',', ':'))
|
||||
geo_b64 = base64.b64encode(geo_json.encode('utf-8')).decode('utf-8')
|
||||
data_return["DATA4GEO_B64"] = geo_b64
|
||||
|
||||
return data_return
|
||||
0
oknardia/web/catalog_utils.py
Normal file
0
oknardia/web/catalog_utils.py
Normal file
@@ -1,96 +1,59 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
from django.shortcuts import render
|
||||
from django.http import HttpRequest, HttpResponse
|
||||
from django.core.exceptions import ObjectDoesNotExist
|
||||
from django.db.models import Count, Sum, F
|
||||
from time import time
|
||||
from oknardia.settings import *
|
||||
from oknardia.models import Seria_Info
|
||||
from web.catalog import all_seria_nav
|
||||
# from oknardia.catalog import all_seria_nav
|
||||
import math
|
||||
from oknardia.models import Building_Info
|
||||
from web.catalog_series import seria_nav # Используем уже существующую seria_nav из catalog_series
|
||||
import os
|
||||
import pytils # вместо Rus2Lat(smth) --> pytils.translit.slugify(smth).lower()
|
||||
|
||||
|
||||
# возвращает корректный seria_id и кортеж для построения навигации по сериям дома
|
||||
def seria_nav(i_seria_id: int = 12) -> (int, dict):
|
||||
query_seria = Seria_Info.objects.raw(
|
||||
'SELECT oknardia_seria_info.id,'
|
||||
' oknardia_seria_info.sName,'
|
||||
' oknardia_seria_info.sSeriaDescription,'
|
||||
' oknardia_seria_info.kRoot_id,'
|
||||
' oknardia_seria_info.kParent_id '
|
||||
'FROM oknardia_seria_info '
|
||||
'WHERE oknardia_seria_info.id = oknardia_seria_info.kRoot_id '
|
||||
'ORDER BY oknardia_seria_info.sName;')
|
||||
error_seria = True
|
||||
for count_seria in query_seria:
|
||||
if count_seria.id == int(i_seria_id):
|
||||
error_seria = False
|
||||
break
|
||||
if error_seria:
|
||||
# Ошибочный seria_id. Такой базовой серии нет и надо ее найти.
|
||||
try:
|
||||
query = Seria_Info.objects.get(id=int(i_seria_id))
|
||||
if query.kRoot_id is None:
|
||||
# базовая серия прописана в kRoot_id
|
||||
i_seria_id = query.kRoot_id
|
||||
else:
|
||||
# == корневой нет
|
||||
# == ищем методом наименьших расстояний
|
||||
min_min = 100000000
|
||||
min_id = i_seria_id
|
||||
for count_seria in query_seria:
|
||||
if math.fabs(int(i_seria_id) - count_seria.id) < min_min:
|
||||
min_min = math.fabs(int(i_seria_id) - count_seria.id)
|
||||
min_id = count_seria.id
|
||||
i_seria_id = min_id
|
||||
except ObjectDoesNotExist:
|
||||
i_seria_id = query_seria[0].id
|
||||
# print(f"-->{seria_id}<--")
|
||||
return all_seria_nav(i_seria_id, query_seria)
|
||||
|
||||
|
||||
def statistic_menu(request: HttpRequest) -> HttpResponse:
|
||||
""" Страница "Статистика" в главном меню
|
||||
|
||||
ВНИМАНИЕ: ТЕХНИЧЕСКИЙ ДОЛГ -- выводятся данные только по сериям зданий. Этого маловато.
|
||||
Можно добавить данные по проемам, предложениям, график распределения цен и т.п.
|
||||
ВНИМАНИЕ: выводятся данные только по сериям зданий. Этого маловато.
|
||||
В будущем, наверное, стоит добавить данные по проемам, предложениям, график распределения цен и т.п.
|
||||
|
||||
:param request: HttpRequest -- входящий http-запрос
|
||||
:return: HttpResponse -- исходящий http-ответ
|
||||
"""
|
||||
time_start = time()
|
||||
to_template = {}
|
||||
to_template: dict[str, object] = {}
|
||||
|
||||
# Используем seria_nav из web.catalog_series, которая уже на ORM
|
||||
seria_id, for_seria_nav = seria_nav(0)
|
||||
to_template.update(for_seria_nav)
|
||||
|
||||
# проверяем какой JS с картами и PieCharts: упакованные или нет (откуда берётся не упакованный -- не помню)
|
||||
path_name = f"{STATIC_BASE_PATH}/{PATH_FOR_JS_MAP}"
|
||||
# print(path_name)
|
||||
if os.path.isfile(f"{path_name}/_ALL{SUFFIX_FOR_MINI_JS_MAP}"):
|
||||
to_template.update({'MAP_JS': f"{PATH_FOR_JS_MAP}/_ALL{SUFFIX_FOR_MINI_JS_MAP}"})
|
||||
else:
|
||||
to_template.update({'MAP_JS': f"{PATH_FOR_JS_MAP}/_ALL{SUFFIX_FOR_JS_MAP}"})
|
||||
# строим диаграмму сколько каких серий и каковы их площади...
|
||||
q_seria_pie = Seria_Info.objects.raw(
|
||||
"SELECT"
|
||||
" oknardia_seria_info.kRoot_id as id,"
|
||||
" COUNT(oknardia_building_info.id) AS num_building,"
|
||||
" SUM(oknardia_building_info.fTotal_Area) AS area_m2 "
|
||||
"FROM oknardia_building_info"
|
||||
" INNER JOIN oknardia_seria_info"
|
||||
" ON oknardia_building_info.kSeria_Link_id = oknardia_seria_info.id "
|
||||
"WHERE oknardia_seria_info.kRoot_id IS NOT NULL "
|
||||
"GROUP BY oknardia_seria_info.kRoot_id "
|
||||
"ORDER BY num_building DESC;")
|
||||
|
||||
# Строим диаграмму, сколько каких серий и каковы их площади...
|
||||
# Переписано с raw SQL на ORM
|
||||
q_seria_pie_orm = (
|
||||
Building_Info.objects
|
||||
.filter(kSeria_Link__kRoot_id__isnull=False)
|
||||
.values('kSeria_Link__kRoot_id')
|
||||
.annotate(
|
||||
id=F('kSeria_Link__kRoot_id'), # Переименовываем для соответствия старому контракту
|
||||
num_building=Count('id'),
|
||||
area_m2=Sum('fTotal_Area')
|
||||
)
|
||||
.order_by('-num_building')
|
||||
)
|
||||
|
||||
data2pie = []
|
||||
for count in q_seria_pie:
|
||||
for count in q_seria_pie_orm:
|
||||
data2pie.append({
|
||||
"ID": count.id,
|
||||
"AREA_M2": count.area_m2,
|
||||
"NUM_BUILDING": count.num_building
|
||||
"ID": count['id'], # Доступ к полям через словарь, т.к. values() возвращает dict
|
||||
"AREA_M2": count['area_m2'],
|
||||
"NUM_BUILDING": count['num_building']
|
||||
})
|
||||
# print(data2pie)
|
||||
|
||||
to_template.update({'DATA2PIE': data2pie})
|
||||
to_template.update({'ticks': float(time()-time_start)})
|
||||
return render(request, "seria_info/all_stat.html", to_template)
|
||||
|
||||
1
oknardia/web/management/__init__.py
Normal file
1
oknardia/web/management/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
1
oknardia/web/management/commands/__init__.py
Normal file
1
oknardia/web/management/commands/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
330
oknardia/web/management/commands/generate_map_js.py
Normal file
330
oknardia/web/management/commands/generate_map_js.py
Normal file
@@ -0,0 +1,330 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
Django management command: generate_map_js
|
||||
|
||||
Генерирует JavaScript-файлы для отрисовки карт с геоданными зданий типовых серий.
|
||||
|
||||
Процесс:
|
||||
1. Получает все корневые серии (где id = kRoot_id)
|
||||
2. Собирает геоданные всех зданий для этих серий
|
||||
3. Генерирует JavaScript файл public/static/js/4maps/_ALL_seria_on_map.js
|
||||
4. Файл содержит координаты всех зданий, привязанные к сериям
|
||||
|
||||
Структура генерируемого файла:
|
||||
- Массив цветов для каждой серии (DimColor)
|
||||
- Объявление переменных серий (c<ID>, s<ID>)
|
||||
- Инициализация Yandex.Maps с PlaceMarks всех зданий
|
||||
"""
|
||||
|
||||
from django.core.management.base import BaseCommand
|
||||
from django.template.loader import render_to_string
|
||||
from oknardia.models import Seria_Info, Building_Info
|
||||
from web.add_func import sanitize_slug
|
||||
from oknardia.settings import STATIC_BASE_PATH, PATH_FOR_JS_MAP, SUFFIX_FOR_JS_MAP
|
||||
import os
|
||||
import time
|
||||
import base64
|
||||
import json
|
||||
|
||||
try:
|
||||
import rjsmin as _rjsmin
|
||||
RJSMIN_AVAILABLE = True
|
||||
except ImportError:
|
||||
RJSMIN_AVAILABLE = False
|
||||
_rjsmin = None
|
||||
|
||||
|
||||
def seria_info_geo_code(seria_ids_str):
|
||||
"""
|
||||
Собирает геоданные для конкретных серий в компактном формате для обфускации.
|
||||
|
||||
Args:
|
||||
seria_ids_str: строка с ID серий через запятую (например "1,8,12,24")
|
||||
|
||||
Returns:
|
||||
dict с ключами:
|
||||
- DATA4GEO: список точек с координатами и информацией о зданиях
|
||||
- DATA4GEO_B64: Base64-закодированный JSON координат (обфускация)
|
||||
- MUNICIPAL_M2, RESIDENTIAL_M2, GOVERNMENT_M2: площади
|
||||
- RESIDENTS, APARTMENTS, ACCOUNTS: количества
|
||||
- CONDITION_MAX, CONDITION_MIN: условия зданий
|
||||
"""
|
||||
seria_ids = [int(id_str.strip()) for id_str in seria_ids_str.split(',') if id_str.strip()]
|
||||
|
||||
data_return = {}
|
||||
seria_2_geo = []
|
||||
geo_compact = [] # Компактный формат для обфускации: [lat, lon, id, ser_id]
|
||||
municipal_m2 = 0
|
||||
residential_m2 = 0
|
||||
government_m2 = 0
|
||||
residents = 0
|
||||
apartments = 0
|
||||
accounts = 0
|
||||
condition_max = 0
|
||||
condition_min = 1000000
|
||||
|
||||
# ORM запрос вместо raw SQL
|
||||
query = Building_Info.objects.filter(
|
||||
kSeria_Link__kRoot_id__in=seria_ids
|
||||
).select_related('kSeria_Link')
|
||||
|
||||
for building in query:
|
||||
# Проверяем наличие координат (не нулевые)
|
||||
if building.fGeoCode_Latitude and building.fGeoCode_Longitude:
|
||||
if int(building.fGeoCode_Latitude) != 0 and int(building.fGeoCode_Longitude) != 0:
|
||||
data_of_point = {
|
||||
"LATITUDE": building.fGeoCode_Latitude,
|
||||
"LONGITUDE": building.fGeoCode_Longitude,
|
||||
"ADDR_ID": building.id,
|
||||
"ADDR_LAT": sanitize_slug(building.sAddress),
|
||||
"ADDR_RUS": building.sAddress,
|
||||
"SER_ID": building.kSeria_Link.kRoot_id if building.kSeria_Link else None
|
||||
}
|
||||
seria_2_geo.append(data_of_point)
|
||||
|
||||
# Компактный формат для обфускации: [широта, долгота, ID адреса, ID серии]
|
||||
geo_compact.append([
|
||||
float(building.fGeoCode_Latitude),
|
||||
float(building.fGeoCode_Longitude),
|
||||
int(building.id),
|
||||
int(building.kSeria_Link.kRoot_id) if building.kSeria_Link else 0
|
||||
])
|
||||
|
||||
# Аккумулируем площади и статистику
|
||||
if building.fMunicipal_Area and building.fMunicipal_Area > 0:
|
||||
municipal_m2 += building.fMunicipal_Area
|
||||
if building.fResidential_Area and building.fResidential_Area > 0:
|
||||
residential_m2 += building.fResidential_Area
|
||||
if building.fGovernment_Area and building.fGovernment_Area > 0:
|
||||
government_m2 += building.fGovernment_Area
|
||||
if building.iNum_Residents and building.iNum_Residents > 0:
|
||||
residents += building.iNum_Residents
|
||||
if building.iNum_Apartments and building.iNum_Apartments > 0:
|
||||
apartments += building.iNum_Apartments
|
||||
if building.iNum_Accounts and building.iNum_Accounts > 0:
|
||||
accounts += building.iNum_Accounts
|
||||
if building.fCondition_House and building.fCondition_House > 0:
|
||||
if building.fCondition_House > condition_max:
|
||||
condition_max = building.fCondition_House
|
||||
if building.fCondition_House < condition_min:
|
||||
condition_min = building.fCondition_House
|
||||
|
||||
# Обфускуем координаты через Base64
|
||||
geo_json = json.dumps(geo_compact, separators=(',', ':'), ensure_ascii=True)
|
||||
geo_b64 = base64.b64encode(geo_json.encode('utf-8')).decode('utf-8')
|
||||
|
||||
data_return.update({
|
||||
"DATA4GEO": seria_2_geo,
|
||||
"DATA4GEO_B64": geo_b64,
|
||||
"MUNICIPAL_M2": municipal_m2,
|
||||
"RESIDENTIAL_M2": residential_m2,
|
||||
"GOVERNMENT_M2": government_m2,
|
||||
"RESIDENTS": residents,
|
||||
"APARTMENTS": apartments,
|
||||
"ACCOUNTS": accounts,
|
||||
"CONDITION_MAX": condition_max,
|
||||
"CONDITION_MIN": condition_min
|
||||
})
|
||||
|
||||
return data_return
|
||||
|
||||
|
||||
def seria_nav(root_series_ids):
|
||||
"""
|
||||
Возвращает информацию для построения навигации по всем корневым сериям.
|
||||
|
||||
Args:
|
||||
root_series_ids: список ID корневых серий
|
||||
|
||||
Returns:
|
||||
dict с информацией о всех корневых сериях для шаблона
|
||||
"""
|
||||
# Получаем информацию о всех корневых сериях для навигации
|
||||
all_root_series = Seria_Info.objects.filter(
|
||||
id__in=root_series_ids
|
||||
).order_by('id')
|
||||
|
||||
seria_nav_dim = []
|
||||
for seria in all_root_series:
|
||||
seria_nav_dim.append({
|
||||
"SERIA_R": seria.sName,
|
||||
"ID2URL": seria.id,
|
||||
"SERIA_L": sanitize_slug(seria.sName)
|
||||
})
|
||||
|
||||
return {"SERIA_NAV_DIM": seria_nav_dim}
|
||||
|
||||
|
||||
def minify_and_obfuscate_js(input_file_path, output_file_path, verbose=0):
|
||||
"""
|
||||
Минифицирует JavaScript файл используя rjsmin (чистый Python, без Node.js).
|
||||
|
||||
Координаты внутри шаблона уже обфускированы через Base64, поэтому основной
|
||||
минификатор просто сжимает синтаксис для экономии трафика.
|
||||
|
||||
Args:
|
||||
input_file_path: путь к исходному файлу
|
||||
output_file_path: путь к результирующему файлу
|
||||
verbose: уровень подробности вывода
|
||||
|
||||
Returns:
|
||||
tuple (успешность, размер_исходного, размер_минифицированного)
|
||||
"""
|
||||
if not RJSMIN_AVAILABLE:
|
||||
if verbose >= 1:
|
||||
print('[!!!] rjsmin не установлен. Минификация пропущена.')
|
||||
return False, os.path.getsize(input_file_path) / 1024, 0
|
||||
|
||||
try:
|
||||
# Читаем исходный файл
|
||||
with open(input_file_path, 'r', encoding='utf-8') as f:
|
||||
js_content = f.read()
|
||||
|
||||
# Минифицируем через rjsmin
|
||||
minified_content = _rjsmin.jsmin(js_content)
|
||||
|
||||
# Пишем результат
|
||||
with open(output_file_path, 'w', encoding='utf-8') as f:
|
||||
f.write(minified_content)
|
||||
|
||||
original_size = os.path.getsize(input_file_path) / 1024
|
||||
minified_size = os.path.getsize(output_file_path) / 1024
|
||||
|
||||
return True, original_size, minified_size
|
||||
|
||||
except Exception as e:
|
||||
if verbose >= 1:
|
||||
print(f'⚠ Ошибка при минификации: {e}')
|
||||
return False, os.path.getsize(input_file_path) / 1024, 0
|
||||
|
||||
|
||||
class Command(BaseCommand):
|
||||
help = 'Генерирует JavaScript-файлы для карт с геоданными зданий серий'
|
||||
|
||||
def add_arguments(self, parser):
|
||||
parser.add_argument(
|
||||
'--force',
|
||||
action='store_true',
|
||||
help='Перегенерировать файлы, даже если они существуют'
|
||||
)
|
||||
|
||||
def handle(self, *args, **options):
|
||||
verbose = int(options.get('verbosity', 1))
|
||||
force = options.get('force', False)
|
||||
|
||||
self.stdout.write(self.style.SUCCESS('=== ГЕНЕРАЦИЯ JAVASCRIPT ДЛЯ КАРТ ===\n'))
|
||||
|
||||
time_start = time.perf_counter()
|
||||
|
||||
# ========== ПОДГОТОВКА ==========
|
||||
path_name = f"{STATIC_BASE_PATH}/{PATH_FOR_JS_MAP}"
|
||||
|
||||
# Проверяем наличие папки
|
||||
if not os.path.exists(path_name):
|
||||
os.makedirs(path_name)
|
||||
if verbose >= 1:
|
||||
self.stdout.write(f'✓ Создана папка: {path_name}\n')
|
||||
|
||||
# ========== ПОЛУЧАЕМ ВСЕ КОРНЕВЫЕ СЕРИИ ==========
|
||||
if verbose >= 1:
|
||||
self.stdout.write('Этап 1: Сбор информации о корневых сериях...\n')
|
||||
|
||||
root_series = Seria_Info.objects.filter(
|
||||
id__in=Seria_Info.objects.all().values_list('kRoot_id', flat=True).distinct()
|
||||
).order_by('id')
|
||||
|
||||
root_series_ids = [seria.id for seria in root_series]
|
||||
|
||||
if verbose >= 1:
|
||||
self.stdout.write(f'✓ Найдено корневых серий: {len(root_series_ids)}\n')
|
||||
|
||||
# ========== ГЕНЕРИРУЕМ ЕДИНЫЙ JS ДЛЯ ВСЕХ СЕРИЙ ==========
|
||||
if verbose >= 1:
|
||||
self.stdout.write('\nЭтап 2: Генерация единого JS-файла для ВСЕ серий...\n')
|
||||
|
||||
time_start_js = time.perf_counter()
|
||||
|
||||
# Собираем ID в строку
|
||||
seria_ids_string = ','.join(str(id) for id in root_series_ids)
|
||||
|
||||
# Получаем геоданные для всех серий
|
||||
to_template = seria_info_geo_code(seria_ids_string)
|
||||
|
||||
# Получаем навигацию для всех корневых серий
|
||||
for_seria_nav = seria_nav(root_series_ids)
|
||||
to_template.update(for_seria_nav)
|
||||
|
||||
# Рендерим шаблон
|
||||
js_content = render_to_string("service/js_4all_seria_map_js.html", to_template)
|
||||
|
||||
# Пишем исходный файл
|
||||
js_file_path = f"{path_name}/_ALL{SUFFIX_FOR_JS_MAP}"
|
||||
js_mini_file_path = f"{path_name}/_ALL{SUFFIX_FOR_JS_MAP}".replace(".js", ".mini.js")
|
||||
|
||||
try:
|
||||
# Сохраняем исходный файл
|
||||
with open(js_file_path, 'w', encoding='utf-8') as js_file:
|
||||
js_file.write(js_content)
|
||||
|
||||
file_size_kb = os.path.getsize(js_file_path) / 1024
|
||||
time_elapsed = time.perf_counter() - time_start_js
|
||||
|
||||
if verbose >= 1:
|
||||
self.stdout.write(
|
||||
f'✓ Написан исходный файл: _ALL{SUFFIX_FOR_JS_MAP}\n'
|
||||
f' Размер: {file_size_kb:.1f} KB\n'
|
||||
)
|
||||
|
||||
# Минифицируем через rjsmin (чистый Python)
|
||||
if verbose >= 1:
|
||||
self.stdout.write('\nЭтап 3: Минификация JavaScript (rjsmin)...\n')
|
||||
|
||||
time_start_minify = time.perf_counter()
|
||||
success, orig_size, mini_size = minify_and_obfuscate_js(js_file_path, js_mini_file_path, verbose)
|
||||
time_minify_elapsed = time.perf_counter() - time_start_minify
|
||||
|
||||
if success and mini_size > 0:
|
||||
compression_ratio = (1 - mini_size / orig_size) * 100
|
||||
if verbose >= 1:
|
||||
self.stdout.write(
|
||||
f'[*] Минификация успешна!\n'
|
||||
f' Исходный файл: {orig_size:.3f} KB\n'
|
||||
f' Минифицированный: {mini_size:.3f} KB\n'
|
||||
f' Сжатие: {compression_ratio:.2f}%\n'
|
||||
f' Время: {time_minify_elapsed:.4f}с\n'
|
||||
)
|
||||
time_elapsed += time_minify_elapsed
|
||||
else:
|
||||
if verbose >= 1:
|
||||
self.stdout.write(f'[!!!] Минификация не применена. Используется исходный файл.\n')
|
||||
|
||||
if verbose >= 2:
|
||||
self.stdout.write(
|
||||
f'[i] Полная статистика по сериям:\n'
|
||||
f' - Жилых м²: {to_template.get("RESIDENTIAL_M2", 0):,.0f}\n'
|
||||
f' - Муниципальных м²: {to_template.get("MUNICIPAL_M2", 0):,.0f}\n'
|
||||
f' - Жильцов: {to_template.get("RESIDENTS", 0):,}\n'
|
||||
f' - Квартир: {to_template.get("APARTMENTS", 0):,}\n'
|
||||
)
|
||||
|
||||
except Exception as e:
|
||||
self.stdout.write(
|
||||
self.style.ERROR(f'✗ ОШИБКА при записи файла: {e}')
|
||||
)
|
||||
return
|
||||
|
||||
# ========== РЕЗУЛЬТАТЫ ==========
|
||||
time_total = time.perf_counter() - time_start
|
||||
|
||||
self.stdout.write(self.style.SUCCESS('\n=== РЕЗУЛЬТАТЫ ==='))
|
||||
self.stdout.write(f'✓ Серий обработано: {len(root_series_ids)}')
|
||||
self.stdout.write(f'✓ Зданий на карте: {len(to_template["DATA4GEO"])}')
|
||||
self.stdout.write(f'✓ JS-файлов создано: 2 (исходный + минифицированный)')
|
||||
self.stdout.write(f'✓ Исходный файл: _ALL{SUFFIX_FOR_JS_MAP}')
|
||||
self.stdout.write(f'✓ Минифицированный: _ALL{SUFFIX_FOR_JS_MAP.replace(".js", ".mini.js")}')
|
||||
self.stdout.write(f'✓ Обфускация: Base64 кодирование координат')
|
||||
|
||||
self.stdout.write(
|
||||
self.style.SUCCESS(f'\n[OK] Генерация завершена! Время: {time_total:.2f}с')
|
||||
)
|
||||
|
||||
624
oknardia/web/management/commands/generate_sitemaps.py
Normal file
624
oknardia/web/management/commands/generate_sitemaps.py
Normal file
@@ -0,0 +1,624 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
from __future__ import annotations
|
||||
|
||||
"""
|
||||
Команда генерации sitemap-файлов проекта.
|
||||
|
||||
Почему реализовано именно так:
|
||||
- Генерация выполняется оффлайн (через management command), чтобы не нагружать веб-запросы.
|
||||
- На выходе всегда создаются статические XML-файлы, которые потом отдает Nginx/прокси.
|
||||
- URL-источники описаны через Django Sitemap API (классы Sitemap), но рендер XML
|
||||
контролируем самостоятельно для точного управления лимитами размера/количества.
|
||||
"""
|
||||
|
||||
from collections import defaultdict
|
||||
from dataclasses import dataclass
|
||||
from datetime import date, datetime
|
||||
from itertools import combinations
|
||||
from pathlib import Path
|
||||
from typing import Iterable
|
||||
from xml.etree import ElementTree as ET
|
||||
|
||||
from django.conf import settings
|
||||
from django.contrib.sitemaps import Sitemap
|
||||
from django.core.management.base import BaseCommand, CommandError
|
||||
from django.db.models import Count, F, Max, Min
|
||||
from django.utils import timezone
|
||||
|
||||
from oknardia.models import (
|
||||
Apartment_Type,
|
||||
BlogPosts,
|
||||
Building_Info,
|
||||
MerchantBrand,
|
||||
PriceOffer,
|
||||
PVCprofiles,
|
||||
Seria_Info,
|
||||
SetKit,
|
||||
Win_MountDim,
|
||||
)
|
||||
from web.add_func import sanitize_slug
|
||||
|
||||
# Namespace схемы sitemap.xml по стандарту sitemaps.org.
|
||||
SITEMAP_XMLNS = "http://www.sitemaps.org/schemas/sitemap/0.9"
|
||||
|
||||
|
||||
@dataclass(slots=True)
|
||||
class SitemapBuildResult:
|
||||
"""Итог генерации sitemap для удобного вывода в CLI и web-обертках."""
|
||||
|
||||
# Общее число URL, записанных во все sitemap-файлы.
|
||||
total_urls: int
|
||||
# Количество созданных файлов (1 = только sitemap.xml, >1 = sitemapindex + sitemapNNNN.xml).
|
||||
files_count: int
|
||||
# Время выполнения генерации в секундах.
|
||||
elapsed_seconds: float
|
||||
# Физический каталог, куда записаны файлы.
|
||||
output_dir: Path
|
||||
|
||||
|
||||
def _as_sitemap_date(value: date | datetime | None) -> str:
|
||||
"""
|
||||
Приводит дату/время к формату `YYYY-MM-DD`.
|
||||
|
||||
Для sitemap нам не нужна точность до секунд: поисковикам достаточно даты.
|
||||
Если значение не передано, используем текущую локальную дату.
|
||||
"""
|
||||
if value is None:
|
||||
return timezone.localdate().isoformat()
|
||||
if isinstance(value, datetime):
|
||||
return value.date().isoformat()
|
||||
return value.isoformat()
|
||||
|
||||
|
||||
class SingleWindowSitemap(Sitemap):
|
||||
"""Источник URL для страниц цен одного проёма (/catalog/standard_opening/price-...)."""
|
||||
|
||||
changefreq = "weekly"
|
||||
priority = 0.5
|
||||
|
||||
def __init__(self, lastmod_value: datetime):
|
||||
# Один timestamp на весь прогон: так проще сравнивать выпуски sitemap.
|
||||
self.lastmod_value = lastmod_value
|
||||
|
||||
def items(self):
|
||||
# Берем только те монтажные размеры, где есть реальные офферы.
|
||||
# Сортировка по числу офферов повторяет историческую логику из raw SQL.
|
||||
mount_ids = (
|
||||
PriceOffer.objects.values("kOffer2MountDim_id")
|
||||
.annotate(num_offer=Count("id"))
|
||||
.order_by("num_offer", "kOffer2MountDim_id")
|
||||
.values_list("kOffer2MountDim_id", flat=True)
|
||||
)
|
||||
# Возвращаем сами объекты Win_MountDim, чтобы location() строил URL без доп. запросов.
|
||||
return Win_MountDim.objects.filter(id__in=mount_ids).only("id", "iWinWidth", "iWinHight")
|
||||
|
||||
def location(self, item: Win_MountDim) -> str:
|
||||
# В БД размеры в см (Decimal с 1 знаком). В URL исторически используются мм,
|
||||
# поэтому умножаем на 10 и приводим к int.
|
||||
width_mm = int(float(item.iWinWidth) * 10)
|
||||
height_mm = int(float(item.iWinHight) * 10)
|
||||
return f"/catalog/standard_opening/price-{width_mm}x{height_mm}mm-tip{item.id}"
|
||||
|
||||
def lastmod(self, item: Win_MountDim) -> datetime:
|
||||
return self.lastmod_value
|
||||
|
||||
|
||||
class BuildingOffersSitemap(Sitemap):
|
||||
"""Источник URL для страниц ценовой выдачи по адресам (/{build_id}/{apart_id}/{slug})."""
|
||||
|
||||
changefreq = "weekly"
|
||||
priority = 0.5
|
||||
|
||||
def __init__(self, lastmod_value: datetime):
|
||||
self.lastmod_value = lastmod_value
|
||||
|
||||
def items(self):
|
||||
# Получаем здания только с валидной привязкой к корневой серии.
|
||||
buildings = list(
|
||||
Building_Info.objects.filter(kSeria_Link__kRoot__isnull=False)
|
||||
.select_related("kSeria_Link__kRoot")
|
||||
.only("id", "sAddress", "kSeria_Link__kRoot")
|
||||
.order_by("id")
|
||||
)
|
||||
|
||||
# Для каждой корневой серии нужен список типов квартир, чтобы собрать итоговые URL.
|
||||
root_ids = {
|
||||
building.kSeria_Link.kRoot_id
|
||||
for building in buildings
|
||||
if building.kSeria_Link_id and building.kSeria_Link.kRoot_id
|
||||
}
|
||||
apartments_by_root: dict[int, list[int]] = defaultdict(list)
|
||||
for root_id, apart_id in Apartment_Type.objects.filter(kSeria_id__in=root_ids).values_list("kSeria_id", "id"):
|
||||
apartments_by_root[root_id].append(apart_id)
|
||||
|
||||
# Генерируем декартово произведение: здание x квартиры его корневой серии.
|
||||
for building in buildings:
|
||||
root_id = building.kSeria_Link.kRoot_id if building.kSeria_Link_id else None
|
||||
if not root_id:
|
||||
continue
|
||||
for apart_id in apartments_by_root.get(root_id, []):
|
||||
yield (building.id, apart_id, sanitize_slug(building.sAddress))
|
||||
|
||||
def location(self, item: tuple[int, int, str]) -> str:
|
||||
build_id, apart_id, address_slug = item
|
||||
# Получаем объект здания и серию для формирования нового роутинга
|
||||
try:
|
||||
building = Building_Info.objects.select_related('kSeria_Link__kRoot').get(id=build_id)
|
||||
seria = building.kSeria_Link.kRoot
|
||||
seria_id = seria.id
|
||||
seria_slug = sanitize_slug((seria.sName or ""))
|
||||
except Exception:
|
||||
# fallback на старый роутинг, если что-то пошло не так
|
||||
return f"/{build_id}/{apart_id}/{address_slug}"
|
||||
# Новый формат: /price/seriaID<seria_id>--<seria_slug>/appartID<apart_id>/addressID<address_id>--<address_slug>/
|
||||
return f"/price/seriaID{seria_id}--{seria_slug}/appartID{apart_id}/addressID{build_id}--{address_slug}/"
|
||||
|
||||
def lastmod(self, item: tuple[int, int, str]) -> datetime:
|
||||
return self.lastmod_value
|
||||
|
||||
|
||||
class CompareOffersSitemap(Sitemap):
|
||||
"""Источник URL для страниц сравнения наборов (/compare_offers/1,2,3...)."""
|
||||
|
||||
# Для compare-страниц изменения редки, поэтому просим роботов не дергать их часто.
|
||||
changefreq = "monthly"
|
||||
priority = 0.35
|
||||
|
||||
def __init__(self, lastmod_value: datetime, min_depth: int = 2, max_depth: int = 4):
|
||||
self.lastmod_value = lastmod_value
|
||||
# Жестко ограничиваем глубину до 2..4, чтобы не получить комбинаторный взрыв.
|
||||
self.min_depth = max(2, min_depth)
|
||||
self.max_depth = min(4, max_depth)
|
||||
|
||||
def items(self):
|
||||
# Берем только активные наборы и строим combinations без повторов/перестановок.
|
||||
set_ids = list(SetKit.objects.filter(sSetActive=True).order_by("id").values_list("id", flat=True))
|
||||
for depth in range(self.min_depth, self.max_depth + 1):
|
||||
for combo in combinations(set_ids, depth):
|
||||
# Формат URL-параметра должен остаться историческим: "1,2,3".
|
||||
yield ",".join(str(item) for item in combo)
|
||||
|
||||
def location(self, item: str) -> str:
|
||||
return f"/compare_offers/{item}"
|
||||
|
||||
def lastmod(self, item: str) -> datetime:
|
||||
return self.lastmod_value
|
||||
|
||||
|
||||
class StaticPagesSitemap(Sitemap):
|
||||
"""Набор важных статических/обзорных страниц, которые не требуют отдельной модели."""
|
||||
|
||||
def __init__(self, items: list[dict]):
|
||||
self._items = items
|
||||
|
||||
def items(self):
|
||||
return self._items
|
||||
|
||||
def location(self, item: dict) -> str:
|
||||
return item["loc"]
|
||||
|
||||
def lastmod(self, item: dict) -> date | datetime | None:
|
||||
return item.get("lastmod")
|
||||
|
||||
def changefreq(self, item: dict) -> str:
|
||||
return item.get("changefreq", "weekly")
|
||||
|
||||
def priority(self, item: dict) -> float:
|
||||
return float(item.get("priority", 0.5))
|
||||
|
||||
|
||||
class BlogListSitemap(Sitemap):
|
||||
"""Страницы пагинации блога: /blog/P0, /blog/P1, ..."""
|
||||
|
||||
changefreq = "weekly"
|
||||
priority = 0.82
|
||||
|
||||
def __init__(self, lastmod_value: date | datetime | None):
|
||||
self.lastmod_value = lastmod_value
|
||||
|
||||
def items(self):
|
||||
posts_qs = BlogPosts.objects.filter(
|
||||
dPostDataBegin__lte=timezone.now(),
|
||||
bPublished=True,
|
||||
bArchive=False,
|
||||
)
|
||||
total_posts = posts_qs.count()
|
||||
if total_posts == 0:
|
||||
return []
|
||||
pages_total = (total_posts - 1) // settings.NUM_BLOG_TIZER_IN_PAGE + 1
|
||||
return list(range(pages_total))
|
||||
|
||||
def location(self, item: int) -> str:
|
||||
return f"/blog/P{item}"
|
||||
|
||||
def lastmod(self, item: int) -> date | datetime | None:
|
||||
return self.lastmod_value
|
||||
|
||||
|
||||
class BlogPostSitemap(Sitemap):
|
||||
"""Публичные посты блога в каноническом URL без page_back."""
|
||||
|
||||
changefreq = "monthly"
|
||||
priority = 0.90
|
||||
|
||||
def items(self):
|
||||
return BlogPosts.objects.filter(
|
||||
dPostDataBegin__lte=timezone.now(),
|
||||
bPublished=True,
|
||||
bArchive=False,
|
||||
).only("id", "sPostHeader", "dPostDataModify")
|
||||
|
||||
def location(self, item: BlogPosts) -> str:
|
||||
return f"/blogpost/{item.id}/{sanitize_slug(item.sPostHeader)}"
|
||||
|
||||
def lastmod(self, item: BlogPosts) -> date | datetime | None:
|
||||
return item.dPostDataModify
|
||||
|
||||
|
||||
class ProfileManufactureSitemap(Sitemap):
|
||||
"""Страницы производителей профилей: /catalog/profile/{id}-{manufacturer}."""
|
||||
|
||||
changefreq = "monthly"
|
||||
priority = 0.92
|
||||
|
||||
def items(self):
|
||||
return list(
|
||||
PVCprofiles.objects.values("sProfileManufacturer")
|
||||
.annotate(first_id=Min("id"), lastmod=Max("dProfileModify"))
|
||||
.order_by("sProfileManufacturer")
|
||||
)
|
||||
|
||||
def location(self, item: dict) -> str:
|
||||
manufacturer_slug = sanitize_slug(item["sProfileManufacturer"])
|
||||
return f"/catalog/profile/{item['first_id']}-{manufacturer_slug}"
|
||||
|
||||
def lastmod(self, item: dict) -> date | datetime | None:
|
||||
return item.get("lastmod")
|
||||
|
||||
|
||||
class ProfileModelSitemap(Sitemap):
|
||||
"""Карточки конкретных профильных систем."""
|
||||
|
||||
changefreq = "monthly"
|
||||
priority = 0.94
|
||||
|
||||
def items(self):
|
||||
return PVCprofiles.objects.only("id", "sProfileManufacturer", "sProfileName", "dProfileModify")
|
||||
|
||||
def location(self, item: PVCprofiles) -> str:
|
||||
manufacturer_slug = sanitize_slug(item.sProfileManufacturer)
|
||||
model_slug = sanitize_slug(item.sProfileName)
|
||||
# Исторически канонический URL использует id модели и в сегменте manufacturer_id, и в segment model_id.
|
||||
return f"/catalog/profile/{item.id}-{manufacturer_slug}/{item.id}-{model_slug}"
|
||||
|
||||
def lastmod(self, item: PVCprofiles) -> date | datetime | None:
|
||||
return item.dProfileModify
|
||||
|
||||
|
||||
class SeriaDetailSitemap(Sitemap):
|
||||
"""Страницы типовых серий домов — это одни из самых важных SEO-страниц проекта."""
|
||||
|
||||
changefreq = "monthly"
|
||||
priority = 0.97
|
||||
|
||||
def items(self):
|
||||
return Seria_Info.objects.filter(id__isnull=False, kRoot_id__isnull=False, id=F("kRoot_id")).only(
|
||||
"id", "sName", "dSeriaInfoModify"
|
||||
)
|
||||
|
||||
def location(self, item: Seria_Info) -> str:
|
||||
return f"/catalog/seria/{sanitize_slug(item.sName)}/all{item.id}"
|
||||
|
||||
def lastmod(self, item: Seria_Info) -> date | datetime | None:
|
||||
return item.dSeriaInfoModify
|
||||
|
||||
|
||||
class CompanyDetailSitemap(Sitemap):
|
||||
"""Страницы брендов/производителей оконных компаний."""
|
||||
|
||||
changefreq = "monthly"
|
||||
priority = 0.91
|
||||
|
||||
def items(self):
|
||||
return list(
|
||||
MerchantBrand.objects.annotate(
|
||||
last_offer_modify=Max("merchantoffice__ouruser__setkit__priceoffer__dOfferModify"),
|
||||
last_office_modify=Max("merchantoffice__dOfficeDataModify"),
|
||||
).only("id", "sMerchantName")
|
||||
)
|
||||
|
||||
def location(self, item: MerchantBrand) -> str:
|
||||
return f"/catalog/company/{item.id}-{sanitize_slug(item.sMerchantName)}"
|
||||
|
||||
def lastmod(self, item: MerchantBrand) -> date | datetime | None:
|
||||
return getattr(item, "last_offer_modify", None) or getattr(item, "last_office_modify", None)
|
||||
|
||||
|
||||
class SitemapXmlWriter:
|
||||
"""
|
||||
Низкоуровневый писатель XML.
|
||||
|
||||
Делит URL на несколько файлов по двум условиям:
|
||||
- число URL в файле;
|
||||
- приблизительный размер файла в байтах.
|
||||
|
||||
Если chunk-файлов больше одного, создается sitemapindex (sitemap.xml),
|
||||
который перечисляет sitemap0000.xml, sitemap0001.xml и т.д.
|
||||
"""
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
output_dir: Path,
|
||||
public_base_url: str,
|
||||
max_items: int,
|
||||
max_file_size: int,
|
||||
max_files_qty: int,
|
||||
):
|
||||
self.output_dir = output_dir
|
||||
# Публичный URL-префикс для ссылок в sitemapindex.
|
||||
self.public_base_url = public_base_url.rstrip("/")
|
||||
self.max_items = max_items
|
||||
self.max_file_size = max_file_size
|
||||
self.max_files_qty = max_files_qty
|
||||
|
||||
self.total_urls = 0
|
||||
self.chunk_files: list[str] = []
|
||||
self.current_urls: list[ET.Element] = []
|
||||
# Небольшой стартовый запас размера на корневые XML-теги.
|
||||
self.current_size = 128
|
||||
|
||||
def cleanup_old(self) -> None:
|
||||
# Перед генерацией удаляем старые sitemap*.xml, чтобы не оставить устаревшие куски.
|
||||
self.output_dir.mkdir(parents=True, exist_ok=True)
|
||||
for file_path in self.output_dir.glob("sitemap*.xml"):
|
||||
file_path.unlink(missing_ok=True)
|
||||
|
||||
def add_url(self, loc: str, lastmod: datetime, changefreq: str, priority: float) -> None:
|
||||
# Собираем XML-элемент URL и оцениваем его вклад в размер файла.
|
||||
url_element = self._build_url_element(loc=loc, lastmod=lastmod, changefreq=changefreq, priority=priority)
|
||||
url_size = len(ET.tostring(url_element, encoding="utf-8"))
|
||||
|
||||
need_flush = False
|
||||
if self.current_urls:
|
||||
# Лимиты применяем только если файл уже что-то содержит:
|
||||
# так мы гарантируем, что хотя бы один URL всегда будет записан.
|
||||
if len(self.current_urls) >= self.max_items:
|
||||
need_flush = True
|
||||
elif self.current_size + url_size > self.max_file_size:
|
||||
need_flush = True
|
||||
|
||||
if need_flush:
|
||||
self._flush_chunk()
|
||||
|
||||
self.current_urls.append(url_element)
|
||||
self.current_size += url_size
|
||||
self.total_urls += 1
|
||||
|
||||
def finalize(self, generated_at: datetime) -> int:
|
||||
# Если уже были chunk-файлы, значит итог должен быть в формате sitemapindex.
|
||||
if self.chunk_files:
|
||||
self._flush_chunk()
|
||||
self._write_sitemap_index(generated_at)
|
||||
return len(self.chunk_files)
|
||||
|
||||
# Иначе пишем единый sitemap.xml с URLSet.
|
||||
self._write_single_sitemap()
|
||||
return 1
|
||||
|
||||
def _flush_chunk(self) -> None:
|
||||
if not self.current_urls:
|
||||
return
|
||||
|
||||
chunk_idx = len(self.chunk_files)
|
||||
if chunk_idx >= self.max_files_qty:
|
||||
raise RuntimeError(
|
||||
"Превышено максимальное количество sitemap-файлов. "
|
||||
f"Текущий лимит: {self.max_files_qty}."
|
||||
)
|
||||
|
||||
file_name = f"sitemap{chunk_idx:04d}.xml"
|
||||
self._write_urlset(self.output_dir / file_name, self.current_urls)
|
||||
self.chunk_files.append(file_name)
|
||||
|
||||
# Сбрасываем буфер для следующего chunk-файла.
|
||||
self.current_urls = []
|
||||
self.current_size = 128
|
||||
|
||||
def _write_single_sitemap(self) -> None:
|
||||
self._write_urlset(self.output_dir / "sitemap.xml", self.current_urls)
|
||||
self.current_urls = []
|
||||
self.current_size = 128
|
||||
|
||||
def _write_sitemap_index(self, generated_at: datetime) -> None:
|
||||
root = ET.Element("sitemapindex", xmlns=SITEMAP_XMLNS)
|
||||
for file_name in self.chunk_files:
|
||||
sitemap_element = ET.SubElement(root, "sitemap")
|
||||
ET.SubElement(sitemap_element, "loc").text = f"{self.public_base_url}/{file_name}"
|
||||
ET.SubElement(sitemap_element, "lastmod").text = _as_sitemap_date(generated_at)
|
||||
|
||||
xml_bytes = ET.tostring(root, encoding="utf-8", xml_declaration=True)
|
||||
(self.output_dir / "sitemap.xml").write_bytes(xml_bytes)
|
||||
|
||||
@staticmethod
|
||||
def _write_urlset(file_path: Path, urls: Iterable[ET.Element]) -> None:
|
||||
root = ET.Element("urlset", xmlns=SITEMAP_XMLNS)
|
||||
for url in urls:
|
||||
root.append(url)
|
||||
xml_bytes = ET.tostring(root, encoding="utf-8", xml_declaration=True)
|
||||
file_path.write_bytes(xml_bytes)
|
||||
|
||||
@staticmethod
|
||||
def _build_url_element(loc: str, lastmod: datetime, changefreq: str, priority: float) -> ET.Element:
|
||||
element = ET.Element("url")
|
||||
ET.SubElement(element, "loc").text = loc
|
||||
ET.SubElement(element, "lastmod").text = _as_sitemap_date(lastmod)
|
||||
ET.SubElement(element, "changefreq").text = changefreq
|
||||
ET.SubElement(element, "priority").text = f"{priority:.2f}"
|
||||
return element
|
||||
|
||||
|
||||
def build_sitemaps(
|
||||
output_dir: Path,
|
||||
site_base_url: str,
|
||||
sitemap_url_prefix: str,
|
||||
max_items: int,
|
||||
max_file_size: int,
|
||||
max_files_qty: int,
|
||||
compare_min_depth: int = 2,
|
||||
compare_max_depth: int = 4,
|
||||
) -> SitemapBuildResult:
|
||||
"""Оркестратор полного прогона сборки sitemap-файлов."""
|
||||
time_start = timezone.now()
|
||||
generated_at = timezone.now()
|
||||
compare_lastmod = generated_at.date().replace(day=1)
|
||||
latest_blog_modify = BlogPosts.objects.filter(
|
||||
dPostDataBegin__lte=timezone.now(),
|
||||
bPublished=True,
|
||||
bArchive=False,
|
||||
).aggregate(lastmod=Max("dPostDataModify"))["lastmod"]
|
||||
latest_profile_modify = PVCprofiles.objects.aggregate(lastmod=Max("dProfileModify"))["lastmod"]
|
||||
latest_seria_modify = Seria_Info.objects.aggregate(lastmod=Max("dSeriaInfoModify"))["lastmod"]
|
||||
latest_company_modify = MerchantBrand.objects.annotate(
|
||||
last_offer_modify=Max("merchantoffice__ouruser__setkit__priceoffer__dOfferModify"),
|
||||
last_office_modify=Max("merchantoffice__dOfficeDataModify"),
|
||||
).aggregate(lastmod=Max("last_offer_modify"), lastmod_office=Max("last_office_modify"))
|
||||
latest_company_date = latest_company_modify.get("lastmod") or latest_company_modify.get("lastmod_office")
|
||||
|
||||
base_url = site_base_url.rstrip("/")
|
||||
url_prefix = sitemap_url_prefix.strip("/")
|
||||
public_sitemap_base = f"{base_url}/{url_prefix}" if url_prefix else base_url
|
||||
|
||||
writer = SitemapXmlWriter(
|
||||
output_dir=output_dir,
|
||||
public_base_url=public_sitemap_base,
|
||||
max_items=max_items,
|
||||
max_file_size=max_file_size,
|
||||
max_files_qty=max_files_qty,
|
||||
)
|
||||
writer.cleanup_old()
|
||||
|
||||
# Источники URL. Порядок можно менять, если нужно управлять наполнением chunk-файлов.
|
||||
sitemaps = [
|
||||
StaticPagesSitemap(
|
||||
items=[
|
||||
{"loc": "/", "lastmod": generated_at, "changefreq": "weekly", "priority": 1.00},
|
||||
{"loc": "/catalog", "lastmod": generated_at, "changefreq": "weekly", "priority": 0.88},
|
||||
{"loc": "/catalog/profile", "lastmod": latest_profile_modify, "changefreq": "weekly", "priority": 0.92},
|
||||
{"loc": "/catalog/seria", "lastmod": latest_seria_modify, "changefreq": "weekly", "priority": 0.95},
|
||||
{"loc": "/catalog/standard_opening", "lastmod": latest_seria_modify, "changefreq": "monthly", "priority": 0.86},
|
||||
{"loc": "/catalog/company", "lastmod": latest_company_date, "changefreq": "weekly", "priority": 0.90},
|
||||
{"loc": "/stat_all/", "lastmod": generated_at, "changefreq": "weekly", "priority": 0.81},
|
||||
{"loc": "/stat/rating/profiles_rank", "lastmod": latest_profile_modify, "changefreq": "monthly", "priority": 0.76},
|
||||
{"loc": "/tariff/", "lastmod": generated_at, "changefreq": "monthly", "priority": 0.85},
|
||||
{"loc": "/contact/", "lastmod": generated_at, "changefreq": "yearly", "priority": 0.60},
|
||||
]
|
||||
),
|
||||
BlogListSitemap(lastmod_value=latest_blog_modify),
|
||||
BlogPostSitemap(),
|
||||
ProfileManufactureSitemap(),
|
||||
ProfileModelSitemap(),
|
||||
SeriaDetailSitemap(),
|
||||
CompanyDetailSitemap(),
|
||||
SingleWindowSitemap(lastmod_value=generated_at),
|
||||
BuildingOffersSitemap(lastmod_value=generated_at),
|
||||
CompareOffersSitemap(
|
||||
lastmod_value=compare_lastmod,
|
||||
min_depth=compare_min_depth,
|
||||
max_depth=compare_max_depth,
|
||||
),
|
||||
]
|
||||
|
||||
for sitemap in sitemaps:
|
||||
for item in sitemap.items():
|
||||
location = sitemap.location(item)
|
||||
lastmod = sitemap.lastmod(item)
|
||||
if not location.startswith("/"):
|
||||
location = f"/{location}"
|
||||
sitemap_changefreq = sitemap.changefreq(item) if callable(getattr(sitemap, "changefreq", None)) else str(sitemap.changefreq)
|
||||
sitemap_priority = sitemap.priority(item) if callable(getattr(sitemap, "priority", None)) else float(sitemap.priority)
|
||||
writer.add_url(
|
||||
loc=f"{base_url}{location}",
|
||||
lastmod=lastmod,
|
||||
changefreq=sitemap_changefreq,
|
||||
priority=sitemap_priority,
|
||||
)
|
||||
|
||||
files_count = writer.finalize(generated_at=generated_at)
|
||||
elapsed = (timezone.now() - time_start).total_seconds()
|
||||
return SitemapBuildResult(
|
||||
total_urls=writer.total_urls,
|
||||
files_count=files_count,
|
||||
elapsed_seconds=elapsed,
|
||||
output_dir=output_dir,
|
||||
)
|
||||
|
||||
|
||||
class Command(BaseCommand):
|
||||
help = "Генерирует sitemap.xml и sitemapNNNN.xml в файловый кэш."
|
||||
|
||||
def add_arguments(self, parser):
|
||||
parser.add_argument(
|
||||
"--compare-min-depth",
|
||||
type=int,
|
||||
default=2,
|
||||
help="Минимальная глубина комбинаций compare_offers (по умолчанию 2).",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--compare-max-depth",
|
||||
type=int,
|
||||
default=4,
|
||||
help="Максимальная глубина комбинаций compare_offers (по умолчанию 4).",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--max-items",
|
||||
type=int,
|
||||
default=40000,
|
||||
help="Максимум URL в одном sitemap-файле (по умолчанию 40000).",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--max-file-size",
|
||||
type=int,
|
||||
default=5242880,
|
||||
help="Максимальный размер sitemap-файла в байтах (по умолчанию 5242880).",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--max-files-qty",
|
||||
type=int,
|
||||
default=998,
|
||||
help="Максимум вложенных sitemap-файлов (по умолчанию 998).",
|
||||
)
|
||||
|
||||
def handle(self, *args, **options):
|
||||
# Валидация глубины compare перед запуском тяжелой части генерации.
|
||||
compare_min_depth = options["compare_min_depth"]
|
||||
compare_max_depth = options["compare_max_depth"]
|
||||
if compare_min_depth > compare_max_depth:
|
||||
raise CommandError("--compare-min-depth не может быть больше --compare-max-depth")
|
||||
|
||||
result = build_sitemaps(
|
||||
output_dir=Path(settings.SITEMAP_ROOT),
|
||||
site_base_url=settings.SITE_BASE_URL,
|
||||
sitemap_url_prefix=settings.SITEMAP_URL_PREFIX,
|
||||
max_items=options["max_items"],
|
||||
max_file_size=options["max_file_size"],
|
||||
max_files_qty=options["max_files_qty"],
|
||||
compare_min_depth=compare_min_depth,
|
||||
compare_max_depth=compare_max_depth,
|
||||
)
|
||||
|
||||
# Человекочитаемый отчет для логов CI/CD и контейнерных entrypoint-скриптов.
|
||||
if result.files_count == 1:
|
||||
self.stdout.write(
|
||||
self.style.SUCCESS(
|
||||
f"Создан единственный sitemap.xml. URL-ов: {result.total_urls}. "
|
||||
f"Время: {result.elapsed_seconds:.2f} сек."
|
||||
)
|
||||
)
|
||||
else:
|
||||
self.stdout.write(
|
||||
self.style.SUCCESS(
|
||||
f"Создан каскад sitemap. Файлов: {result.files_count}. URL-ов: {result.total_urls}. "
|
||||
f"Время: {result.elapsed_seconds:.2f} сек."
|
||||
)
|
||||
)
|
||||
|
||||
1123
oknardia/web/management/commands/make_rating.py
Normal file
1123
oknardia/web/management/commands/make_rating.py
Normal file
File diff suppressed because it is too large
Load Diff
189
oknardia/web/management/commands/populate_seo_fields.py
Normal file
189
oknardia/web/management/commands/populate_seo_fields.py
Normal file
@@ -0,0 +1,189 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
from __future__ import annotations
|
||||
|
||||
"""
|
||||
Management-команда для автозаполнения SEO-полей (sSlug, sMetaDescription, sMetaKeywords)
|
||||
у всех существующих записей блога.
|
||||
|
||||
Эта команда используется один раз при миграции на новую версию,
|
||||
которая добавила автогенерацию SEO-полей в save() метод BlogPosts.
|
||||
|
||||
Использование:
|
||||
python manage.py populate_seo_fields
|
||||
python manage.py populate_seo_fields --dry-run # только показать что будет сделано
|
||||
python manage.py populate_seo_fields --clean # очистить все SEO-поля перед заполнением
|
||||
"""
|
||||
|
||||
import re
|
||||
from datetime import datetime
|
||||
|
||||
from django.core.management.base import BaseCommand
|
||||
from django.utils import timezone
|
||||
|
||||
from oknardia.models import BlogPosts
|
||||
from web.add_func import sanitize_slug, safe_html_spec_symbols
|
||||
|
||||
|
||||
class Command(BaseCommand):
|
||||
help = "Автозаполняет SEO-поля (sSlug, sMetaDescription, sMetaKeywords) для всех записей блога"
|
||||
|
||||
def add_arguments(self, parser):
|
||||
parser.add_argument(
|
||||
"--dry-run",
|
||||
action="store_true",
|
||||
help="Только показать, что будет сделано, без сохранения в БД",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--clean",
|
||||
action="store_true",
|
||||
help="Очистить все SEO-поля перед заполнением (для переделки)",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--force",
|
||||
action="store_true",
|
||||
help="Перезаполнить SEO-поля (даже если они уже содержат значения)",
|
||||
)
|
||||
|
||||
def handle(self, *args, **options):
|
||||
dry_run = options.get("dry_run", False)
|
||||
clean = options.get("clean", False)
|
||||
force = options.get("force", False)
|
||||
|
||||
self.stdout.write(self.style.HTTP_INFO("=" * 70))
|
||||
self.stdout.write(self.style.HTTP_INFO("АВТОЗАПОЛНЕНИЕ SEO-ПОЛЕЙ БЛОГА"))
|
||||
self.stdout.write(self.style.HTTP_INFO("=" * 70))
|
||||
|
||||
# Получаем все посты
|
||||
posts_qs = BlogPosts.objects.all()
|
||||
total_posts = posts_qs.count()
|
||||
self.stdout.write(f"\n✓ Всего записей в блоге: {total_posts}")
|
||||
|
||||
if total_posts == 0:
|
||||
self.stdout.write(self.style.WARNING("⚠ Записей не найдено. Нечего заполнять."))
|
||||
return
|
||||
|
||||
# Опционально очищаем
|
||||
if clean and not dry_run:
|
||||
self.stdout.write("\n🧹 Очищаем существующие SEO-поля...")
|
||||
posts_qs.update(sSlug="", sMetaDescription="", sMetaKeywords="")
|
||||
self.stdout.write(self.style.SUCCESS(" ✓ SEO-поля очищены"))
|
||||
|
||||
# Фильтруем посты по пустым полям
|
||||
if force:
|
||||
filtered_posts = posts_qs
|
||||
self.stdout.write(f"\n✓ Режим FORCE: будут переполнены ВСЕ {total_posts} записей")
|
||||
else:
|
||||
filtered_posts = posts_qs.filter(
|
||||
sSlug="", # noqa: F841
|
||||
) | posts_qs.filter(sMetaDescription="") | posts_qs.filter(sMetaKeywords="")
|
||||
filtered_posts = posts_qs.filter(
|
||||
sSlug="",
|
||||
) | posts_qs.filter(sMetaDescription="") | posts_qs.filter(sMetaKeywords="")
|
||||
|
||||
posts_to_update = filtered_posts.count()
|
||||
self.stdout.write(f"✓ Записей для обновления: {posts_to_update}")
|
||||
|
||||
if posts_to_update == 0:
|
||||
self.stdout.write(self.style.SUCCESS("\n✅ Все записи уже имеют заполненные SEO-поля!"))
|
||||
return
|
||||
|
||||
# Статистика по типам полей
|
||||
stats = {
|
||||
"sSlug": 0,
|
||||
"sMetaDescription": 0,
|
||||
"sMetaKeywords": 0,
|
||||
"updated": 0,
|
||||
"errors": 0,
|
||||
}
|
||||
|
||||
# Обновляем каждый пост
|
||||
self.stdout.write("\n🔄 Обробатываем посты...\n")
|
||||
|
||||
for idx, post in enumerate(filtered_posts, 1):
|
||||
try:
|
||||
old_values = {
|
||||
"sSlug": post.sSlug,
|
||||
"sMetaDescription": post.sMetaDescription,
|
||||
"sMetaKeywords": post.sMetaKeywords,
|
||||
}
|
||||
|
||||
# Генерируем sSlug
|
||||
if not post.sSlug and post.sPostHeader:
|
||||
post.sSlug = sanitize_slug(post.sPostHeader, max_length=200)
|
||||
stats["sSlug"] += 1
|
||||
|
||||
# Генерируем sMetaDescription
|
||||
if not post.sMetaDescription and post.sPostContent:
|
||||
content_clean = re.sub(r"<cut[\s\S]*?>", "", post.sPostContent, flags=re.IGNORECASE)
|
||||
tizer = safe_html_spec_symbols(content_clean)
|
||||
|
||||
if len(tizer) > 160:
|
||||
# Обрезаем по последнему пробелу перед 160-й позицией
|
||||
tizer = tizer[:160].rsplit(" ", 1)[0] + "..." if " " in tizer[:160] else tizer[:160]
|
||||
|
||||
post.sMetaDescription = tizer
|
||||
stats["sMetaDescription"] += 1
|
||||
|
||||
# Генерируем sMetaKeywords
|
||||
if not post.sMetaKeywords and post.sPostHeader:
|
||||
header_clean = safe_html_spec_symbols(post.sPostHeader).strip()
|
||||
fixed_keywords = "oknardia, окнардия, блог, публикация"
|
||||
post.sMetaKeywords = f"{fixed_keywords}, {header_clean}"[:256]
|
||||
stats["sMetaKeywords"] += 1
|
||||
|
||||
new_values = {
|
||||
"sSlug": post.sSlug,
|
||||
"sMetaDescription": post.sMetaDescription,
|
||||
"sMetaKeywords": post.sMetaKeywords,
|
||||
}
|
||||
|
||||
# Логируем изменения
|
||||
changes = []
|
||||
if old_values["sSlug"] != new_values["sSlug"]:
|
||||
changes.append(f"sSlug: '{old_values['sSlug'][:30]}...' → '{new_values['sSlug'][:30]}...'")
|
||||
if old_values["sMetaDescription"] != new_values["sMetaDescription"]:
|
||||
desc_old = (old_values["sMetaDescription"] or "").strip() or "(пусто)"
|
||||
desc_new = new_values.get("sMetaDescription", "").strip() or "(пусто)"
|
||||
changes.append(f"sMetaDescription: '{desc_old[:40]}...' → '{desc_new[:40]}...'")
|
||||
if old_values["sMetaKeywords"] != new_values["sMetaKeywords"]:
|
||||
kw_old = (old_values["sMetaKeywords"] or "").strip() or "(пусто)"
|
||||
kw_new = new_values.get("sMetaKeywords", "").strip() or "(пусто)"
|
||||
changes.append(f"sMetaKeywords: '{kw_old[:40]}...' → '{kw_new[:40]}...'")
|
||||
|
||||
# Вывод текущего прогресса
|
||||
self.stdout.write(
|
||||
f" [{idx:3d}/{posts_to_update}] Post #{post.id}: {post.sPostHeader[:50]}..."
|
||||
)
|
||||
if changes:
|
||||
for change in changes:
|
||||
self.stdout.write(f" → {change}")
|
||||
self.stdout.write("")
|
||||
|
||||
# Сохраняем
|
||||
if not dry_run:
|
||||
post.save(update_fields=["sSlug", "sMetaDescription", "sMetaKeywords"])
|
||||
stats["updated"] += 1
|
||||
|
||||
except Exception as e:
|
||||
self.stdout.write(self.style.ERROR(f" ❌ Ошибка при обработке поста #{post.id}: {str(e)}"))
|
||||
stats["errors"] += 1
|
||||
|
||||
# Итоговой отчет
|
||||
self.stdout.write("\n" + "=" * 70)
|
||||
self.stdout.write(self.style.SUCCESS("ИТОГОВЫЙ ОТЧЕТ"))
|
||||
self.stdout.write("=" * 70)
|
||||
|
||||
self.stdout.write(f"\n✓ sSlug заполнено: {stats['sSlug']} раз")
|
||||
self.stdout.write(f"✓ sMetaDescription заполнено: {stats['sMetaDescription']} раз")
|
||||
self.stdout.write(f"✓ sMetaKeywords заполнено: {stats['sMetaKeywords']} раз")
|
||||
self.stdout.write(f"✓ Записей обновлено в БД: {stats['updated']}")
|
||||
self.stdout.write(f"✗ Ошибок при обработке: {stats['errors']}")
|
||||
|
||||
if dry_run:
|
||||
self.stdout.write(self.style.WARNING("\n⚠️ Режим DRY-RUN: изменения НЕ были сохранены в БД"))
|
||||
else:
|
||||
self.stdout.write(self.style.SUCCESS(f"\n✅ Обновлено {stats['updated']} записей успешно!"))
|
||||
|
||||
if stats["errors"] > 0:
|
||||
self.stdout.write(self.style.ERROR(f"\n❌ Было {stats['errors']} ошибок. Проверьте логи."))
|
||||
|
||||
140
oknardia/web/management/commands/regenerate_seria_prerender.py
Normal file
140
oknardia/web/management/commands/regenerate_seria_prerender.py
Normal file
@@ -0,0 +1,140 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
from __future__ import annotations
|
||||
|
||||
from pathlib import Path
|
||||
|
||||
from django.conf import settings
|
||||
from django.core.management.base import BaseCommand, CommandError
|
||||
from django.db.models import F
|
||||
from django.test import RequestFactory
|
||||
|
||||
from oknardia.models import Seria_Info
|
||||
from web import catalog_series
|
||||
from web.add_func import sanitize_slug
|
||||
|
||||
|
||||
class Command(BaseCommand):
|
||||
"""Пересоздает pre-render шаблоны <20><>ля статических данных страниц серий (/catalog/seria/.../all<ID>).
|
||||
|
||||
ВАЖНО: Кешируются ТОЛЬКО статические данные (схемы, график, карта, статистика).
|
||||
Верхняя статья рендерится ДИНАМИЧЕСКИ из БД, чтобы изменения через админку
|
||||
были видны без перезагрузки контейнера.
|
||||
Таблица оконных проёмов также пересчитывается при каждом запросе.
|
||||
|
||||
Создаёт 3 файла для каждой серии:
|
||||
- _id_static_flaps.html (схемы открывания)
|
||||
- _id_static_graph.html (график ввода в эксплуатацию)
|
||||
- _id_static_map_stats.html (карта и статистика)
|
||||
"""
|
||||
|
||||
help = "Пересоздает pre-render шаблоны (3 файла) для статических данных выбранных или всех корневых серий."
|
||||
|
||||
def add_arguments(self, parser):
|
||||
parser.add_argument(
|
||||
"--seria-id",
|
||||
type=int,
|
||||
action="append",
|
||||
default=[],
|
||||
help="ID серии (можно передавать несколько раз). По умолчанию пересоздаются все корневые серии.",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--force",
|
||||
action="store_true",
|
||||
help="Пересоздать даже если pre-render файлы уже существуют.",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--dry-run",
|
||||
action="store_true",
|
||||
help="Только показать, что будет сделано, без генерации файлов.",
|
||||
)
|
||||
|
||||
def handle(self, *args, **options):
|
||||
seria_ids: list[int] = options["seria_id"]
|
||||
force: bool = options["force"]
|
||||
dry_run: bool = options["dry_run"]
|
||||
|
||||
# Берем только корневые серии, потому что для них строятся канонические URL /all<ID>.
|
||||
query = Seria_Info.objects.filter(id=F("kRoot_id")).only("id", "sName").order_by("id")
|
||||
if seria_ids:
|
||||
query = query.filter(id__in=seria_ids)
|
||||
|
||||
targets = list(query)
|
||||
if not targets:
|
||||
raise CommandError("Не найдено подходящих корневых серий для пересоздания pre-render.")
|
||||
|
||||
templates_root = Path(settings.TEMPLATES[0]["DIRS"][0])
|
||||
prepared_dir = templates_root / settings.PATH_FOR_SERIA_INFO_HTML_INCLUDE
|
||||
prepared_dir.mkdir(parents=True, exist_ok=True)
|
||||
|
||||
request_factory = RequestFactory()
|
||||
created = 0
|
||||
planned = 0
|
||||
skipped = 0
|
||||
|
||||
for seria in targets:
|
||||
# Проверяем существование вс<D0B2><D181>х 3 файлов (верхняя статья НЕ кешируется)
|
||||
target_files = [
|
||||
prepared_dir / f"{seria.id}_id_static_flaps.html",
|
||||
prepared_dir / f"{seria.id}_id_static_graph.html",
|
||||
prepared_dir / f"{seria.id}_id_static_map_stats.html",
|
||||
]
|
||||
all_exist = all(f.exists() for f in target_files)
|
||||
|
||||
if all_exist and not force:
|
||||
skipped += 1
|
||||
self.stdout.write(f"SKIP {seria.id}: все кеш-файлы уже существуют")
|
||||
continue
|
||||
|
||||
if dry_run:
|
||||
action = "REGEN" if all_exist else "CREATE"
|
||||
self.stdout.write(f"{action} {seria.id}: 3 файла (flaps, graph, map_stats)")
|
||||
planned += 1
|
||||
continue
|
||||
|
||||
# Удаляем старые файлы перед пересоздаванием
|
||||
for f in target_files:
|
||||
if f.exists():
|
||||
f.unlink()
|
||||
|
||||
slug = sanitize_slug(seria.sName)
|
||||
request = request_factory.get(f"/catalog/seria/{slug}/all{seria.id}")
|
||||
|
||||
# В команде принудительно включаем «production-mode» для вьюхи,
|
||||
# чтобы она прошла тяжелую ветку и пересоздала pre-render файлы.
|
||||
old_debug = catalog_series.DEBUG
|
||||
try:
|
||||
catalog_series.DEBUG = False
|
||||
response = catalog_series.catalog_seria_info(request, slug, seria.id)
|
||||
finally:
|
||||
catalog_series.DEBUG = old_debug
|
||||
|
||||
if response.status_code != 200:
|
||||
raise CommandError(
|
||||
f"Серия {seria.id}: ожидался status=200, получен {response.status_code}."
|
||||
)
|
||||
|
||||
# Проверяем, что все 3 файла были созданы
|
||||
missing_files = [f for f in target_files if not f.exists()]
|
||||
if missing_files:
|
||||
raise CommandError(
|
||||
f"Серия {seria.id}: не созданы файлы: {[f.name for f in missing_files]}"
|
||||
)
|
||||
|
||||
created += 1
|
||||
self.stdout.write(
|
||||
self.style.SUCCESS(f"OK {seria.id}: 3 кеш-файла созданы")
|
||||
)
|
||||
|
||||
if dry_run:
|
||||
self.stdout.write(
|
||||
self.style.SUCCESS(
|
||||
f"DRY-RUN. Обработано: {len(targets)}. Б<><D091>дет создано/пересоздано: {planned}. Пропущено: {skipped}."
|
||||
)
|
||||
)
|
||||
else:
|
||||
self.stdout.write(
|
||||
self.style.SUCCESS(
|
||||
f"Готово. Обработано: {len(targets)}. Создано/пересоздано: {created} × 3 файла. Пропущено: {skipped}."
|
||||
)
|
||||
)
|
||||
|
||||
202
oknardia/web/management/commands/regenerate_seria_roots.py
Normal file
202
oknardia/web/management/commands/regenerate_seria_roots.py
Normal file
@@ -0,0 +1,202 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
Django management command: regenerate_seria_roots
|
||||
|
||||
Пересчет корневых серий (root series) для всех серий домов.
|
||||
|
||||
Идея:
|
||||
Серии домов могут иметь сложную иерархию (дерево потомственности) из-за того,
|
||||
что на разных сайтах одна и та же серия обозначалась по-разному.
|
||||
Например серия II-57 записывалась как 2-57, И-57, П-57 и т.п.
|
||||
|
||||
Эта команда:
|
||||
1. Находит "корневые" серии - те, что используются в Apartment_Type (у них есть квартиры)
|
||||
2. Для всех остальных серий ищет их корневую серию, двигаясь вверх по дереву иерархии
|
||||
3. Устанавливает найденную корневую серию в поле kRoot_id каждой серии
|
||||
|
||||
Результат:
|
||||
- Все серии-алиасы (синонимы) указывают на одну корневую серию
|
||||
- Это позволяет консолидировать данные по сериям дома
|
||||
"""
|
||||
|
||||
from django.core.management.base import BaseCommand
|
||||
from oknardia.models import Seria_Info, Apartment_Type
|
||||
|
||||
|
||||
class Command(BaseCommand):
|
||||
help = 'Пересчитывает корневые серии для всей иерархии серий домов'
|
||||
|
||||
def add_arguments(self, parser):
|
||||
# Django уже добавляет --verbosity автоматически
|
||||
# 0 = минимум (только ошибки)
|
||||
# 1 = нормально (точки)
|
||||
# 2 = подробно (информация)
|
||||
# 3 = очень подробно (таблица)
|
||||
pass
|
||||
|
||||
def handle(self, *args, **options):
|
||||
verbose = int(options.get('verbosity', 1))
|
||||
|
||||
self.stdout.write(self.style.SUCCESS('=== ПЕРЕСЧЕТ КОРНЕВЫХ СЕРИЙ ===\n'))
|
||||
|
||||
time_start = self.get_time()
|
||||
|
||||
# ========== ЭТАП 1: Находим корневые серии ==========
|
||||
if verbose >= 1:
|
||||
self.stdout.write('Этап 1: Ищем корневые серии в таблице квартир...\n')
|
||||
|
||||
# Получаем все УНИКАЛЬНЫЕ серии, которые используются в квартирах
|
||||
root_series_ids = list(
|
||||
Apartment_Type.objects.values_list('kSeria_id', flat=True).distinct()
|
||||
)
|
||||
|
||||
# Получаем объекты корневых серий (для вывода их названий)
|
||||
root_series = Seria_Info.objects.filter(id__in=root_series_ids)
|
||||
|
||||
if verbose >= 1:
|
||||
self.stdout.write(f'✓ Найдено корневых серий: {len(root_series_ids)}\n')
|
||||
|
||||
# Устанавливаем для корневых серий kRoot_id = own_id
|
||||
for seria in root_series:
|
||||
seria.kRoot_id = seria.id # Серия сама себе корень
|
||||
seria.save()
|
||||
|
||||
if verbose >= 3:
|
||||
# Очень подробный вывод - таблица
|
||||
self.stdout.write(
|
||||
f' ✓ {seria.id:04d} | {seria.sName:<30} | корневая'
|
||||
)
|
||||
elif verbose >= 2:
|
||||
# Подробный - с названием
|
||||
self.stdout.write(f' ✓ {seria.id:04d} {seria.sName}')
|
||||
elif verbose >= 1:
|
||||
# Нормально - точки
|
||||
self.stdout.write('.', ending='')
|
||||
|
||||
if verbose < 2:
|
||||
self.stdout.write('\n')
|
||||
self.stdout.write('')
|
||||
|
||||
# ========== ЭТАП 2: Обрабатываем все серии ==========
|
||||
if verbose >= 1:
|
||||
self.stdout.write('\nЭтап 2: Главная магия - обрабатываем все серии в иерархии...\n')
|
||||
|
||||
if verbose >= 3:
|
||||
# Заголовок таблицы для очень подробного режима
|
||||
self.stdout.write('-' * 110)
|
||||
self.stdout.write(
|
||||
f'{"ID":>5} | {"Название":<35} | {"Родитель":>10} | '
|
||||
f'{"Путь":<40} | {"Результат":<20}'
|
||||
)
|
||||
self.stdout.write('-' * 110)
|
||||
|
||||
all_series = Seria_Info.objects.all()
|
||||
count_with_root = 0
|
||||
count_without_root = 0
|
||||
count_errors = 0
|
||||
count_root = 0
|
||||
|
||||
for seria in all_series:
|
||||
try:
|
||||
# Если это уже корневая серия, пропускаем
|
||||
if seria.id in root_series_ids:
|
||||
count_root += 1
|
||||
if verbose >= 3:
|
||||
self.stdout.write(
|
||||
f'{seria.id:>5} | {seria.sName:<35} | '
|
||||
f'{str(seria.kParent_id or "-"):>10} | '
|
||||
f'(корневая) | ✓'
|
||||
)
|
||||
elif verbose >= 2:
|
||||
self.stdout.write(f' {seria.id:04d}: корневая')
|
||||
elif verbose >= 1:
|
||||
self.stdout.write('.', ending='')
|
||||
continue
|
||||
|
||||
# Движемся вверх по дереву потомок → предок
|
||||
current_id = seria.kParent_id
|
||||
path_trace = []
|
||||
|
||||
# Рекурсивно ищем корневую серию
|
||||
while current_id is not None:
|
||||
path_trace.append(current_id)
|
||||
try:
|
||||
parent_seria = Seria_Info.objects.get(id=current_id)
|
||||
|
||||
# Проверяем: либо у родителя нет родителя, либо родитель - корневая
|
||||
if parent_seria.kParent_id is None or current_id in root_series_ids:
|
||||
break
|
||||
|
||||
current_id = parent_seria.kParent_id
|
||||
except Seria_Info.DoesNotExist:
|
||||
current_id = None
|
||||
break
|
||||
|
||||
# Проверяем, что нашли корневую серию
|
||||
if current_id and current_id in root_series_ids:
|
||||
seria.kRoot_id = current_id
|
||||
seria.save()
|
||||
|
||||
if verbose >= 3:
|
||||
path_str = ' → '.join(str(i) for i in path_trace)
|
||||
self.stdout.write(
|
||||
f'{seria.id:>5} | {seria.sName:<35} | '
|
||||
f'{str(seria.kParent_id or "-"):>10} | '
|
||||
f'{path_str:<40} | ✓ Корень #{current_id}'
|
||||
)
|
||||
elif verbose >= 2:
|
||||
self.stdout.write(f' {seria.id:04d}: корень → {current_id}')
|
||||
elif verbose >= 1:
|
||||
self.stdout.write('+', ending='')
|
||||
|
||||
count_with_root += 1
|
||||
else:
|
||||
seria.kRoot_id = None
|
||||
seria.save()
|
||||
|
||||
if verbose >= 3:
|
||||
path_str = ' → '.join(str(i) for i in path_trace) if path_trace else '(нет)'
|
||||
self.stdout.write(
|
||||
f'{seria.id:>5} | {seria.sName:<35} | '
|
||||
f'{str(seria.kParent_id or "-"):>10} | '
|
||||
f'{path_str:<40} | ✗ Нет корня'
|
||||
)
|
||||
elif verbose >= 2:
|
||||
self.stdout.write(f' {seria.id:04d}: Нет корня')
|
||||
elif verbose >= 1:
|
||||
self.stdout.write('-', ending='')
|
||||
|
||||
count_without_root += 1
|
||||
|
||||
except Exception as e:
|
||||
if verbose >= 3:
|
||||
self.stdout.write(
|
||||
self.style.ERROR(f'{seria.id:>5} | ОШИБКА: {str(e):<80}')
|
||||
)
|
||||
elif verbose >= 1:
|
||||
self.stdout.write('E', ending='')
|
||||
count_errors += 1
|
||||
|
||||
if verbose >= 3:
|
||||
self.stdout.write('-' * 110)
|
||||
elif verbose < 2:
|
||||
self.stdout.write('\n')
|
||||
|
||||
# ========== РЕЗУЛЬТАТЫ ==========
|
||||
self.stdout.write(self.style.SUCCESS('\n\n=== РЕЗУЛЬТАТЫ ==='))
|
||||
self.stdout.write(f'✓ Корневых серий (обработаны на этапе 1): {count_root}')
|
||||
self.stdout.write(f'✓ Серий с найденным корнем: {count_with_root}')
|
||||
self.stdout.write(f'⚠ Серий без корня: {count_without_root}')
|
||||
if count_errors > 0:
|
||||
self.stdout.write(self.style.ERROR(f'✗ Ошибок: {count_errors}'))
|
||||
|
||||
time_elapsed = self.get_time() - time_start
|
||||
self.stdout.write(
|
||||
self.style.SUCCESS(f'\n✅ Пересчет завершен! Время: {time_elapsed:.2f}с')
|
||||
)
|
||||
|
||||
@staticmethod
|
||||
def get_time():
|
||||
import time
|
||||
return time.perf_counter()
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -3,9 +3,11 @@
|
||||
from django.shortcuts import render, redirect
|
||||
from django.http import HttpRequest, HttpResponse
|
||||
from django.utils.dateformat import format
|
||||
from django.utils import timezone
|
||||
from django.db.models import F, Q, ExpressionWrapper, BooleanField, Max, Count, Avg
|
||||
from oknardia.models import LogVisitPriceReport, SetKit
|
||||
from oknardia.settings import *
|
||||
from web.add_func import normalize, get_rating_set_for_stars, sum_through
|
||||
from web.add_func import normalize, get_rating_set_for_stars, sum_through, sanitize_slug
|
||||
# from time import time
|
||||
import django.utils.dateformat
|
||||
import time
|
||||
@@ -13,20 +15,84 @@ import json
|
||||
import re
|
||||
import pytils
|
||||
|
||||
# Сигнальные значения для поиска min/max: заведомо вне диапазона реальных данных
|
||||
_INI_MAX = -100_000
|
||||
_INI_MIN = 1_000_000
|
||||
|
||||
def get_last_user_visit_cookies(request: HttpRequest) -> list:
|
||||
""" Служебная функция: проверяет есть ли куки о последних посещениях пользователя, и если есть возвращает их
|
||||
|
||||
:param request: HttpRequest -- входящий http-запрос
|
||||
:return LastVisit: json -- загруженный json-объект из куки LastVisit
|
||||
def _color_hi(value, val_min: float, val_max: float, threshold=None, epsilon: float = 0.001) -> str | None:
|
||||
"""Цвет ячейки "чем больше, тем лучше": зеленее → значение ближе к max.
|
||||
|
||||
:param value: значение поля для текущей строки
|
||||
:param val_min: минимум по всей выборке (из первого прогона)
|
||||
:param val_max: максимум по всей выборке
|
||||
:param threshold: нижний порог: значения <= threshold считаются "нет данных" и не окрашиваются
|
||||
:param epsilon: минимальный разброс, при котором окраска имеет смысл
|
||||
:return: hex-строка цвета или None
|
||||
"""
|
||||
if "LastVisit" in request.COOKIES:
|
||||
try:
|
||||
v = float(value)
|
||||
except (TypeError, ValueError):
|
||||
return None
|
||||
if val_max == _INI_MAX or val_min == _INI_MIN or val_max - val_min < epsilon:
|
||||
return None
|
||||
if threshold is not None and v <= threshold:
|
||||
return None
|
||||
if v <= val_min:
|
||||
return None
|
||||
ratio = (v - val_min) / (val_max - val_min)
|
||||
c = 255 - int(ratio * 128)
|
||||
return f"#{c:02x}ff{c:02x}"
|
||||
|
||||
|
||||
def _color_lo(value, val_min: float, val_max: float, threshold=None, epsilon: float = 0.001) -> str | None:
|
||||
"""Цвет ячейки "чем меньше, тем лучше": зеленее → значение ближе к min.
|
||||
|
||||
:param value: значение поля для текущей строки
|
||||
:param val_min: минимум по всей выборке
|
||||
:param val_max: максимум по всей выборке
|
||||
:param threshold: нижний порог: значения <= threshold не окрашиваются
|
||||
:param epsilon: минимальный разброс
|
||||
:return: hex-строка цвета или None
|
||||
"""
|
||||
try:
|
||||
v = float(value)
|
||||
except (TypeError, ValueError):
|
||||
return None
|
||||
if val_max == _INI_MAX or val_min == _INI_MIN or val_max - val_min < epsilon:
|
||||
return None
|
||||
if threshold is not None and v <= threshold:
|
||||
return None
|
||||
if v >= val_max:
|
||||
return None
|
||||
ratio = (v - val_min) / (val_max - val_min)
|
||||
c = 127 + int(ratio * 128)
|
||||
return f"#{c:02x}ff{c:02x}"
|
||||
|
||||
|
||||
def _bounds(items: list, field: str, threshold=None) -> tuple[float, float]:
|
||||
"""Вычисляет (min, max) значений поля field по списку items, игнорируя None и <= threshold.
|
||||
|
||||
:param items: список объектов (SetKit с аннотациями)
|
||||
:param field: имя атрибута
|
||||
:param threshold: значения <= threshold исключаются из выборки
|
||||
:return: (min, max) или (_INI_MIN, _INI_MAX) если нет валидных значений
|
||||
"""
|
||||
vals = []
|
||||
for item in items:
|
||||
raw = getattr(item, field, None)
|
||||
if raw is None:
|
||||
continue
|
||||
try:
|
||||
return json.loads(request.COOKIES["LastVisit"])
|
||||
except (json.decoder.JSONDecodeError, TypeError, ValueError, KeyError, AttributeError):
|
||||
return []
|
||||
else:
|
||||
return []
|
||||
v = float(raw)
|
||||
except (TypeError, ValueError):
|
||||
continue
|
||||
if threshold is not None and v <= threshold:
|
||||
continue
|
||||
vals.append(v)
|
||||
if not vals:
|
||||
return _INI_MIN, _INI_MAX
|
||||
return min(vals), max(vals)
|
||||
|
||||
|
||||
def get_last_user_visit_list(list_visit: list) -> list:
|
||||
@@ -80,8 +146,8 @@ def compare_offers(request: HttpRequest, to_compare: str = "1,2") -> HttpRespons
|
||||
:param to_compare: str -- список ,через запятую, id оконных наборов (Set) для сравнения
|
||||
:return: HttpResponse --
|
||||
"""
|
||||
time_start = time.time()
|
||||
to_template = {}
|
||||
time_start = time.perf_counter()
|
||||
to_template: dict[str, object] = {}
|
||||
try:
|
||||
# Этот блок нужен для 302-переадресации, когда разные URL отдают одинаковые страницы.
|
||||
# Например, такое происходит для страницы: /compare_offers/1,2 и /compare_offers/2,1
|
||||
@@ -107,55 +173,57 @@ def compare_offers(request: HttpRequest, to_compare: str = "1,2") -> HttpRespons
|
||||
if to_compare != list_fine:
|
||||
return redirect(f"/compare_offers/{list_fine}")
|
||||
try:
|
||||
q_set_kit = SetKit.objects.raw(
|
||||
f"SELECT "
|
||||
f"oknardia_setkit.id, oknardia_setkit.sSetName, oknardia_setkit.sSetDescription,"
|
||||
f"oknardia_setkit.sSetClimateControl, oknardia_setkit.sSetSill, oknardia_setkit.sSetImplementAll,"
|
||||
f"oknardia_setkit.sSetImplementHandles, oknardia_setkit.sSetImplementHinges,"
|
||||
f"oknardia_setkit.sSetImplementLatch, oknardia_setkit.sSetImplementLimiter,"
|
||||
f"oknardia_setkit.sSetImplementCatch, oknardia_setkit.sSetPanes, oknardia_setkit.sSetSlope,"
|
||||
f"oknardia_setkit.sSetDelivery, oknardia_setkit.bSetDelivery, oknardia_setkit.sSetUninstallInstall,"
|
||||
f"oknardia_setkit.bSetUninstallInstall, oknardia_setkit.sSetOtherConditions,"
|
||||
f"oknardia_setkit.fSetRating, oknardia_setkit.iSetNumEval, oknardia_setkit.iSetImpressions,"
|
||||
f"oknardia_setkit.iSetViews, oknardia_setkit.sSetActive, oknardia_setkit.dSetModify,"
|
||||
f"(oknardia_setkit.dSetCommercialUntil > NOW()) AS bCommercial,"
|
||||
f"oknardia_glazing.sGlazingReflectionAndAbsorptionOfHeat, oknardia_glazing.sGlazingBriefDescription,"
|
||||
f"oknardia_glazing.sGlazingDescription, oknardia_glazing.fGlazingSoundproofing,"
|
||||
f"oknardia_glazing.fGlazingRating, oknardia_glazing.sGlazingMark,"
|
||||
f"oknardia_glazing.fGlazingHeatTransfer, oknardia_glazing.fGlazingLightTransmission,"
|
||||
f"oknardia_glazing.fGlazingPassingSun, oknardia_glazing.sGlazingLightReflectance,"
|
||||
f"oknardia_glazing.sGlazingManufacturer, oknardia_glazing.iGlazingCamerasN,"
|
||||
f"oknardia_glazing.sGlazingToning, oknardia_glazing.iGlazingThickness,"
|
||||
f"oknardia_merchantoffice.dOfficeDataCreate, oknardia_merchantoffice.sOfficeName,"
|
||||
f"oknardia_merchantoffice.sOfficeStatus, oknardia_merchantoffice.sOfficePhones,"
|
||||
f"oknardia_merchantoffice.sOfficeEmails, oknardia_merchantoffice.sOfficeDescription,"
|
||||
f"oknardia_merchantoffice.sOfficeDiscountMetaFormula, oknardia_merchantoffice.fOfficeGeoCode_Latitude,"
|
||||
f"oknardia_merchantoffice.fOfficeGeoCode_Longitude, oknardia_merchantoffice.sOfficeAddress,"
|
||||
f"oknardia_ouruser.sUserAvatarImg, oknardia_ouruser.sUserJobTitle, oknardia_ouruser.bUserSubscribe,"
|
||||
f"oknardia_ouruser.sUserPhone, oknardia_ouruser.sUserStatus, oknardia_merchantbrand.id AS MERCHANT_ID,"
|
||||
f"oknardia_merchantbrand.sMerchantMainURL, oknardia_merchantbrand.sMerchantName,"
|
||||
f"oknardia_merchantbrand.pMerchantLogo, oknardia_pvcprofiles.id AS PROFILE_ID,"
|
||||
f"oknardia_pvcprofiles.sProfileName, oknardia_pvcprofiles.sProfileBriefDescription,"
|
||||
f"oknardia_pvcprofiles.sProfileReinforcement, oknardia_pvcprofiles.sProfileDescription,"
|
||||
f"oknardia_pvcprofiles.fProfileHeatTransf, oknardia_pvcprofiles.sProfileSealDescription,"
|
||||
f"oknardia_pvcprofiles.fProfileSeals, oknardia_pvcprofiles.fProfileSoundproofing,"
|
||||
f"oknardia_pvcprofiles.iProfileCameras, oknardia_pvcprofiles.iProfileGlazingThickness,"
|
||||
f"oknardia_pvcprofiles.iProfileHeight, oknardia_pvcprofiles.iProfileRabbet,"
|
||||
f"oknardia_pvcprofiles.iProfileThickness, oknardia_pvcprofiles.sProfileColor,"
|
||||
f"oknardia_pvcprofiles.sProfileFillet, oknardia_pvcprofiles.sProfileManufacturer,"
|
||||
f"oknardia_pvcprofiles.sProfileOther, oknardia_pvcprofiles.fProfileRating "
|
||||
f"FROM oknardia_setkit"
|
||||
f" INNER JOIN oknardia_pvcprofiles"
|
||||
f" ON oknardia_setkit.kSet2PVCprofiles_id = oknardia_pvcprofiles.id"
|
||||
f" INNER JOIN oknardia_glazing"
|
||||
f" ON oknardia_setkit.kSet2Glazing_id = oknardia_glazing.id"
|
||||
f" INNER JOIN oknardia_ouruser"
|
||||
f" ON oknardia_setkit.kSet2User_id = oknardia_ouruser.id"
|
||||
f" INNER JOIN oknardia_merchantoffice"
|
||||
f" ON oknardia_ouruser.kMerchantOffice_id = oknardia_merchantoffice.id"
|
||||
f" INNER JOIN oknardia_merchantbrand"
|
||||
f" ON oknardia_merchantoffice.kMerchantName_id = oknardia_merchantbrand.id "
|
||||
f"WHERE oknardia_setkit.id IN ({to_compare})")
|
||||
q_set_kit = (
|
||||
SetKit.objects
|
||||
.filter(id__in=list_fin)
|
||||
.annotate(
|
||||
# Активность коммерческого предложения (аналог dSetCommercialUntil > NOW())
|
||||
bCommercial=ExpressionWrapper(
|
||||
Q(dSetCommercialUntil__gt=timezone.now()),
|
||||
output_field=BooleanField()
|
||||
),
|
||||
# Алиасы из MerchantBrand (SetKit → OurUser → MerchantOffice → MerchantBrand)
|
||||
MERCHANT_ID=F('kSet2User__kMerchantOffice__kMerchantName'),
|
||||
sMerchantName=F('kSet2User__kMerchantOffice__kMerchantName__sMerchantName'),
|
||||
sMerchantMainURL=F('kSet2User__kMerchantOffice__kMerchantName__sMerchantMainURL'),
|
||||
pMerchantLogo=F('kSet2User__kMerchantOffice__kMerchantName__pMerchantLogo'),
|
||||
# Алиасы из PVCprofiles (SetKit → kSet2PVCprofiles)
|
||||
PROFILE_ID=F('kSet2PVCprofiles'),
|
||||
sProfileName=F('kSet2PVCprofiles__sProfileName'),
|
||||
sProfileBriefDescription=F('kSet2PVCprofiles__sProfileBriefDescription'),
|
||||
sProfileManufacturer=F('kSet2PVCprofiles__sProfileManufacturer'),
|
||||
sProfileColor=F('kSet2PVCprofiles__sProfileColor'),
|
||||
iProfileCameras=F('kSet2PVCprofiles__iProfileCameras'),
|
||||
iProfileThickness=F('kSet2PVCprofiles__iProfileThickness'),
|
||||
iProfileGlazingThickness=F('kSet2PVCprofiles__iProfileGlazingThickness'),
|
||||
fProfileHeatTransf=F('kSet2PVCprofiles__fProfileHeatTransf'),
|
||||
fProfileSeals=F('kSet2PVCprofiles__fProfileSeals'),
|
||||
sProfileSealDescription=F('kSet2PVCprofiles__sProfileSealDescription'),
|
||||
fProfileSoundproofing=F('kSet2PVCprofiles__fProfileSoundproofing'),
|
||||
iProfileHeight=F('kSet2PVCprofiles__iProfileHeight'),
|
||||
iProfileRabbet=F('kSet2PVCprofiles__iProfileRabbet'),
|
||||
sProfileFillet=F('kSet2PVCprofiles__sProfileFillet'),
|
||||
sProfileReinforcement=F('kSet2PVCprofiles__sProfileReinforcement'),
|
||||
sProfileOther=F('kSet2PVCprofiles__sProfileOther'),
|
||||
fProfileRating=F('kSet2PVCprofiles__fProfileRating'),
|
||||
sProfileDescription=F('kSet2PVCprofiles__sProfileDescription'),
|
||||
# Алиасы из Glazing (SetKit → kSet2Glazing)
|
||||
iGlazingCamerasN=F('kSet2Glazing__iGlazingCamerasN'),
|
||||
iGlazingThickness=F('kSet2Glazing__iGlazingThickness'),
|
||||
sGlazingBriefDescription=F('kSet2Glazing__sGlazingBriefDescription'),
|
||||
sGlazingDescription=F('kSet2Glazing__sGlazingDescription'),
|
||||
sGlazingMark=F('kSet2Glazing__sGlazingMark'),
|
||||
sGlazingManufacturer=F('kSet2Glazing__sGlazingManufacturer'),
|
||||
fGlazingHeatTransfer=F('kSet2Glazing__fGlazingHeatTransfer'),
|
||||
fGlazingSoundproofing=F('kSet2Glazing__fGlazingSoundproofing'),
|
||||
fGlazingLightTransmission=F('kSet2Glazing__fGlazingLightTransmission'),
|
||||
sGlazingLightReflectance=F('kSet2Glazing__sGlazingLightReflectance'),
|
||||
fGlazingPassingSun=F('kSet2Glazing__fGlazingPassingSun'),
|
||||
sGlazingReflectionAndAbsorptionOfHeat=F('kSet2Glazing__sGlazingReflectionAndAbsorptionOfHeat'),
|
||||
sGlazingToning=F('kSet2Glazing__sGlazingToning'),
|
||||
fGlazingRating=F('kSet2Glazing__fGlazingRating'),
|
||||
)
|
||||
)
|
||||
except SetKit.DoesNotExist:
|
||||
return redirect("/compare_offers/1,2")
|
||||
list_set_kit = list(q_set_kit)
|
||||
@@ -166,356 +234,156 @@ def compare_offers(request: HttpRequest, to_compare: str = "1,2") -> HttpRespons
|
||||
except (ValueError, TypeError):
|
||||
return render("/compare_offers/1,2")
|
||||
# ПРЕДВАРИТЕЛЬНЫЙ "ПРОГОН"
|
||||
# Для того, чтобы "покрасить" ячейки таблицы сравнения в цвета, нужно для некоторых полей найти min и max...
|
||||
ini_max = -100000
|
||||
ini_min = 1000000
|
||||
max_i_profile_cameras = max_f_profile_seals = max_i_profile_thickness = max_i_profile_glazing_thickness = \
|
||||
max_f_profile_heat_transf = max_f_profile_soundproofing = max_i_profile_rabbet = max_i_profile_height = \
|
||||
max_i_glazing_cameras_n = max_i_glazing_thickness = max_f_glazing_heat_transfer = max_rating_set = \
|
||||
max_f_glazing_soundproofing = max_f_glazing_light_transmission = max_f_glazing_passing_sun = ini_max
|
||||
min_i_profile_cameras = min_f_profile_seals = min_i_profile_thickness = min_i_profile_glazing_thickness = \
|
||||
min_f_profile_heat_transf = min_f_profile_soundproofing = min_i_profile_rabbet = min_i_profile_height = \
|
||||
min_i_glazing_cameras_n = min_i_glazing_thickness = min_f_glazing_heat_transfer = min_rating_set = \
|
||||
min_f_glazing_soundproofing = min_f_glazing_light_transmission = min_f_glazing_passing_sun = ini_min
|
||||
list_of_merchant_name = []
|
||||
list_of_profile_name = []
|
||||
list_of_glazing_brief_description = []
|
||||
for i in list_set_kit:
|
||||
if i.sMerchantName not in list_of_merchant_name:
|
||||
list_of_merchant_name.append(i.sMerchantName)
|
||||
if i.sProfileName not in list_of_profile_name:
|
||||
list_of_profile_name.append(i.sProfileName)
|
||||
if i.sGlazingMark not in list_of_glazing_brief_description:
|
||||
list_of_glazing_brief_description.append(i.sGlazingMark)
|
||||
profile_num_cameras = sum_through(i.iProfileCameras)
|
||||
if profile_num_cameras > 0: # Общее число камер профиля (рама+створка)
|
||||
if profile_num_cameras > max_i_profile_cameras:
|
||||
max_i_profile_cameras = profile_num_cameras
|
||||
if profile_num_cameras < min_i_profile_cameras:
|
||||
min_i_profile_cameras = profile_num_cameras
|
||||
if i.iProfileThickness > 0: # Контуров уплотнения
|
||||
if i.fProfileSeals > max_f_profile_seals:
|
||||
max_f_profile_seals = i.fProfileSeals
|
||||
if i.fProfileSeals < min_f_profile_seals:
|
||||
min_f_profile_seals = i.fProfileSeals
|
||||
if i.iProfileThickness > 10: # Монтажная ширина профиля
|
||||
if i.iProfileThickness > max_i_profile_thickness:
|
||||
max_i_profile_thickness = i.iProfileThickness
|
||||
if i.iProfileThickness < min_i_profile_thickness:
|
||||
min_i_profile_thickness = i.iProfileThickness
|
||||
if i.iProfileGlazingThickness > 4: # Максимальная толщина стеклопакета
|
||||
if i.iProfileGlazingThickness > max_i_profile_glazing_thickness:
|
||||
max_i_profile_glazing_thickness = i.iProfileGlazingThickness
|
||||
if i.iProfileGlazingThickness < min_i_profile_glazing_thickness:
|
||||
min_i_profile_glazing_thickness = i.iProfileGlazingThickness
|
||||
if i.fProfileHeatTransf > 0: # Сопротивление теплопередаче
|
||||
if i.fProfileHeatTransf > max_f_profile_heat_transf:
|
||||
max_f_profile_heat_transf = i.fProfileHeatTransf
|
||||
if i.fProfileHeatTransf < min_f_profile_heat_transf:
|
||||
min_f_profile_heat_transf = i.fProfileHeatTransf
|
||||
if i.fProfileSoundproofing > 0: # Коэффициент звукоизоляции
|
||||
if i.fProfileSoundproofing > max_f_profile_soundproofing:
|
||||
max_f_profile_soundproofing = i.fProfileSoundproofing
|
||||
if i.fProfileSoundproofing < min_f_profile_soundproofing:
|
||||
min_f_profile_soundproofing = i.fProfileSoundproofing
|
||||
if i.iProfileRabbet > 1: # Фальц
|
||||
if i.iProfileRabbet > max_i_profile_rabbet:
|
||||
max_i_profile_rabbet = i.iProfileRabbet
|
||||
if i.iProfileRabbet < min_i_profile_rabbet:
|
||||
min_i_profile_rabbet = i.iProfileRabbet
|
||||
if i.iProfileHeight > 12: # Высота в световом проеме
|
||||
if i.iProfileHeight > max_i_profile_height:
|
||||
max_i_profile_height = i.iProfileHeight
|
||||
if i.iProfileHeight < min_i_profile_height:
|
||||
min_i_profile_height = i.iProfileHeight
|
||||
if i.iGlazingCamerasN > 0: # Камер стеклопакета
|
||||
if i.iGlazingCamerasN > max_i_glazing_cameras_n:
|
||||
max_i_glazing_cameras_n = i.iGlazingCamerasN
|
||||
if i.iGlazingCamerasN < min_i_glazing_cameras_n:
|
||||
min_i_glazing_cameras_n = i.iGlazingCamerasN
|
||||
if i.iGlazingThickness > 4: # Толщина стеклопакета
|
||||
if i.iGlazingThickness > max_i_glazing_thickness:
|
||||
max_i_glazing_thickness = i.iGlazingThickness
|
||||
if i.iGlazingThickness < min_i_glazing_thickness:
|
||||
min_i_glazing_thickness = i.iGlazingThickness
|
||||
if i.fGlazingHeatTransfer > 0.05: # Сопротивление теплопередаче стеклопакета Ro (м²×°C/Вт)
|
||||
if i.fGlazingHeatTransfer > max_f_glazing_heat_transfer:
|
||||
max_f_glazing_heat_transfer = i.fGlazingHeatTransfer
|
||||
if i.fGlazingHeatTransfer < min_f_glazing_heat_transfer:
|
||||
min_f_glazing_heat_transfer = i.fGlazingHeatTransfer
|
||||
if i.fGlazingSoundproofing > 5: # Коэффициент звукоизоляции стеклопакета
|
||||
if i.fGlazingSoundproofing > max_f_glazing_soundproofing:
|
||||
max_f_glazing_soundproofing = i.fGlazingSoundproofing
|
||||
if i.fGlazingSoundproofing < min_f_glazing_soundproofing:
|
||||
min_f_glazing_soundproofing = i.fGlazingSoundproofing
|
||||
if i.fGlazingLightTransmission > 5: # Коэффициент светопропускания стеклопакета
|
||||
if i.fGlazingLightTransmission > max_f_glazing_light_transmission:
|
||||
max_f_glazing_light_transmission = i.fGlazingLightTransmission
|
||||
if i.fGlazingLightTransmission < min_f_glazing_light_transmission:
|
||||
min_f_glazing_light_transmission = i.fGlazingLightTransmission
|
||||
if i.fGlazingPassingSun > 5: # Коэффициент солнцепропускания стеклопакета
|
||||
if i.fGlazingPassingSun > max_f_glazing_passing_sun:
|
||||
max_f_glazing_passing_sun = i.fGlazingPassingSun
|
||||
if i.fGlazingPassingSun < min_f_glazing_passing_sun:
|
||||
min_f_glazing_passing_sun = i.fGlazingPassingSun
|
||||
if i.fSetRating > 0.05: # Рейтинг НАБОРА!
|
||||
if i.fSetRating > max_rating_set:
|
||||
max_rating_set = i.fSetRating
|
||||
if i.fSetRating < min_rating_set:
|
||||
min_rating_set = i.fSetRating
|
||||
# Вычисляем min/max по каждому параметру для дальнейшей покраски ячеек.
|
||||
# Камеры профиля требуют sum_through() — обрабатываем отдельно.
|
||||
cameras_vals = [
|
||||
c for i in list_set_kit
|
||||
if (c := sum_through(i.iProfileCameras)) is not None and c > 0
|
||||
]
|
||||
min_cameras = min(cameras_vals) if cameras_vals else _INI_MIN
|
||||
max_cameras = max(cameras_vals) if cameras_vals else _INI_MAX
|
||||
|
||||
# Остальные поля — через _bounds() с соответствующими порогами
|
||||
# (threshold: значения <= порога считаются "нет данных" и исключаются из диапазона)
|
||||
min_seals, max_seals = _bounds(list_set_kit, 'fProfileSeals', threshold=0)
|
||||
min_thick, max_thick = _bounds(list_set_kit, 'iProfileThickness', threshold=10)
|
||||
min_glaz_d, max_glaz_d = _bounds(list_set_kit, 'iProfileGlazingThickness', threshold=4)
|
||||
min_heat_p, max_heat_p = _bounds(list_set_kit, 'fProfileHeatTransf', threshold=0)
|
||||
min_sound_p, max_sound_p = _bounds(list_set_kit, 'fProfileSoundproofing', threshold=0)
|
||||
min_rabbet, max_rabbet = _bounds(list_set_kit, 'iProfileRabbet', threshold=1)
|
||||
min_height, max_height = _bounds(list_set_kit, 'iProfileHeight', threshold=12)
|
||||
min_gl_cam, max_gl_cam = _bounds(list_set_kit, 'iGlazingCamerasN', threshold=0)
|
||||
min_gl_thick, max_gl_thick = _bounds(list_set_kit, 'iGlazingThickness', threshold=3)
|
||||
min_heat_g, max_heat_g = _bounds(list_set_kit, 'fGlazingHeatTransfer', threshold=0.05)
|
||||
min_sound_g, max_sound_g = _bounds(list_set_kit, 'fGlazingSoundproofing', threshold=5)
|
||||
min_light, max_light = _bounds(list_set_kit, 'fGlazingLightTransmission', threshold=5)
|
||||
min_sun, max_sun = _bounds(list_set_kit, 'fGlazingPassingSun', threshold=5)
|
||||
min_rating, max_rating = _bounds(list_set_kit, 'fSetRating', threshold=0.05)
|
||||
|
||||
list_of_merchant_name = list({i.sMerchantName for i in list_set_kit})
|
||||
list_of_profile_name = list({i.sProfileName for i in list_set_kit})
|
||||
list_of_glazing_brief = list({i.sGlazingMark for i in list_set_kit})
|
||||
|
||||
# ОКОНЧАТЕЛЬНЫЙ ПРОГОН
|
||||
# Передаём данные из SQL-запроса шаблон. Иногда надо вычислять цвета и прочее.
|
||||
# Много макаронного стиля кодинга, из-за того что иначе придется передавать в функции большие массивы QuerySet.
|
||||
# А это жрет много памяти.
|
||||
# Формируем список словарей для шаблона; цвета вычисляются через хелперы _color_hi / _color_lo.
|
||||
dim = []
|
||||
for i in list_set_kit:
|
||||
# построим массив "цветов" для рейтинга "Общее число камер профиля (рама+створка)" (чем больше, тем лучше)
|
||||
profile_num_cameras = sum_through(i.iProfileCameras)
|
||||
if max_i_profile_cameras == ini_max or min_i_profile_cameras == ini_min or profile_num_cameras <= 1 \
|
||||
or profile_num_cameras == min_i_profile_cameras or max_i_profile_cameras - min_i_profile_cameras < 0.001:
|
||||
profile_num_cameras_color = None
|
||||
else:
|
||||
color_ratio = (profile_num_cameras - min_i_profile_cameras) / (
|
||||
max_i_profile_cameras - min_i_profile_cameras)
|
||||
profile_num_cameras_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Контуров уплотнения" (чем больше, тем лучше)
|
||||
if max_f_profile_seals == ini_max or min_f_profile_seals == ini_min or i.fProfileSeals <= 0 \
|
||||
or i.fProfileSeals == min_f_profile_seals or max_f_profile_seals - min_f_profile_seals < 0.001:
|
||||
profile_seals_color = None
|
||||
else:
|
||||
color_ratio = (i.fProfileSeals - min_f_profile_seals) / (max_f_profile_seals - min_f_profile_seals)
|
||||
profile_seals_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Монтажная ширина профиля" (чем больше, тем лучше)
|
||||
if max_i_profile_thickness == ini_max or min_i_profile_thickness == ini_min or i.iProfileThickness <= 10 \
|
||||
or i.iProfileThickness == min_i_profile_thickness \
|
||||
or max_i_profile_thickness - min_i_profile_thickness < 0.001:
|
||||
profile_thickness_color = None
|
||||
else:
|
||||
color_ratio = (i.iProfileThickness - min_i_profile_thickness) / (max_i_profile_thickness
|
||||
- min_i_profile_thickness)
|
||||
profile_thickness_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Максимальная толщина стеклопакета" (чем больше, тем лучше)
|
||||
if max_i_profile_glazing_thickness == ini_max or min_i_profile_glazing_thickness == ini_min \
|
||||
or i.iProfileGlazingThickness <= 4 or i.iProfileGlazingThickness == min_i_profile_glazing_thickness \
|
||||
or max_i_profile_glazing_thickness - min_i_profile_glazing_thickness < 0.001:
|
||||
profile_glazing_thickness_color = None
|
||||
else:
|
||||
color_ratio = (i.iProfileGlazingThickness
|
||||
- min_i_profile_glazing_thickness) / (max_i_profile_glazing_thickness
|
||||
- min_i_profile_glazing_thickness)
|
||||
profile_glazing_thickness_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Сопротивление теплопередаче" (чем больше, тем лучше)
|
||||
if max_f_profile_heat_transf == ini_max or min_f_profile_heat_transf == ini_min \
|
||||
or i.fProfileHeatTransf == min_f_profile_heat_transf \
|
||||
or max_f_profile_heat_transf - min_f_profile_heat_transf < 0.001:
|
||||
profile_heat_transf_color = None
|
||||
else:
|
||||
color_ratio = (i.fProfileHeatTransf - min_f_profile_heat_transf) / (max_f_profile_heat_transf
|
||||
- min_f_profile_heat_transf)
|
||||
profile_heat_transf_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Коэффициент звукоизоляции" (чем больше, тем лучше)
|
||||
if max_f_profile_soundproofing == ini_max or min_f_profile_soundproofing == ini_min \
|
||||
or i.fProfileSoundproofing == min_f_profile_soundproofing \
|
||||
or max_f_profile_soundproofing - min_f_profile_soundproofing < 0.001:
|
||||
profile_soundproofing_color = None
|
||||
else:
|
||||
color_ratio = (i.fProfileSoundproofing - min_f_profile_soundproofing) / (max_f_profile_soundproofing
|
||||
- min_f_profile_soundproofing)
|
||||
profile_soundproofing_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Фальц" (чем больше, тем лучше)
|
||||
if max_i_profile_rabbet == ini_max or min_i_profile_rabbet == ini_min or i.iProfileRabbet <= 1 \
|
||||
or i.iProfileRabbet == min_i_profile_rabbet or max_i_profile_rabbet - min_i_profile_rabbet < 0.001:
|
||||
profile_rabbet_color = None
|
||||
else:
|
||||
color_ratio = (i.iProfileRabbet - min_i_profile_rabbet) / (max_i_profile_rabbet - min_i_profile_rabbet)
|
||||
profile_rabbet_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Высота в световом проеме" (чем меньше, тем лучше)
|
||||
if max_i_profile_rabbet == ini_max or min_i_profile_height == ini_min or i.iProfileHeight <= 12 \
|
||||
or i.iProfileHeight == max_i_profile_height or max_i_profile_height - min_i_profile_height < 0.01:
|
||||
profile_height_color = None
|
||||
else:
|
||||
color_ratio = (i.iProfileHeight - min_i_profile_height) / (max_i_profile_height - min_i_profile_height)
|
||||
profile_height_color = f"#{127 + int(color_ratio * 128):02x}ff{127 + int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Камер стеклопакета" (чем больше, тем лучше)
|
||||
if max_i_glazing_cameras_n == ini_max or min_i_profile_height == ini_min \
|
||||
or i.iGlazingCamerasN == min_i_glazing_cameras_n \
|
||||
or max_i_glazing_cameras_n - min_i_glazing_cameras_n < 0.001:
|
||||
glazing_cameras_n_color = None
|
||||
else:
|
||||
color_ratio = (i.iGlazingCamerasN - min_i_glazing_cameras_n) / (max_i_glazing_cameras_n
|
||||
- min_i_glazing_cameras_n)
|
||||
glazing_cameras_n_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Толщина стеклопакета" (чем больше, тем лучше)
|
||||
if max_i_glazing_thickness == ini_max or min_i_glazing_thickness == ini_min or i.iGlazingThickness <= 3 \
|
||||
or i.iGlazingThickness == min_i_glazing_thickness \
|
||||
or max_i_glazing_thickness - min_i_glazing_thickness < 0.001:
|
||||
glazing_thickness_color = None
|
||||
else:
|
||||
color_ratio = (i.iGlazingThickness - min_i_glazing_thickness) / (max_i_glazing_thickness
|
||||
- min_i_glazing_thickness)
|
||||
glazing_thickness_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Сопротивление теплопередаче стеклопакета" (чем больше, тем лучше)
|
||||
if max_f_glazing_heat_transfer == ini_max or min_f_glazing_heat_transfer == ini_min \
|
||||
or i.fGlazingHeatTransfer <= 0.05 or i.fGlazingHeatTransfer == min_f_glazing_heat_transfer \
|
||||
or max_f_glazing_heat_transfer - min_f_glazing_heat_transfer < 0.001:
|
||||
glazing_heat_transfer_color = None
|
||||
else:
|
||||
color_ratio = (i.fGlazingHeatTransfer - min_f_glazing_heat_transfer) / (max_f_glazing_heat_transfer
|
||||
- min_f_glazing_heat_transfer)
|
||||
glazing_heat_transfer_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Коэффициент звукоизоляции стеклопакета" (чем больше, тем лучше)
|
||||
if max_f_glazing_soundproofing == ini_max or min_f_glazing_soundproofing == ini_min \
|
||||
or i.fGlazingSoundproofing <= 5 or i.fGlazingSoundproofing == min_f_glazing_heat_transfer \
|
||||
or max_f_glazing_soundproofing - min_f_glazing_soundproofing < 0.001:
|
||||
glazing_soundproofing_color = None
|
||||
else:
|
||||
color_ratio = (i.fGlazingSoundproofing - min_f_glazing_soundproofing) / (max_f_glazing_soundproofing
|
||||
- min_f_glazing_soundproofing)
|
||||
glazing_soundproofing_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Коэффициент светопропускания стеклопакета" (чем больше, тем лучше)
|
||||
if max_f_glazing_light_transmission == ini_max or min_f_glazing_light_transmission == ini_min \
|
||||
or i.fGlazingLightTransmission <= 5 or i.fGlazingLightTransmission == min_f_glazing_light_transmission \
|
||||
or max_f_glazing_light_transmission - min_f_glazing_light_transmission < 0.002:
|
||||
glazing_light_transmission_color = None
|
||||
else:
|
||||
color_ratio = (i.fGlazingLightTransmission
|
||||
- min_f_glazing_light_transmission) / (max_f_glazing_light_transmission
|
||||
- min_f_glazing_light_transmission)
|
||||
glazing_light_transmission_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
# построим массив "цветов" для рейтинга "Коэффициент солнцепропускания стеклопакета" (чем меньше, тем лучше)
|
||||
if max_f_glazing_passing_sun == ini_max or min_f_glazing_passing_sun == ini_min or i.fGlazingPassingSun <= 5 \
|
||||
or i.fGlazingPassingSun == max_f_glazing_passing_sun \
|
||||
or max_f_glazing_passing_sun - min_f_glazing_passing_sun < 0.0001:
|
||||
glazing_passing_sun_color = None
|
||||
else:
|
||||
color_ratio = (i.fGlazingPassingSun - min_f_glazing_passing_sun) / (max_f_glazing_passing_sun
|
||||
- min_f_glazing_passing_sun)
|
||||
glazing_passing_sun_color = f"#{127 + int(color_ratio * 128):02x}ff{127 + int(color_ratio * 128):02x}"
|
||||
########################################################################
|
||||
# построим массив цветов "звездочек" для рейтинга наборов
|
||||
|
||||
# Рейтинг НАБОРА — особая логика со "звёздочками"
|
||||
if i.fSetRating > RARING_SET_MAX:
|
||||
rating_set_n = RARING_SET_MAX
|
||||
rating_set_color = "#80ff80"
|
||||
elif i.fSetRating < RARING_SET_MIN + 0.05 or max_rating_set - min_rating_set < 0.001:
|
||||
elif i.fSetRating < RARING_SET_MIN + 0.05 or max_rating - min_rating < 0.001:
|
||||
rating_set_n = RARING_SET_MIN
|
||||
rating_set_color = ""
|
||||
else:
|
||||
try:
|
||||
rating_set_n = i.fSetRating * (RARING_SET_MAX - RARING_SET_MIN) / RARING_STAR
|
||||
color_ratio = (i.fSetRating - min_rating_set) / (max_rating_set - min_rating_set)
|
||||
rating_set_color = f"#{255 - int(color_ratio * 128):02x}ff{255 - int(color_ratio * 128):02x}"
|
||||
rating_set_color = _color_hi(i.fSetRating, min_rating, max_rating)
|
||||
except (ZeroDivisionError, TypeError):
|
||||
rating_set_color = None
|
||||
rating_set_n = RARING_SET_MIN
|
||||
# print RatingSet
|
||||
|
||||
list2_del = f",{to_compare},"
|
||||
dim.append({
|
||||
"MERCHANT": i.sMerchantName,
|
||||
"MERCHANT_ID": i.MERCHANT_ID,
|
||||
"IS_COMMERCIAL": i.bCommercial,
|
||||
"MERCHANT_T": pytils.translit.slugify(i.sMerchantName),
|
||||
'MERCHANT_URL': i.sMerchantMainURL,
|
||||
'MERCHANT_URL_SHOT': re.sub("(?:^http://|^https://|/$|www\.)", "", i.sMerchantMainURL),
|
||||
"SET_NAME": i.sSetName,
|
||||
"MERCHANT_LOGO": i.pMerchantLogo,
|
||||
"RATING_SET": get_rating_set_for_stars(i.fSetRating),
|
||||
"RATING_SET_N": rating_set_n,
|
||||
"RATING_SET_COLOR": rating_set_color,
|
||||
"PROFILE_ID": i.PROFILE_ID,
|
||||
"PROFILE_NAME": i.sProfileName,
|
||||
"PROFILE_NAME_T": pytils.translit.slugify(i.sProfileName),
|
||||
"PROFILE_MANUFACTURER": i.sProfileManufacturer,
|
||||
"PROFILE_MANUFACTURER_T": pytils.translit.slugify(i.sProfileManufacturer),
|
||||
"PROFILE_NUM_COLOR": i.sProfileColor,
|
||||
"PROFILE_NUM_CAMERAS": i.iProfileCameras, # Число камер рамы/створки
|
||||
"PROFILE_NUM_CAMERAS_COLOR": profile_num_cameras_color, # Число камер рамы/створки ЦВЕТА
|
||||
"PROFILE_THICKNESS": i.iProfileThickness, # Монтажная ширина профиля
|
||||
"PROFILE_THICKNESS_COLOR": profile_thickness_color, # Окраска Монтажная ширина профиля ЦВЕТА
|
||||
"PROFILE_GLAZING_THICKNESS": i.iProfileGlazingThickness, # Максимальная толщина стеклопакета
|
||||
"PROFILE_GLAZING_THICKNESS_COLOR": profile_glazing_thickness_color, # Макс-толщина стеклопакета ЦВЕТА
|
||||
"PROFILE_HEAT_TRANSFER": i.fProfileHeatTransf, # Сопротивление теплопередаче
|
||||
"PROFILE_HEAT_TRANSFER_COLOR": profile_heat_transf_color, # Сопротивление теплопередаче ЦВЕТА
|
||||
"PROFILE_NUM_SEALS": i.fProfileSeals, # Контуров уплотнения
|
||||
"PROFILE_NUM_SEALS_COLOR": profile_seals_color, # Контуров уплотнения ЦВЕТА
|
||||
"PROFILE_SEAL_DESCRIPTION": i.sProfileSealDescription,
|
||||
"PROFILE_SOUND_PROOFING": i.fProfileSoundproofing, # Коэффициент звукоизоляции
|
||||
"PROFILE_SOUND_PROOFING_COLOR": profile_soundproofing_color, # Коэффициент звукоизоляции ЦВЕТА
|
||||
"PROFILE_HEIGHT": i.iProfileHeight, # Высота в световом проеме
|
||||
"PROFILE_HEIGHT_COLOR": profile_height_color, # Высота в световом проеме ЦВЕТА
|
||||
"PROFILE_RABBET": i.iProfileRabbet, # Фальц
|
||||
"PROFILE_RABBET_COLOR": profile_rabbet_color, # Фальц ЦВЕТА
|
||||
"PROFILE_FILLET": i.sProfileFillet, # Штапик
|
||||
"PROFILE_REINFORCEMENT": i.sProfileReinforcement, # Армирование профиля
|
||||
"PROFILE_OTHER": i.sProfileOther,
|
||||
"SET_ID": i.id, # id-набора
|
||||
"SET_CLIMATE_CONTROL": i.sSetClimateControl, # климат контроль
|
||||
"SET_STILL": i.sSetSill, # Подоконник
|
||||
"SET_IMPLEMENTS_ALL": i.sSetImplementAll, # Фурнитура
|
||||
"SET_IMPLEMENTS_HANDLES": i.sSetImplementHandles, # Фурнитура: Ручки
|
||||
"SET_IMPLEMENTS_HINGES": i.sSetImplementHinges, # Фурнитура: Петли
|
||||
"SET_IMPLEMENTS_LATCH": i.sSetImplementLatch, # Фурнитура: механизма запирания (запор)
|
||||
"SET_IMPLEMENTS_LIMITER": i.sSetImplementLimiter, # Фурнитура: Ограничитель
|
||||
"SET_IMPLEMENTS_CATCH": i.sSetImplementCatch, # Фурнитура: Фиксаторы открывания
|
||||
"SET_PANES": i.sSetPanes, # Водоотлив
|
||||
"SET_SLOPE": i.sSetSlope, # Откос
|
||||
"SET_DELIVERY": i.sSetDelivery, # Доставка (условия
|
||||
"SET_DELIVERY_B": i.bSetDelivery, # Доставка (да/нет)
|
||||
"SET_UNINSTALL_INSTALL": i.sSetUninstallInstall, # Монтаж/демонтаж (условия)
|
||||
"SET_UNINSTALL_INSTALL_B": i.bSetUninstallInstall, # Монтаж/демонтаж (да/нет)
|
||||
"SET_OTHER_CONDITIONS": i.sSetOtherConditions, # Прочие условия
|
||||
"GLAZING_CAMERAS_NUM": i.iGlazingCamerasN, # Камер стеклопакета
|
||||
"GLAZING_CAMERAS_COLOR": glazing_cameras_n_color, # Камер стеклопакета ЦВЕТА
|
||||
"GLAZING_THICKNESS": i.iGlazingThickness, # Толщина стеклопакета
|
||||
"GLAZING_THICKNESS_COLOR": glazing_thickness_color, # Толщина стеклопакета
|
||||
"GLAZING_BRIEF_DESCRIPTION": re.sub(u",[\s\d]+мм", "", i.sGlazingBriefDescription), # Кратко о стеклопакете
|
||||
"GLAZING_MARK": i.sGlazingMark, # Схема, марка, маркировка, модель стеклопакета
|
||||
"GLAZING_MANUFACTURER": i.sGlazingManufacturer, # Производитель стеклопакета
|
||||
"GLAZING_HEAT_TRANSFER": i.fGlazingHeatTransfer, # Сопротивление теплопередаче стеклопакета Ro (м²×°C/Вт)
|
||||
"GLAZING_HEAT_TRANSFER_COLOR": glazing_heat_transfer_color, # Сопротивление теплопередаче стеклопакета ЦВЕТ
|
||||
"GLAZING_SOUNDPROOFING": i.fGlazingSoundproofing, # Коэффициент звукоизоляции стеклопакета
|
||||
"GLAZING_SOUNDPROOFING_COLOR": glazing_soundproofing_color, # Коэффициент звукоизоляции стеклопакета ЦВЕТА
|
||||
"GLAZING_LIGHT_TRANSMISSION": i.fGlazingLightTransmission, # Коэффициент светопропускания стеклопакета
|
||||
"GLAZING_LIGHT_TRANSMISSION_COLOR": glazing_light_transmission_color, # Коэффициент светопропускания ЦВЕТА
|
||||
"GLAZING_LIGHT_REFLECTION": i.sGlazingLightReflectance, # Коэффициент светоотражения внешний/внутренний
|
||||
"GLAZING_PASSING_SUN": i.fGlazingPassingSun, # Коэффициент солнцепропускания стеклопакета
|
||||
"GLAZING_PASSING_SUN_COLOR": glazing_passing_sun_color, # Коэффициент солнцепропускания ЦВЕТ
|
||||
"MERCHANT": i.sMerchantName,
|
||||
"MERCHANT_ID": i.MERCHANT_ID,
|
||||
"IS_COMMERCIAL": i.bCommercial,
|
||||
"MERCHANT_T": sanitize_slug(i.sMerchantName),
|
||||
"MERCHANT_URL": i.sMerchantMainURL,
|
||||
"MERCHANT_URL_SHOT": re.sub(r"(?:^https?://|/$|www\.)", "", i.sMerchantMainURL),
|
||||
"SET_NAME": i.sSetName,
|
||||
"MERCHANT_LOGO": i.pMerchantLogo,
|
||||
"RATING_SET": get_rating_set_for_stars(i.fSetRating),
|
||||
"RATING_SET_N": rating_set_n,
|
||||
"RATING_SET_COLOR": rating_set_color,
|
||||
"PROFILE_ID": i.PROFILE_ID,
|
||||
"PROFILE_NAME": i.sProfileName,
|
||||
"PROFILE_NAME_T": sanitize_slug(i.sProfileName),
|
||||
"PROFILE_MANUFACTURER": i.sProfileManufacturer,
|
||||
"PROFILE_MANUFACTURER_T": sanitize_slug(i.sProfileManufacturer),
|
||||
"PROFILE_NUM_COLOR": i.sProfileColor,
|
||||
"PROFILE_NUM_CAMERAS": i.iProfileCameras, # Число камер рамы/створки
|
||||
"PROFILE_NUM_CAMERAS_COLOR": _color_hi(profile_num_cameras, min_cameras, max_cameras, threshold=1),
|
||||
"PROFILE_THICKNESS": i.iProfileThickness, # Монтажная ширина профиля
|
||||
"PROFILE_THICKNESS_COLOR": _color_hi(i.iProfileThickness, min_thick, max_thick, threshold=10),
|
||||
"PROFILE_GLAZING_THICKNESS": i.iProfileGlazingThickness, # Макс. толщина стеклопакета
|
||||
"PROFILE_GLAZING_THICKNESS_COLOR": _color_hi(i.iProfileGlazingThickness, min_glaz_d, max_glaz_d, threshold=4),
|
||||
"PROFILE_HEAT_TRANSFER": i.fProfileHeatTransf, # Сопротивление теплопередаче
|
||||
"PROFILE_HEAT_TRANSFER_COLOR": _color_hi(i.fProfileHeatTransf, min_heat_p, max_heat_p),
|
||||
"PROFILE_NUM_SEALS": i.fProfileSeals, # Контуров уплотнения
|
||||
"PROFILE_NUM_SEALS_COLOR": _color_hi(i.fProfileSeals, min_seals, max_seals, threshold=0),
|
||||
"PROFILE_SEAL_DESCRIPTION": i.sProfileSealDescription,
|
||||
"PROFILE_SOUND_PROOFING": i.fProfileSoundproofing, # Коэффициент звукоизоляции профиля
|
||||
"PROFILE_SOUND_PROOFING_COLOR": _color_hi(i.fProfileSoundproofing, min_sound_p, max_sound_p),
|
||||
"PROFILE_HEIGHT": i.iProfileHeight, # Высота в световом проеме (меньше = лучше)
|
||||
"PROFILE_HEIGHT_COLOR": _color_lo(i.iProfileHeight, min_height, max_height, threshold=12),
|
||||
"PROFILE_RABBET": i.iProfileRabbet, # Фальц
|
||||
"PROFILE_RABBET_COLOR": _color_hi(i.iProfileRabbet, min_rabbet, max_rabbet, threshold=1),
|
||||
"PROFILE_FILLET": i.sProfileFillet, # Штапик
|
||||
"PROFILE_REINFORCEMENT": i.sProfileReinforcement, # Армирование профиля
|
||||
"PROFILE_OTHER": i.sProfileOther,
|
||||
"SET_ID": i.id,
|
||||
"SET_CLIMATE_CONTROL": i.sSetClimateControl,
|
||||
"SET_STILL": i.sSetSill,
|
||||
"SET_IMPLEMENTS_ALL": i.sSetImplementAll,
|
||||
"SET_IMPLEMENTS_HANDLES": i.sSetImplementHandles,
|
||||
"SET_IMPLEMENTS_HINGES": i.sSetImplementHinges,
|
||||
"SET_IMPLEMENTS_LATCH": i.sSetImplementLatch,
|
||||
"SET_IMPLEMENTS_LIMITER": i.sSetImplementLimiter,
|
||||
"SET_IMPLEMENTS_CATCH": i.sSetImplementCatch,
|
||||
"SET_PANES": i.sSetPanes,
|
||||
"SET_SLOPE": i.sSetSlope,
|
||||
"SET_DELIVERY": i.sSetDelivery,
|
||||
"SET_DELIVERY_B": i.bSetDelivery,
|
||||
"SET_UNINSTALL_INSTALL": i.sSetUninstallInstall,
|
||||
"SET_UNINSTALL_INSTALL_B": i.bSetUninstallInstall,
|
||||
"SET_OTHER_CONDITIONS": i.sSetOtherConditions,
|
||||
"GLAZING_CAMERAS_NUM": i.iGlazingCamerasN, # Камер стеклопакета
|
||||
"GLAZING_CAMERAS_COLOR": _color_hi(i.iGlazingCamerasN, min_gl_cam, max_gl_cam),
|
||||
"GLAZING_THICKNESS": i.iGlazingThickness, # Толщина стеклопакета
|
||||
"GLAZING_THICKNESS_COLOR": _color_hi(i.iGlazingThickness, min_gl_thick, max_gl_thick, threshold=3),
|
||||
"GLAZING_BRIEF_DESCRIPTION": re.sub(r",[\s\d]+мм", "", i.sGlazingBriefDescription),
|
||||
"GLAZING_MARK": i.sGlazingMark,
|
||||
"GLAZING_MANUFACTURER": i.sGlazingManufacturer,
|
||||
"GLAZING_HEAT_TRANSFER": i.fGlazingHeatTransfer, # Ro стеклопакета (м²×°C/Вт)
|
||||
"GLAZING_HEAT_TRANSFER_COLOR": _color_hi(i.fGlazingHeatTransfer, min_heat_g, max_heat_g, threshold=0.05),
|
||||
"GLAZING_SOUNDPROOFING": i.fGlazingSoundproofing, # Звукоизоляция стеклопакета
|
||||
"GLAZING_SOUNDPROOFING_COLOR": _color_hi(i.fGlazingSoundproofing, min_sound_g, max_sound_g, threshold=5),
|
||||
"GLAZING_LIGHT_TRANSMISSION": i.fGlazingLightTransmission,
|
||||
"GLAZING_LIGHT_TRANSMISSION_COLOR": _color_hi(i.fGlazingLightTransmission, min_light, max_light, threshold=5, epsilon=0.002),
|
||||
"GLAZING_LIGHT_REFLECTION": i.sGlazingLightReflectance,
|
||||
"GLAZING_PASSING_SUN": i.fGlazingPassingSun, # Солнцепропускание (меньше = лучше)
|
||||
"GLAZING_PASSING_SUN_COLOR": _color_lo(i.fGlazingPassingSun, min_sun, max_sun, threshold=5, epsilon=0.0001),
|
||||
"GLAZING_REFLECTION_AND_ABSORPTION": i.sGlazingReflectionAndAbsorptionOfHeat,
|
||||
# Коэффициент теплоотражения/теплопоглощения стеклопакета
|
||||
"GLAZING_TONING": i.sGlazingToning, # Тонирование стеклопакета
|
||||
"URL_W_DEL": list2_del.replace(f",{i.id},", ",")[1:-1] # Тонирование стеклопакета
|
||||
"GLAZING_TONING": i.sGlazingToning,
|
||||
"URL_W_DEL": list2_del.replace(f",{i.id},", ",")[1:-1],
|
||||
})
|
||||
to_template.update({'SET_LIST': dim,
|
||||
'LIST_MERCHANT': list_of_merchant_name,
|
||||
'LIST_PROFILE': list_of_profile_name,
|
||||
'LIST_GLAZING': list_of_glazing_brief_description})
|
||||
'LIST_GLAZING': list_of_glazing_brief})
|
||||
# Предложения для добавления в сравнения:
|
||||
if len(list_set_kit) < 7:
|
||||
try:
|
||||
q_set_kit = SetKit.objects.raw(
|
||||
f"SELECT "
|
||||
f" oknardia_setkit.id, oknardia_setkit.sSetName,"
|
||||
f" oknardia_setkit.dSetModify, oknardia_setkit.fSetRating,"
|
||||
f" oknardia_merchantbrand.sMerchantName,"
|
||||
f" MAX(oknardia_priceoffer.dOfferModify) AS dLastData,"
|
||||
f" TO_DAYS(NOW()) - TO_DAYS(MAX(oknardia_priceoffer.dOfferModify)) AS deltaData "
|
||||
f"FROM oknardia_ouruser"
|
||||
f" INNER JOIN oknardia_setkit"
|
||||
f" ON oknardia_ouruser.id = oknardia_setkit.kSet2User_id"
|
||||
f" INNER JOIN oknardia_merchantoffice"
|
||||
f" ON oknardia_merchantoffice.id = oknardia_ouruser.kMerchantOffice_id"
|
||||
f" INNER JOIN oknardia_merchantbrand"
|
||||
f" ON oknardia_merchantbrand.id = oknardia_merchantoffice.kMerchantName_id"
|
||||
f" INNER JOIN oknardia_priceoffer"
|
||||
f" ON oknardia_setkit.id = oknardia_priceoffer.kOffer2SetKit_id "
|
||||
f"WHERE oknardia_setkit.id NOT IN (%s) "
|
||||
f"GROUP BY oknardia_setkit.id,"
|
||||
f" oknardia_setkit.sSetName,"
|
||||
f" oknardia_merchantbrand.sMerchantName,"
|
||||
f" oknardia_setkit.fSetRating "
|
||||
f"ORDER BY dLastData DESC "
|
||||
f"LIMIT 25;" % to_compare)
|
||||
q_set_kit = (
|
||||
SetKit.objects
|
||||
.exclude(id__in=list_fin) # исключаем уже сравниваемые наборы
|
||||
.filter(priceoffer__isnull=False) # только наборы с ценовыми предложениями
|
||||
.annotate(
|
||||
dLastData=Max('priceoffer__dOfferModify'),
|
||||
sMerchantName=F('kSet2User__kMerchantOffice__kMerchantName__sMerchantName'),
|
||||
)
|
||||
.order_by('-dLastData')[:25]
|
||||
)
|
||||
dim = []
|
||||
for i in q_set_kit:
|
||||
# Вычисляем deltaData в Python (аналог TO_DAYS(NOW()) - TO_DAYS(MAX(dOfferModify)))
|
||||
i.deltaData = (
|
||||
(timezone.now().date() - i.dLastData.date()).days
|
||||
if i.dLastData else 999
|
||||
)
|
||||
if i.deltaData < 100:
|
||||
early_data = pytils.dt.distance_of_time_in_words(
|
||||
int(django.utils.dateformat.format(i.dLastData, 'U')), accuracy=2
|
||||
@@ -534,12 +402,9 @@ def compare_offers(request: HttpRequest, to_compare: str = "1,2") -> HttpRespons
|
||||
except SetKit.DoesNotExist:
|
||||
pass
|
||||
to_template.update({
|
||||
# получаем последние визиты клиента через куки
|
||||
'LAST_VISIT': get_last_user_visit_list(get_last_user_visit_cookies(request)[:3]),
|
||||
# получаем последние визиты всех посетителей из базы
|
||||
# id2log, log_visit = get_last_all_user_visit_list()
|
||||
'LOG_VISIT': get_last_all_user_visit_list(),
|
||||
'ticks': float(time.time() - time_start)
|
||||
'ticks': float(time.perf_counter() - time_start)
|
||||
})
|
||||
return render(request, "report/report_compare_set.html", to_template)
|
||||
|
||||
@@ -551,28 +416,23 @@ def show_rating_components(request: HttpRequest, win_set: str = "1") -> HttpResp
|
||||
:param win_set: str -- id оконного набора, для которого показать состав рейтинга
|
||||
:return: HttpResponse --
|
||||
"""
|
||||
time_start = time.time()
|
||||
to_template = {}
|
||||
time_start = time.perf_counter()
|
||||
to_template: dict[str, object] = {}
|
||||
try:
|
||||
win_set = int(win_set)
|
||||
except ValueError:
|
||||
win_set = 1
|
||||
q = SetKit.objects.raw(
|
||||
f"SELECT oknardia_pvcprofiles.fProfileRating, oknardia_glazing.fGlazingRating,"
|
||||
f" oknardia_setkit.fSetRating, oknardia_setkit.id, MAX(oknardia_priceoffer.dOfferModify) AS dPriceModify,"
|
||||
f" COUNT(oknardia_priceoffer.id) AS NumOffer, AVG(oknardia_priceoffer.fOfferRating) AS fOfferRatingAvg "
|
||||
f"FROM oknardia_setkit"
|
||||
f" INNER JOIN oknardia_glazing"
|
||||
f" ON oknardia_setkit.kSet2Glazing_id = oknardia_glazing.id"
|
||||
f" INNER JOIN oknardia_pvcprofiles"
|
||||
f" ON oknardia_setkit.kSet2PVCprofiles_id = oknardia_pvcprofiles.id"
|
||||
f" INNER JOIN oknardia_priceoffer"
|
||||
f" ON oknardia_priceoffer.kOffer2SetKit_id = oknardia_setkit.id "
|
||||
f"WHERE oknardia_setkit.id = {win_set} "
|
||||
f"GROUP BY oknardia_pvcprofiles.fProfileRating,"
|
||||
f" oknardia_glazing.fGlazingRating,"
|
||||
f" oknardia_setkit.fSetRating,"
|
||||
f" oknardia_setkit.id;")
|
||||
q = (
|
||||
SetKit.objects
|
||||
.filter(id=win_set)
|
||||
.annotate(
|
||||
dPriceModify=Max('priceoffer__dOfferModify'),
|
||||
NumOffer=Count('priceoffer__id'),
|
||||
fOfferRatingAvg=Avg('priceoffer__fOfferRating'),
|
||||
fProfileRating=F('kSet2PVCprofiles__fProfileRating'),
|
||||
fGlazingRating=F('kSet2Glazing__fGlazingRating'),
|
||||
)
|
||||
)
|
||||
raring_list = list(q)
|
||||
f_rating_service = raring_list[0].fSetRating - RARING_WEIGHT_PVC_PROFILE_IN_SET * normalize(
|
||||
raring_list[0].fProfileRating, val_max=RARING_PVC_PROFILE_MAX
|
||||
@@ -596,5 +456,5 @@ def show_rating_components(request: HttpRequest, win_set: str = "1") -> HttpResp
|
||||
"коммерческое предложение, коммерческих предложения,"
|
||||
" коммерческих предложений"),
|
||||
'TEST': win_set,
|
||||
'ticks': float(time.time() - time_start)})
|
||||
'ticks': float(time.perf_counter() - time_start)})
|
||||
return render(request, "report/show_rating_components.html", to_template)
|
||||
|
||||
@@ -3,10 +3,9 @@ from django.shortcuts import render, redirect
|
||||
from django.http import HttpRequest, HttpResponse
|
||||
from oknardia.models import PVCprofiles
|
||||
from oknardia.settings import *
|
||||
from web.add_func import normalize, get_rating_set_for_stars
|
||||
from web.add_func import normalize, get_rating_set_for_stars, sanitize_slug
|
||||
from time import time
|
||||
import json
|
||||
import pytils
|
||||
|
||||
|
||||
def ratings(request: HttpRequest) -> HttpResponse:
|
||||
@@ -35,7 +34,7 @@ def profiles_rating(request: HttpRequest) -> HttpResponse:
|
||||
keys = [RANK_PVCP_HEAT_TRANSFER_NAME, RANK_PVCP_SOUNDPROOFING_NAME, RANK_PVCP_SEALS_NAME,
|
||||
RANK_PVCP_HEIGHT_NAME, RANK_PVCP_G_THICKNESS_NAME, RANK_PVCP_THICKNESS_NAME,
|
||||
RANK_PVCP_RABBET_NAME, RANK_PVCP_CAMERAS_NUM_NAME, RANK_PVCP_CAMERAS_POPULARITY_NAME]
|
||||
to_template = {'KEYS': keys}
|
||||
to_template: dict[str, object] = {'KEYS': keys}
|
||||
for profile in q_pvc_profiles:
|
||||
try:
|
||||
received_json = json.loads(profile.sProfileDescription)
|
||||
@@ -73,9 +72,9 @@ def profiles_rating(request: HttpRequest) -> HttpResponse:
|
||||
"ID": profile.id,
|
||||
"R_REAL": rating_real,
|
||||
"BRAND": profile.sProfileManufacturer,
|
||||
"BRAND_URL": pytils.translit.slugify(profile.sProfileManufacturer),
|
||||
"BRAND_URL": sanitize_slug(profile.sProfileManufacturer),
|
||||
"NAME": profile.sProfileName,
|
||||
"NAME_URL": pytils.translit.slugify(profile.sProfileName),
|
||||
"NAME_URL": sanitize_slug(profile.sProfileName),
|
||||
"K_ARR": k_arr,
|
||||
"RATING_STAR": get_rating_set_for_stars(profile.fProfileRating),
|
||||
"RATING_N": profile.fProfileRating,
|
||||
|
||||
@@ -1,36 +1,6 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
from django.shortcuts import render, redirect
|
||||
from django.shortcuts import render
|
||||
from django.http import HttpRequest, HttpResponse
|
||||
from oknardia.models import PVCprofiles, Seria_Info, Win_MountDim, Building_Info, SetKit
|
||||
from datetime import datetime, timezone
|
||||
import django.utils.dateformat
|
||||
import django.utils.timezone
|
||||
from oknardia.settings import *
|
||||
import time
|
||||
import random
|
||||
import pytils
|
||||
|
||||
|
||||
# Главная страница для вызова служебных процедур.
|
||||
def service(request: HttpRequest) -> HttpResponse:
|
||||
""" Страница для вызова служебных процедур
|
||||
|
||||
:param request: HttpRequest
|
||||
:return: HttpResponse
|
||||
"""
|
||||
time_start = time.time()
|
||||
# проверка на аутентификацию
|
||||
print(request.user.is_authenticated)
|
||||
if not request.user.is_authenticated:
|
||||
return redirect("/service/not-denice")
|
||||
return render(request, "service/index.html", {'ticks': float(time.time()-time_start)})
|
||||
|
||||
|
||||
# страничка, на которую переадресует служебный интерфейс, если нет аутентификации.
|
||||
def not_denice(request):
|
||||
time_start = time.time()
|
||||
return render(request, "service/not_denice.html", {'ticks': float(time.time()-time_start)})
|
||||
|
||||
|
||||
def tmp(request: HttpRequest) -> HttpResponse:
|
||||
""" Страница для тестирования верстки текста в блоге
|
||||
@@ -38,187 +8,4 @@ def tmp(request: HttpRequest) -> HttpResponse:
|
||||
:param request:
|
||||
:return:
|
||||
"""
|
||||
t_start = time.time()
|
||||
return render(request, "service/tmp.html", {'TAU': float(time.time()-t_start)})
|
||||
|
||||
|
||||
SITEMAP_MAX_ITEM = 40000 # максимальное число URL-ов в sitemap.xml -- 50000
|
||||
SITEMAP_MAX_FILE_SIZE = 5242880 # максимальный размер файла sitemap.xml -- 10Mb (10485760 байт)
|
||||
SITEMAP_MAX_FILES_QTY = 998 # максимальный число вложенных sitemap.xml -- 1000
|
||||
|
||||
|
||||
def str_time() -> str:
|
||||
""" Возвращает текущее время в ISO 8601 со смещением от текущего часового пояса
|
||||
"""
|
||||
return django.utils.dateformat.format(django.utils.timezone.now(), 'c')
|
||||
|
||||
|
||||
def make_site_maps (request: HttpRequest) -> HttpResponse:
|
||||
"""Функция создания sitemap.xml ... периодически надо вызывать через crone
|
||||
|
||||
:param request: request
|
||||
:return: HttpResponse ( msg )
|
||||
"""
|
||||
msg = ""
|
||||
time_start = time.time()
|
||||
count_total_item = 0
|
||||
count_item_per_file = 0
|
||||
count_file = 0
|
||||
# форматирование даты-времени в ISO 8601 со смещением от текущего часового пояса
|
||||
# str_time = django.utils.dateformat.format(django.utils.timezone.now(), 'c') # форматирование даты в ISO 8601
|
||||
# ПОЛУЧАЕМ ВСЕ СТРАНИЧКИ С ЦЕНАМИ ДЛЯ ОДИНОЧНОГО ПРОЕМА
|
||||
q1 = Win_MountDim.objects.raw("SELECT"
|
||||
" oknardia_win_mountdim.iWinWidth,"
|
||||
" oknardia_win_mountdim.iWinHight,"
|
||||
" oknardia_win_mountdim.id,"
|
||||
" COUNT(oknardia_priceoffer.kOffer2MountDim_id) AS NumOffer,"
|
||||
" oknardia_win_mountdim.sFlapConfig "
|
||||
"FROM oknardia_priceoffer"
|
||||
" INNER JOIN oknardia_win_mountdim"
|
||||
" ON oknardia_priceoffer.kOffer2MountDim_id = oknardia_win_mountdim.id "
|
||||
"GROUP BY oknardia_win_mountdim.id,"
|
||||
" oknardia_win_mountdim.iWinWidth,"
|
||||
" oknardia_win_mountdim.iWinHight,"
|
||||
" oknardia_win_mountdim.sFlapConfig "
|
||||
"ORDER BY COUNT(oknardia_priceoffer.kOffer2MountDim_id);")
|
||||
for i in q1:
|
||||
msg += f" <url>\n" \
|
||||
f" <loc>https://oknardia.ru/tsena-odnogo-okna/{int(i.iWinWidth*10)}x{int(i.iWinHight*10)}mm/tip{i.id}</loc>\n"\
|
||||
f" <lastmod>{str_time()}</lastmod>\n <changefreq>weekly</changefreq>\n <priority>0.5</priority>\n" \
|
||||
f" </url>\n"
|
||||
count_total_item += 1
|
||||
# print "~~~> ", countTotalItem, " ::: /compare_offers/", Count
|
||||
count_item_per_file += 1
|
||||
if (count_item_per_file > SITEMAP_MAX_ITEM) or (len(msg) > SITEMAP_MAX_FILE_SIZE):
|
||||
# Файл sitemap.xml заполнен... нужно записать и продолжить записывать в следующем
|
||||
msg = f"<?xml version='1.0' encoding='UTF-8'?>" \
|
||||
f"<urlset xmlns='http://www.sitemaps.org/schemas/sitemap/0.9'>\n{msg}</urlset>"
|
||||
with open(f"{SITEMAP_ROOT}sitemap{count_file:04d}.xml", "w", encoding="utf-8") as f:
|
||||
f.write(msg)
|
||||
count_item_per_file = 0
|
||||
count_file += 1 # счетчик файлов
|
||||
if count_file > SITEMAP_MAX_FILES_QTY: # максимально число файлов SITEMAP_MAX_FILES_QTY
|
||||
break
|
||||
msg = "" # обнулить буфер для записи файла
|
||||
# ВСЕ СТРАНИЧКИ С ЦЕНОВЫМИ ПРЕДЛОЖЕНИЯМИ ПО АДРЕСАМ
|
||||
q1 = Building_Info.objects.raw(
|
||||
"SELECT DISTINCT oknardia_building_info.sAddress, oknardia_building_info.id as id,"
|
||||
" oknardia_apartment_type.id AS ap_id "
|
||||
"FROM oknardia_building_info"
|
||||
" INNER JOIN oknardia_seria_info"
|
||||
" ON oknardia_building_info.kSeria_Link_id = oknardia_seria_info.id"
|
||||
" INNER JOIN oknardia_apartment_type"
|
||||
" ON oknardia_apartment_type.kSeria_id = oknardia_seria_info.kRoot_id "
|
||||
"ORDER BY oknardia_building_info.id;")
|
||||
list_build = list(q1)
|
||||
random.shuffle(list_build) # перемешиваем случайным образом, чтобы поисковики видели изменения sitemap
|
||||
for i in list_build:
|
||||
msg += f" <url>\n <loc>https://oknardia.ru/{i.id}/{i.ap_id}/{pytils.translit.slugify(i.sAddress)}</loc>\n" \
|
||||
f" <lastmod>{str_time()}</lastmod>\n <changefreq>weekly</changefreq>\n <priority>0.5</priority>\n" \
|
||||
f" </url>\n"
|
||||
count_total_item += 1
|
||||
# print("===> ", count_total_item, " ::: ", i.id, '/', i.ap_id, '/', pytils.translit.slugify(i.sAddress), sep="")
|
||||
count_item_per_file += 1
|
||||
if (count_item_per_file > SITEMAP_MAX_ITEM) or (len(msg) > SITEMAP_MAX_FILE_SIZE):
|
||||
# Файл sitemap.xml заполнен... нужно записать и продолжить записывать в следующем
|
||||
msg = f"<?xml version='1.0' encoding='UTF-8'?>\n" \
|
||||
f"<urlset xmlns='http://www.sitemaps.org/schemas/sitemap/0.9'>\n{msg}</urlset>"
|
||||
with open(f"{SITEMAP_ROOT}sitemap{count_file:04d}.xml", "w", encoding="utf-8") as f:
|
||||
f.write(msg)
|
||||
count_item_per_file = 0
|
||||
count_file += 1 # счетчик файлов
|
||||
if count_file > SITEMAP_MAX_FILES_QTY: # максимально число файлов SITEMAP_MAX_FILES_QTY
|
||||
break
|
||||
msg = "" # обнулить буфер для записи файла
|
||||
|
||||
# ДОБАВЛЯЕМ В SITEMAP ВСЕ СТРАНИЧКИ СО СРВНЕНИЕМ НАБОРОВ
|
||||
dim_comp = compare()
|
||||
random.shuffle(dim_comp)
|
||||
for i in dim_comp:
|
||||
msg += f" <url>\n <loc>https://oknardia.ru/compare_offers/{i}</loc>\n <lastmod>{str_time()}</lastmod>\n" \
|
||||
f" <changefreq>weekly</changefreq>\n <priority>0.45</priority>\n </url>\n"
|
||||
count_total_item += 1
|
||||
count_item_per_file += 1
|
||||
if (count_item_per_file > SITEMAP_MAX_ITEM) or (len(msg) > SITEMAP_MAX_FILE_SIZE):
|
||||
# Файл sitemap.xml заполнен... нужно записать и продолжить записывать в следующем
|
||||
msg = f"<?xml version='1.0' encoding='UTF-8'?>\n" \
|
||||
f"<urlset xmlns='http://www.sitemaps.org/schemas/sitemap/0.9'>\n{msg}</urlset>"
|
||||
with open(f"{SITEMAP_ROOT}sitemap{count_file:04d}.xml", "w", encoding="utf-8") as f:
|
||||
f.write(msg)
|
||||
count_item_per_file = 0
|
||||
count_file += 1 # счетчик файлов
|
||||
msg = "" # обнулить буфер для записи файла
|
||||
if count_file > SITEMAP_MAX_FILES_QTY: # максимально число файлов SITEMAP_MAX_FILES_QTY
|
||||
break
|
||||
|
||||
# ЗАВЕРШАЕМ
|
||||
if count_file == 0:
|
||||
# Все ссылки уместились в один sitemap.xml... просто его записать
|
||||
with open(f"{SITEMAP_ROOT}sitemap.xml", "w", encoding="utf-8") as f:
|
||||
f.write(f"<?xml version='1.0' encoding='UTF-8'?>\n"
|
||||
f"<urlset xmlns='http://www.sitemaps.org/schemas/sitemap/0.9'>\n{msg}</urlset>")
|
||||
print(SITEMAP_ROOT)
|
||||
msg = f"Создан единственный sitemap.xml\nВсего ссылок: {count_total_item:06d}"
|
||||
else:
|
||||
# Файлов sitemap.xml много.
|
||||
# Создаем завершающий файл sitemap
|
||||
with open(f"{SITEMAP_ROOT}sitemap{count_file:04d}.xml", "w", encoding="utf-8") as f:
|
||||
f.write(f"<?xml version='1.0' encoding='UTF-8'?>\n<urlset "
|
||||
f"xmlns='http://www.sitemaps.org/schemas/sitemap/0.9'>\n{msg}</urlset>")
|
||||
# Создаём объединяющий sitemap.xml с перечислением всего множества sitemap-файлов...
|
||||
msg = "<?xml version='1.0' encoding='UTF-8'?>\n" \
|
||||
"<sitemapindex xmlns='http://www.sitemaps.org/schemas/sitemap/0.9'>\n"
|
||||
for i in range(0, count_file+1):
|
||||
msg += f" <sitemap>\n <loc>https://oknardia.ru/sitemap{i:04d}.xml</loc>\n" \
|
||||
f" <lastmod>{str_time()}</lastmod>\n </sitemap>\n"
|
||||
msg += u"</sitemapindex>"
|
||||
with open(f"{SITEMAP_ROOT}sitemap.xml", "w", encoding="utf-8") as f:
|
||||
f.write(msg)
|
||||
msg = f"Создан каскадный sitemap.xml\nВсего вложенных файлов: {count_file+1:04d}\n" \
|
||||
f"Всего ссылок: {count_total_item:08d}"
|
||||
print(msg)
|
||||
return HttpResponse(f"<pre>{msg}\n\nвремя выполнения: {float(time.time()-time_start)} сек.</pre>")
|
||||
|
||||
|
||||
def compare() -> list:
|
||||
""" Возвращает список сравнения из всех возможных вариантов сравнения оконных наборов (из доступных в базе)
|
||||
|
||||
:return: список сравнения
|
||||
"""
|
||||
q_set_kit = SetKit.objects.raw('SELECT oknardia_setkit.id, oknardia_setkit.sSetActive '
|
||||
'FROM oknardia_setkit '
|
||||
'WHERE oknardia_setkit.sSetActive = TRUE')
|
||||
count = 0
|
||||
dim_comp = []
|
||||
l_set_kit = list(q_set_kit)
|
||||
for i1 in l_set_kit:
|
||||
for i2 in l_set_kit:
|
||||
if i1.id >= i2.id:
|
||||
continue
|
||||
dim_comp.append(f"{i1.id},{i2.id}")
|
||||
count += 1
|
||||
for i3 in l_set_kit:
|
||||
if i2.id >= i3.id:
|
||||
continue
|
||||
dim_comp.append(f"{i1.id},{i2.id},{i3.id}")
|
||||
count += 1
|
||||
for i4 in l_set_kit:
|
||||
if i3.id >= i4.id:
|
||||
continue
|
||||
dim_comp.append(f"{i1.id},{i2.id},{i3.id},{i4.id}")
|
||||
count += 1
|
||||
for i5 in l_set_kit:
|
||||
if i4.id >= i5.id:
|
||||
continue
|
||||
dim_comp.append(f"{i1.id},{i2.id},{i3.id},{i4.id},{i5.id}")
|
||||
count += 1
|
||||
for i6 in l_set_kit:
|
||||
if i5.id >= i6.id:
|
||||
continue
|
||||
dim_comp.append(f"{i1.id},{i2.id},{i3.id},{i4.id},{i5.id},{i6.id}")
|
||||
count += 1
|
||||
# random.shuffle(dim_comp)
|
||||
# for i1 in dim_comp:
|
||||
# print(i1)
|
||||
# print(f"---------------{count}---------------")
|
||||
return dim_comp
|
||||
return render(request, "service/tmp.html")
|
||||
264
oknardia/web/test_prices.py
Normal file
264
oknardia/web/test_prices.py
Normal file
@@ -0,0 +1,264 @@
|
||||
from datetime import timedelta
|
||||
from decimal import Decimal
|
||||
from unittest.mock import patch
|
||||
|
||||
from django.contrib.auth.models import User
|
||||
from django.db import connection
|
||||
from django.http import HttpResponse
|
||||
from django.test import RequestFactory, TestCase
|
||||
from django.utils import timezone
|
||||
|
||||
from oknardia.models import (
|
||||
Apartment_Type,
|
||||
Glazing,
|
||||
MerchantBrand,
|
||||
MerchantOffice,
|
||||
MountDim2Apartment,
|
||||
OurUser,
|
||||
PVCprofiles,
|
||||
PriceOffer,
|
||||
Seria_Info,
|
||||
SetKit,
|
||||
)
|
||||
from web.prices import redirect_one_win_price_legacy, report_one_win_price, report_price_frame
|
||||
|
||||
|
||||
class ReportOneWinPriceTests(TestCase):
|
||||
"""Регрессионные тесты для ORM-версии report_one_win_price."""
|
||||
|
||||
def setUp(self) -> None:
|
||||
self.factory = RequestFactory()
|
||||
django_user = User.objects.create_user(username="price-tester", password="secret")
|
||||
self.our_user = OurUser.objects.create(kDjangoUser=django_user)
|
||||
|
||||
# Тестовая SQLite-схема в проекте может быть legacy-вариантом с flap_config вместо sFlapConfig.
|
||||
# Для тестов report_one_win_price явно добавляем sFlapConfig, чтобы код проверялся в целевом режиме.
|
||||
with connection.cursor() as cursor:
|
||||
cursor.execute("PRAGMA table_info(oknardia_win_mountdim)")
|
||||
existing_columns = {row[1] for row in cursor.fetchall()}
|
||||
if "sFlapConfig" not in existing_columns:
|
||||
cursor.execute("ALTER TABLE oknardia_win_mountdim ADD COLUMN sFlapConfig varchar(32)")
|
||||
# if "flap_config" in existing_columns:
|
||||
# cursor.execute(
|
||||
# "UPDATE oknardia_win_mountdim SET sFlapConfig = flap_config "
|
||||
# "WHERE sFlapConfig IS NULL"
|
||||
# )
|
||||
|
||||
self.brand = MerchantBrand.objects.create(
|
||||
sMerchantName="Оконный бренд",
|
||||
sMerchantMainURL="https://example.com",
|
||||
)
|
||||
self.office = MerchantOffice.objects.create(
|
||||
sOfficeName="Оконный бренд — офис",
|
||||
kMerchantName=self.brand,
|
||||
sOfficePhones="+7(495)123-45-67",
|
||||
sOfficeAddress="Москва, Тестовая улица, 1",
|
||||
sOfficeDiscountMetaFormula="{'discount': {'10000': 5}}",
|
||||
)
|
||||
self.our_user.kMerchantOffice = self.office
|
||||
self.our_user.save(update_fields=["kMerchantOffice"])
|
||||
|
||||
with connection.cursor() as cursor:
|
||||
insert_columns = [
|
||||
"iWinWidth",
|
||||
"iWinHight",
|
||||
"iWinDepth",
|
||||
"sFlapConfig",
|
||||
"sDescripion",
|
||||
"bIsDoor",
|
||||
"bIsNearDoor",
|
||||
"iWinLimit",
|
||||
"dMountXYZDataCreate",
|
||||
"dMountXYZModify",
|
||||
]
|
||||
insert_values = [
|
||||
Decimal("67.0"),
|
||||
Decimal("216.0"),
|
||||
Decimal("15.0"),
|
||||
"[>]",
|
||||
"Тестовый проём",
|
||||
0,
|
||||
0,
|
||||
Decimal("5.0"),
|
||||
]
|
||||
if "flap_config" in existing_columns:
|
||||
insert_columns.insert(3, "flap_config")
|
||||
insert_values.insert(3, "[>]")
|
||||
columns_sql = ", ".join(insert_columns)
|
||||
placeholders_sql = ", ".join(["?"] * len(insert_values)) + ", CURRENT_TIMESTAMP, CURRENT_TIMESTAMP"
|
||||
cursor.execute(
|
||||
f"INSERT INTO oknardia_win_mountdim ({columns_sql}) VALUES ({placeholders_sql})",
|
||||
insert_values,
|
||||
)
|
||||
self.window_id = cursor.lastrowid
|
||||
self.seria = Seria_Info.objects.create(sName="П-44")
|
||||
self.apartment = Apartment_Type.objects.create(
|
||||
sNameApartment="1-комнатная",
|
||||
kSeria=self.seria,
|
||||
)
|
||||
MountDim2Apartment.objects.create(
|
||||
kApartment=self.apartment,
|
||||
kMountDim_id=self.window_id,
|
||||
iQuantity=1,
|
||||
)
|
||||
|
||||
self.glazing = Glazing.objects.create(
|
||||
sGlazingName="Тестовый стеклопакет",
|
||||
sGlazingBriefDescription="Двухкамерный стеклопакет",
|
||||
sGlazingMark="4-10-4-10-4",
|
||||
sGlazingToning="нет",
|
||||
kGlazing2User=self.our_user,
|
||||
)
|
||||
self.profile = PVCprofiles.objects.create(
|
||||
sProfileName="Profile Test",
|
||||
sProfileBriefDescription="Профиль для теста",
|
||||
sProfileManufacturer="Test Manufacturer",
|
||||
sProfileSealDescription="чёрный",
|
||||
sProfileReinforcement="сталь",
|
||||
kProfile2User=self.our_user,
|
||||
fProfileRating=4.2,
|
||||
)
|
||||
self.set_kit = SetKit.objects.create(
|
||||
sSetName="Тестовый набор",
|
||||
kSet2User=self.our_user,
|
||||
kSet2PVCprofiles=self.profile,
|
||||
kSet2Glazing=self.glazing,
|
||||
sSetImplementAll="Фурнитура",
|
||||
sSetImplementHandles="Ручки",
|
||||
sSetImplementHinges="Петли",
|
||||
sSetImplementLatch="Запоры",
|
||||
sSetImplementLimiter="Ограничитель",
|
||||
sSetImplementCatch="Фиксатор",
|
||||
sSetSill="Подоконник",
|
||||
sSetSlope="Откос",
|
||||
sSetPanes="Отлив",
|
||||
sSetDelivery="Доставка",
|
||||
bSetDelivery=True,
|
||||
sSetUninstallInstall="Монтаж",
|
||||
bSetUninstallInstall=True,
|
||||
sSetOtherConditions="Прочие условия",
|
||||
sSetClimateControl="Климат",
|
||||
fSetRating=4.5,
|
||||
dSetCommercialUntil=timezone.now() + timedelta(days=30),
|
||||
)
|
||||
self.active_offer = PriceOffer.objects.create(
|
||||
kOffer2MountDim_id=self.window_id,
|
||||
kOfferFromUser=self.our_user,
|
||||
kOffer2SetKit=self.set_kit,
|
||||
sOfferFlapConfig="[>]",
|
||||
fOfferPrice=Decimal("12345.00"),
|
||||
sOfferActive=True,
|
||||
)
|
||||
self.archived_offer = PriceOffer.objects.create(
|
||||
kOffer2MountDim_id=self.window_id,
|
||||
kOfferFromUser=self.our_user,
|
||||
kOffer2SetKit=self.set_kit,
|
||||
sOfferFlapConfig="[<]",
|
||||
fOfferPrice=Decimal("11111.00"),
|
||||
sOfferActive=False,
|
||||
)
|
||||
|
||||
@patch("web.prices.get_flaps_for_mini_pictures", return_value="img/test-mini.png")
|
||||
@patch(
|
||||
"web.prices.get_flaps_for_big_pictures",
|
||||
return_value={
|
||||
"FLAP_DIM": [{
|
||||
"iWinWidth": Decimal("67.0"),
|
||||
"iWinHight": Decimal("216.0"),
|
||||
"iWinWidth_mm": 670,
|
||||
"iWinHight_mm": 2160,
|
||||
}],
|
||||
"WIN_DIM": [],
|
||||
},
|
||||
)
|
||||
def test_report_one_win_price_renders_expected_context(
|
||||
self,
|
||||
mocked_big_pictures,
|
||||
mocked_mini_pictures,
|
||||
):
|
||||
"""Вьюха должна собирать тот же ключевой контекст, но уже без raw SQL."""
|
||||
request = self.factory.get(
|
||||
f"/catalog/standard_opening/price-670x2160mm-tip{self.window_id}",
|
||||
)
|
||||
captured = {}
|
||||
|
||||
def fake_render(_request, template_name, context):
|
||||
captured["template_name"] = template_name
|
||||
captured["context"] = context
|
||||
return HttpResponse("ok")
|
||||
|
||||
with patch("web.prices.render", side_effect=fake_render):
|
||||
response = report_one_win_price(request, "670", "2160", str(self.window_id))
|
||||
|
||||
context = captured["context"]
|
||||
self.assertEqual(response.status_code, 200)
|
||||
self.assertEqual(captured["template_name"], "price/price_offers_for_one_window.html")
|
||||
self.assertEqual(context["WIN_ID"], self.window_id)
|
||||
self.assertEqual(context["MOUNT_DIM_PER_OFFER"], 1)
|
||||
self.assertEqual(context["NUM_ARCHIVE_OFFER"], 1)
|
||||
self.assertIn("2", context["NUM_TOTAL_OFFER_N_WORD"])
|
||||
self.assertEqual(len(context["LIST_FLAP_VARIATION"]), 1)
|
||||
self.assertEqual(context["LIST_FLAP_VARIATION"][0].sOfferFlapConfig, "[>]")
|
||||
self.assertTrue(context["LIST_FLAP_VARIATION"][0].STR_NUM.startswith("вариант"))
|
||||
self.assertEqual(context["LIST_FLAP_VARIATION"][0].IMG_MINI, "img/test-mini.png")
|
||||
self.assertEqual(len(context["SERIA_FOR_WIN"]), 1)
|
||||
self.assertEqual(context["SERIA_FOR_WIN"][0].sName, self.seria.sName)
|
||||
self.assertEqual(len(context["PRICE_FRAME"]), 1)
|
||||
self.assertEqual(context["PRICE_FRAME"][0]["SETS_NAME"], self.set_kit.sSetName)
|
||||
self.assertEqual(context["PRICE_FRAME"][0]["MERCHANT"], self.brand.sMerchantName)
|
||||
self.assertEqual(context["PRICE_FRAME"][0]["DIM"][0]["IMG_MINI"], "img/test-mini.png")
|
||||
self.assertIn("META_DATA_PUBLISH", context)
|
||||
self.assertTrue(mocked_big_pictures.called)
|
||||
self.assertTrue(mocked_mini_pictures.called)
|
||||
|
||||
def test_report_one_win_price_redirects_to_canonical_dimensions(self):
|
||||
"""Если SEO-размеры в URL неверные, вьюха должна редиректить на канонический URL."""
|
||||
request = self.factory.get(
|
||||
f"/catalog/standard_opening/price-999x999mm-tip{self.window_id}",
|
||||
)
|
||||
|
||||
response = report_one_win_price(request, "999", "999", str(self.window_id))
|
||||
|
||||
self.assertEqual(response.status_code, 301)
|
||||
self.assertEqual(
|
||||
response["Location"],
|
||||
f"/catalog/standard_opening/price-670x2160mm-tip{self.window_id}/",
|
||||
)
|
||||
|
||||
def test_legacy_one_win_url_redirects_to_canonical_url(self):
|
||||
"""Старый URL страницы одного окна должен отдавать 301 на новый канонический путь."""
|
||||
request = self.factory.get(
|
||||
f"/tsena-odnogo-okna/670x2160mm/tip{self.window_id}",
|
||||
)
|
||||
|
||||
response = redirect_one_win_price_legacy(request, "670", "2160", str(self.window_id))
|
||||
|
||||
self.assertEqual(response.status_code, 301)
|
||||
self.assertEqual(
|
||||
response["Location"],
|
||||
f"/catalog/standard_opening/price-670x2160mm-tip{self.window_id}/",
|
||||
)
|
||||
|
||||
def test_report_price_frame_for_apartment_keeps_template_contract(self):
|
||||
"""ORM-ветка для квартир должна сохранять ключи контекста для price_list*."""
|
||||
frame = report_price_frame(
|
||||
apartment_id=self.apartment.id,
|
||||
mount_dim_per_offer=1,
|
||||
address_longitude=0,
|
||||
address_latitude=0,
|
||||
frame_begin_n=0,
|
||||
brand_id=0,
|
||||
win_id=0,
|
||||
)
|
||||
|
||||
self.assertIn("META_DATA_PUBLISH", frame)
|
||||
self.assertIn("PRICE_FRAME", frame)
|
||||
self.assertIn("N", frame)
|
||||
self.assertEqual(len(frame["PRICE_FRAME"]), 1)
|
||||
offer = frame["PRICE_FRAME"][0]
|
||||
self.assertEqual(offer["SETS_ID"], self.set_kit.id)
|
||||
self.assertEqual(offer["MERCHANT"], self.brand.sMerchantName)
|
||||
self.assertEqual(offer["FIN_PRICE"], self.active_offer.fOfferPrice)
|
||||
self.assertEqual(len(offer["DIM"]), 1)
|
||||
self.assertEqual(offer["DIM"][0]["QUANTITY"], 1)
|
||||
|
||||
@@ -1,3 +1,257 @@
|
||||
from django.test import TestCase
|
||||
from datetime import timedelta
|
||||
from decimal import Decimal
|
||||
import json
|
||||
from unittest.mock import patch
|
||||
|
||||
from django.contrib.auth.models import User
|
||||
from django.db import connection
|
||||
from django.http import HttpResponse
|
||||
from django.test import RequestFactory, TestCase
|
||||
from django.utils import timezone
|
||||
|
||||
from oknardia.settings import CATALOG_RECORD_FOR_PROFILE_MODEL
|
||||
from web.catalog import catalog_profile_model
|
||||
from oknardia.models import (
|
||||
BlogPosts,
|
||||
Catalog2Profile,
|
||||
Glazing,
|
||||
MerchantBrand,
|
||||
MerchantOffice,
|
||||
OurUser,
|
||||
PVCprofiles,
|
||||
PriceOffer,
|
||||
SetKit,
|
||||
)
|
||||
|
||||
|
||||
class CatalogProfileViewTests(TestCase):
|
||||
"""Регрессионные тесты для вьюхи каталога профилей."""
|
||||
|
||||
def setUp(self) -> None:
|
||||
# Базовый пользователь нужен, потому что профиль ссылается на OurUser.
|
||||
django_user = User.objects.create_user(username="tester", password="secret")
|
||||
self.our_user = OurUser.objects.create(kDjangoUser=django_user)
|
||||
self.factory = RequestFactory()
|
||||
|
||||
def _get_context(self, response):
|
||||
"""Достаёт итоговый контекст из ответа тестового клиента."""
|
||||
context = response.context
|
||||
if isinstance(context, list):
|
||||
return context[-1]
|
||||
return context
|
||||
|
||||
def _create_profile(self, *, name: str, brief: str, manufacturer: str, days_ago: int) -> PVCprofiles:
|
||||
"""Создаёт профиль с нужными полями и фиксирует дату изменения вручную."""
|
||||
profile = PVCprofiles.objects.create(
|
||||
sProfileName=name,
|
||||
sProfileBriefDescription=brief,
|
||||
sProfileManufacturer=manufacturer,
|
||||
kProfile2User=self.our_user,
|
||||
fProfileRating=3.5,
|
||||
)
|
||||
# В модели стоит auto_now=True, поэтому после создания дату правим отдельным update.
|
||||
modified_at = timezone.now() - timedelta(days=days_ago)
|
||||
PVCprofiles.objects.filter(pk=profile.pk).update(dProfileModify=modified_at)
|
||||
profile.refresh_from_db()
|
||||
return profile
|
||||
|
||||
def _create_catalog_profile_model_fixture(self, *, manufacturer: str = "Альфа"):
|
||||
"""Собирает минимальный набор данных для карточки профиля."""
|
||||
profile = PVCprofiles.objects.create(
|
||||
sProfileName="Alpha Basic",
|
||||
sProfileBriefDescription="Альфа База",
|
||||
sProfileManufacturer=manufacturer,
|
||||
kProfile2User=self.our_user,
|
||||
fProfileRating=4.25,
|
||||
sProfileDescription=json.dumps({"html": "<p>Дополнительная информация о профиле.</p>"}),
|
||||
sProfileOther="Контур: 2; Цвет: Белый",
|
||||
)
|
||||
PVCprofiles.objects.filter(pk=profile.pk).update(dProfileModify=timezone.now() - timedelta(days=10))
|
||||
profile.refresh_from_db()
|
||||
|
||||
sibling = PVCprofiles.objects.create(
|
||||
sProfileName="Alpha Plus",
|
||||
sProfileBriefDescription="Альфа Плюс",
|
||||
sProfileManufacturer=manufacturer,
|
||||
kProfile2User=self.our_user,
|
||||
fProfileRating=3.75,
|
||||
)
|
||||
|
||||
brand = MerchantBrand.objects.create(
|
||||
sMerchantName="Окно-Мир",
|
||||
sMerchantMainURL="https://example.com",
|
||||
)
|
||||
office = MerchantOffice.objects.create(
|
||||
sOfficeName="Окно-Мир Москва",
|
||||
kMerchantName=brand,
|
||||
sOfficeEmails="info@example.com",
|
||||
sOfficePhones="+7(495)000-00-00",
|
||||
)
|
||||
self.our_user.kMerchantOffice = office
|
||||
self.our_user.save(update_fields=["kMerchantOffice"])
|
||||
|
||||
glazing = Glazing.objects.create(
|
||||
sGlazingName="Тёплый пакет",
|
||||
sGlazingBriefDescription="Теплый двухкамерный стеклопакет",
|
||||
kGlazing2User=self.our_user,
|
||||
)
|
||||
setkit = SetKit.objects.create(
|
||||
sSetName="Набор-Альфа",
|
||||
kSet2User=self.our_user,
|
||||
kSet2PVCprofiles=profile,
|
||||
kSet2Glazing=glazing,
|
||||
sSetDescription="Комплект для теста",
|
||||
sSetClimateControl="Климат",
|
||||
sSetSill="Подоконник",
|
||||
sSetImplementAll="Фурнитура",
|
||||
sSetImplementHandles="Ручки",
|
||||
sSetImplementHinges="Петли",
|
||||
sSetImplementLatch="Запоры",
|
||||
sSetImplementLimiter="Ограничитель",
|
||||
sSetImplementCatch="Фиксатор",
|
||||
sSetPanes="Водоотлив",
|
||||
sSetSlope="Откос",
|
||||
sSetDelivery="Доставка",
|
||||
bSetDelivery=True,
|
||||
sSetUninstallInstall="Монтаж",
|
||||
bSetUninstallInstall=True,
|
||||
sSetOtherConditions="Прочее",
|
||||
fSetRating=4.1,
|
||||
dSetCommercialUntil=timezone.now(),
|
||||
)
|
||||
# В текущей схеме таблицы поле открывания называется flap_config, а не sFlapConfig.
|
||||
win_flap_column = "flap_" + "config"
|
||||
with connection.cursor() as cursor:
|
||||
cursor.execute(
|
||||
f"INSERT INTO oknardia_win_mountdim "
|
||||
f"(iWinWidth, iWinHight, iWinDepth, {win_flap_column}, sDescripion, bIsDoor, bIsNearDoor, iWinLimit, dMountXYZDataCreate, dMountXYZModify) "
|
||||
f"VALUES (?, ?, ?, ?, ?, ?, ?, ?, CURRENT_TIMESTAMP, CURRENT_TIMESTAMP)",
|
||||
[Decimal("120.0"), Decimal("140.0"), Decimal("15.0"), "[>][<]", "Окно тестовое", 0, 0, Decimal("5.0")],
|
||||
)
|
||||
win_id = cursor.lastrowid
|
||||
PriceOffer.objects.create(
|
||||
kOffer2MountDim_id=win_id,
|
||||
kOfferFromUser=self.our_user,
|
||||
kOffer2SetKit=setkit,
|
||||
sOfferFlapConfig="[>][<]",
|
||||
fOfferPrice=Decimal("12345.00"),
|
||||
)
|
||||
|
||||
blog = BlogPosts.objects.create(
|
||||
sPostHeader="Описание профиля",
|
||||
kBlogAuthorUser=self.our_user,
|
||||
sPostContent="<p>Основной текст</p><cut><p>Скрыто</p>",
|
||||
sImgForBlogSocial="img/catalog-profile.jpg",
|
||||
bCatalog=True,
|
||||
iCatalogSort=1,
|
||||
dPostDataBegin=timezone.now(),
|
||||
)
|
||||
BlogPosts.objects.filter(pk=blog.pk).update(dPostDataModify=timezone.now() - timedelta(days=1))
|
||||
blog.refresh_from_db()
|
||||
Catalog2Profile.objects.create(
|
||||
kProfile=profile,
|
||||
kBlogCatalog=blog,
|
||||
sCatalogCardType=CATALOG_RECORD_FOR_PROFILE_MODEL,
|
||||
)
|
||||
|
||||
return profile, sibling, brand, blog
|
||||
|
||||
def test_catalog_profile_handles_empty_catalog(
|
||||
self,
|
||||
):
|
||||
"""Пустой каталог не должен падать и должен отдавать ожидаемый контекст."""
|
||||
with self.assertNumQueries(1):
|
||||
response = self.client.get("/catalog/profile/")
|
||||
|
||||
context = self._get_context(response)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
self.assertEqual(context["CATALOG_PROFILE_NUM"], "0 профилей")
|
||||
self.assertEqual(context["CATALOG_MANUFACT_NUM"], 0)
|
||||
self.assertEqual(context["CATALOG_PROFILE_MAN1_NAME2"], [])
|
||||
|
||||
|
||||
def test_catalog_profile_groups_and_sorts_profiles(
|
||||
self,
|
||||
):
|
||||
"""Каталог должен группировать профили по производителю и сохранять сортировку."""
|
||||
self._create_profile(name="Alpha Basic", brief="Альфа База", manufacturer="Альфа", days_ago=5)
|
||||
self._create_profile(name="Alpha Plus", brief="Альфа Плюс", manufacturer="Альфа", days_ago=2)
|
||||
self._create_profile(name="Beta Light", brief="Бета Лайт", manufacturer="Бета", days_ago=1)
|
||||
self._create_profile(name="Hidden", brief="Скрытый", manufacturer="", days_ago=7)
|
||||
|
||||
with self.assertNumQueries(1):
|
||||
response = self.client.get("/catalog/profile/")
|
||||
|
||||
context = self._get_context(response)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
|
||||
# Пустой производитель не должен превращаться в отдельную группу.
|
||||
groups = context["CATALOG_PROFILE_MAN1_NAME2"]
|
||||
self.assertEqual(len(groups), 2)
|
||||
self.assertEqual([group["PROF_MAN"] for group in groups], ["Альфа", "Бета"])
|
||||
|
||||
alpha_group = groups[0]
|
||||
self.assertEqual(alpha_group["PROF_MAN_T"], "alfa")
|
||||
self.assertEqual(
|
||||
[item["PROF_NAME"] for item in alpha_group["PROF_MAN_LIST"]],
|
||||
["Альфа База", "Альфа Плюс"],
|
||||
)
|
||||
self.assertEqual(
|
||||
[item["PROF_NAME_T"] for item in alpha_group["PROF_MAN_LIST"]],
|
||||
["alpha-basic", "alpha-plus"],
|
||||
)
|
||||
|
||||
beta_group = groups[1]
|
||||
self.assertEqual(beta_group["PROF_MAN_T"], "beta")
|
||||
self.assertEqual([item["PROF_NAME"] for item in beta_group["PROF_MAN_LIST"]], ["Бета Лайт"])
|
||||
|
||||
# Проверяем итоговые счетчики и структуру контекста.
|
||||
self.assertEqual(context["CATALOG_MANUFACT_NUM"], 2)
|
||||
self.assertEqual(context["CATALOG_PROFILE_NUM"], "4 профиля")
|
||||
|
||||
def test_catalog_profile_model_redirects_to_canonical_url(
|
||||
self,
|
||||
):
|
||||
"""При неверных slug страница должна отправлять на канонический URL."""
|
||||
profile = self._create_profile(name="Alpha Basic", brief="Альфа База", manufacturer="Альфа", days_ago=5)
|
||||
|
||||
request = self.factory.get(f"/catalog/profile/{profile.id}-wrong/{profile.id}-wrong/")
|
||||
response = catalog_profile_model(request, profile.id, "wrong", profile.id, "wrong")
|
||||
|
||||
self.assertEqual(response.status_code, 302)
|
||||
self.assertEqual(response["Location"], f"/catalog/profile/{profile.id}-alfa/{profile.id}-alpha-basic")
|
||||
|
||||
def test_catalog_profile_model_renders_related_data(
|
||||
self,
|
||||
):
|
||||
"""Карточка профиля должна собираться через ORM и отдавать все ключевые блоки."""
|
||||
profile, sibling, brand, blog = self._create_catalog_profile_model_fixture()
|
||||
request = self.factory.get(f"/catalog/profile/{profile.id}-alfa/{profile.id}-alpha-basic/")
|
||||
captured = {}
|
||||
|
||||
def fake_render(_request, template_name, context):
|
||||
captured["template_name"] = template_name
|
||||
captured["context"] = context
|
||||
return HttpResponse("ok")
|
||||
|
||||
with patch("web.catalog.render", side_effect=fake_render):
|
||||
with self.assertNumQueries(4):
|
||||
response = catalog_profile_model(request, profile.id, "alfa", profile.id, "alpha-basic")
|
||||
|
||||
context = captured["context"]
|
||||
self.assertEqual(response.status_code, 200)
|
||||
self.assertEqual(captured["template_name"], "catalog/catalog_of_profiles_model.html")
|
||||
self.assertEqual(context["CATALOG_MODEL"].id, profile.id)
|
||||
self.assertEqual(context["CATALOG_URL"], f"{profile.id}-alfa")
|
||||
self.assertEqual(context["CATALOG_URL2"], f"{profile.id}-alfa/{profile.id}-alpha-basic")
|
||||
self.assertEqual(len(context["MERCHANTS"]), 1)
|
||||
self.assertEqual(context["MERCHANTS"][0]["MERCHANT_NAME"], brand.sMerchantName)
|
||||
self.assertEqual(context["MERCHANTS"][0]["MERCHANT_OFFERS"], 1)
|
||||
self.assertEqual(len(context["PROFILES"]), 1)
|
||||
self.assertEqual(context["PROFILES"][0]["PROFILE_ID"], sibling.id)
|
||||
self.assertEqual(len(context["PROFILE_DETAIL"]), 1)
|
||||
self.assertEqual(context["PROFILE_DETAIL"][0].sPostContent, blog.sPostContent)
|
||||
self.assertEqual(context["IMG_FOR_BLOG"], blog.sImgForBlogSocial)
|
||||
self.assertEqual(context["PUB_DAT"].date(), blog.dPostDataModify.date())
|
||||
self.assertEqual(context["LIST_OTHER"], ["<b>Контур:</b>2", "<b>Цвет:</b>Белый"])
|
||||
|
||||
# Create your tests here.
|
||||
|
||||
@@ -38,7 +38,7 @@ def menu_login_logout(request: HttpRequest) -> HttpResponse:
|
||||
# В дальнейшем, в случае высоких нагрузок на сервис, возможна простая деградация
|
||||
# с помощью отключения этого блока. Также возможен перенос исполнения функционала
|
||||
# LOGIN-LOGOUT на отдельный сервер.
|
||||
to_template = {} # словарь, для передачи шаблону
|
||||
to_template: dict[str, object] = {} # словарь, для передачи шаблону
|
||||
template = "user_manager/login-logout.html" # шаблон для подгрузки GOOGLE CAPTCHA
|
||||
if request.user.is_authenticated:
|
||||
to_template.update({'LOGGED_USER': request.user.username})
|
||||
@@ -56,7 +56,7 @@ def confirm_email(request: HttpRequest, user_id: str = "1", hash_part_12: str =
|
||||
:return response: исходящий http-ответ
|
||||
"""
|
||||
time_start = time()
|
||||
to_template = {} # словарь, для передачи шаблону
|
||||
to_template: dict[str, object] = {} # словарь, для передачи шаблону
|
||||
to_template.update({'CONFIRM_OK': "NO"})
|
||||
template = "index.html" # шаблон, о том, что email не подтвержден
|
||||
try:
|
||||
@@ -100,7 +100,7 @@ def restore_password(request: HttpRequest, user_id: str = "1", hash_part_12: str
|
||||
:return response: исходящий http-ответ
|
||||
"""
|
||||
time_start = time()
|
||||
to_template = {} # словарь, для передачи шаблону
|
||||
to_template: dict[str, object] = {} # словарь, для передачи шаблону
|
||||
to_template.update({'CONFIRM_OK': "NO"})
|
||||
template = "index.html" # шаблон, о том, что email не подтвержден
|
||||
try:
|
||||
@@ -138,7 +138,7 @@ def change_password(request: HttpRequest) -> HttpResponse:
|
||||
if request.method != 'POST':
|
||||
return HttpResponseRedirect("/")
|
||||
try:
|
||||
to_template = {} # словарь, для передачи шаблону
|
||||
to_template: dict[str, object] = {} # словарь, для передачи шаблону
|
||||
to_template.update({'CONFIRM_OK': "NO"})
|
||||
template = "user_manager/popup_confirm_email_or_restore_password_bad.html" # шаблон, о том, что всякие ошибки
|
||||
try:
|
||||
@@ -189,7 +189,7 @@ def form_user_menu_processing(request: HttpRequest) -> HttpResponse:
|
||||
return HttpResponseRedirect("/")
|
||||
if request.POST['status'] == "":
|
||||
return HttpResponseRedirect("/")
|
||||
to_template = {} # словарь, для передачи шаблону
|
||||
to_template: dict[str, object] = {} # словарь, для передачи шаблону
|
||||
template = "user_manager/login-logout_after.html" # шаблон для подгрузки GOOGLE CAPTCHA
|
||||
|
||||
# БЛОК -- LOGOUT
|
||||
|
||||
@@ -2,46 +2,28 @@
|
||||
from django.shortcuts import render, redirect
|
||||
from django.http import HttpRequest, HttpResponse
|
||||
from django.core.mail import send_mail
|
||||
from django.db.models import ExpressionWrapper, FloatField, F, Count
|
||||
from django.db.models.functions import Abs
|
||||
from smtplib import SMTPException
|
||||
from oknardia.models import Seria_Info, Building_Info, Apartment_Type
|
||||
from web.add_func import get_yandex_geocode_by_address, get_geo_distance
|
||||
import json
|
||||
import datetime
|
||||
from web.add_func import get_yandex_geocode_by_address, get_geo_distance, sanitize_slug
|
||||
import time
|
||||
import pytils
|
||||
|
||||
# from django.core.context_processors import csrf
|
||||
|
||||
|
||||
def main_init(request: HttpRequest) -> HttpResponse:
|
||||
""" Главная страница (статичная, только с проверками куков)
|
||||
""" Главная страница (статичная, только с проверками кук)
|
||||
|
||||
:param request: входящий http-запрос
|
||||
:return response: исходящий http-ответ
|
||||
"""
|
||||
to_template = {} # словарь, для передачи шаблону
|
||||
to_template: dict[str, object] = {} # словарь, для передачи шаблону
|
||||
num_viz = 0 # как будто первый визит
|
||||
# проверяем куки числа визита
|
||||
if "NumVisit" in request.COOKIES:
|
||||
# стоят куки, и это не первый визит
|
||||
num_viz = request.COOKIES["NumVisit"] # читаем число визитов
|
||||
num_viz = int(num_viz) + 1 # увеличиваем порядковый номер визитов
|
||||
# ПРОВЕРЯЧЕМ КУКИ ПРОСМОТРЕ ЦЕНОВЫХ ПРЕДЛОЖЕНИЙ
|
||||
if "LastVisit" in request.COOKIES:
|
||||
# стоят куки
|
||||
last_visit = json.loads(request.COOKIES["LastVisit"])
|
||||
last_visit2 = []
|
||||
for i in last_visit:
|
||||
last_visit2.append({
|
||||
"Time": datetime.datetime.fromtimestamp(i["Time"]),
|
||||
"LastURL": i["LastURL"],
|
||||
"LastAddress": i["LastAddress"],
|
||||
"LastApart": i["LastApart"]
|
||||
})
|
||||
to_template.update({'LAST_VISIT': last_visit2[:3]})
|
||||
else:
|
||||
to_template.update({'LAST_VISIT': None})
|
||||
to_template.update({'META_DOCUMENT_STATE': u"Static"}) # Эта страничка статичная (в шаблон)
|
||||
to_template.update({'NV': num_viz})
|
||||
# to_template.update(csrf(request)) # токен, для метода POST и GET
|
||||
response = render(request, "index.html", to_template)
|
||||
@@ -55,7 +37,7 @@ def tariff(request: HttpRequest) -> HttpResponse:
|
||||
:param request: входящий http-запрос
|
||||
:return response: исходящий http-ответ
|
||||
"""
|
||||
to_template = {} # для передачи в шаблон
|
||||
to_template: dict[str, object] = {} # для передачи в шаблон
|
||||
if request.method == 'POST':
|
||||
# print request.POST
|
||||
if 'tariff' in request.POST and 'email_' in request.POST \
|
||||
@@ -105,6 +87,22 @@ def contact(request: HttpRequest) -> HttpResponse:
|
||||
return render(request, "contact.html", {})
|
||||
|
||||
|
||||
def _fmt(value: object, fmt: str = ".1f", threshold: float = 0, default: str = "Нет данных") -> str:
|
||||
"""Вспомогательная функция: форматирует числовое поле здания или возвращает заглушку.
|
||||
|
||||
:param value: значение поля модели (числовое)
|
||||
:param fmt: строка формата для f-string, например '.1f' или '.0f'
|
||||
:param threshold: значения < threshold считаются «нет данных» (обычно 0 или -1)
|
||||
:param default: строка-заглушка при отсутствии данных
|
||||
"""
|
||||
try:
|
||||
if float(value) < threshold:
|
||||
return default
|
||||
return f"{value:{fmt}}"
|
||||
except (TypeError, ValueError):
|
||||
return default
|
||||
|
||||
|
||||
def get_address(request: HttpRequest) -> HttpResponse:
|
||||
""" Вызывается после ввода пользователем адреса. Получает строку с адресом методом POST
|
||||
|
||||
@@ -113,17 +111,17 @@ def get_address(request: HttpRequest) -> HttpResponse:
|
||||
:param request: request
|
||||
:return: response
|
||||
░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░"""
|
||||
time_start = time.time()
|
||||
time_start = time.perf_counter()
|
||||
if request.method != 'POST':
|
||||
return redirect("/")
|
||||
if 'address' not in request.POST:
|
||||
return redirect("/")
|
||||
addr = request.POST['address']
|
||||
to_template = {}
|
||||
to_template: dict[str, object] = {}
|
||||
try:
|
||||
q = Building_Info.objects.get(sAddress=addr)
|
||||
# Если QuerySet не содержит GeoCode (такое бывает, что в Яндекс-Картах не было каких-то данных),
|
||||
# то пробуем получить GeoCode повторно (вдруг, у Яндекс-Карт расширилась база адресов)
|
||||
# то пробуем получить GeoCode повторно (вдруг у Яндекс-Карт расширилась база адресов)
|
||||
if int(q.fGeoCode_Longitude) != 0 and int(q.fGeoCode_Latitude != 0):
|
||||
# print("координаты не ноль")
|
||||
to_template.update({'LATITUDE': str(q.fGeoCode_Latitude).replace(",", "."),
|
||||
@@ -141,112 +139,59 @@ def get_address(request: HttpRequest) -> HttpResponse:
|
||||
# print(geocode)
|
||||
to_template.update({'LATITUDE': str(geocode[0]).replace(",", ".")})
|
||||
to_template.update({'LONGITUDE': str(geocode[1]).replace(",", ".")})
|
||||
q = Building_Info.objects.raw(
|
||||
f"SELECT oknardia_building_info.*, "
|
||||
f"ABS({geocode[0]} - oknardia_building_info.fGeoCode_Latitude) + "
|
||||
f"ABS({geocode[1]} - oknardia_building_info.fGeoCode_Longitude) AS R2 "
|
||||
f"FROM oknardia_building_info "
|
||||
f"ORDER BY R2 "
|
||||
f"LIMIT 1;")[0]
|
||||
if q.R2 > 0.67: # Если расстояние между точками больше 670 метров, то не показываем результат
|
||||
to_template.update({'ticks': float(time.time()-time_start)})
|
||||
# Ищем ближайшее здание по манхэттенскому расстоянию (lat/lon в градусах, ~0.01 ≈ 1 км)
|
||||
q = (Building_Info.objects
|
||||
.annotate(
|
||||
R2=ExpressionWrapper(
|
||||
Abs(float(geocode[0]) - F('fGeoCode_Latitude'))
|
||||
+ Abs(float(geocode[1]) - F('fGeoCode_Longitude')),
|
||||
output_field=FloatField()
|
||||
)
|
||||
)
|
||||
.order_by('R2')
|
||||
.first())
|
||||
if q is None or q.R2 > 0.67: # Если расстояние > ~670 метров или ничего нет — не показываем
|
||||
to_template.update({'ticks': float(time.perf_counter()-time_start)})
|
||||
to_template.update({'addr': addr})
|
||||
return render(request, "popup/popup_incorrect_address.html", to_template)
|
||||
addr = q.sAddress
|
||||
# print("addr", addr)
|
||||
to_template.update({'ADDRESS_ID': q.id,
|
||||
'SERIA': q.sSerias_Project})
|
||||
if q.fTotal_Area < 0:
|
||||
to_template.update({'TOTAL_AREA': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'TOTAL_AREA': f"{q.fTotal_Area: .1f}"})
|
||||
to_template.update({'CADASTRE_NUM': q.sCadastre_Num_Area})
|
||||
if q.fLand_Area < 0:
|
||||
to_template.update({'LAND': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'LAND': f"{q.fLand_Area: .1f}"})
|
||||
to_template.update({'INVENTORY_NUM': q.sInventory_Num})
|
||||
if q.iNum_Apartments < 0:
|
||||
to_template.update({'NUM_APARTMENTS': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'NUM_APARTMENTS': q.iNum_Apartments})
|
||||
to_template.update({'TYPE_BUILDING': q.sType})
|
||||
if q.iNum_Apartments < 0:
|
||||
to_template.update({'STOREYS': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'STOREYS': q.iStoreys})
|
||||
if q.fCommon_Area < 0:
|
||||
to_template.update({'COMMON_AREA': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'COMMON_AREA': f"{q.fCommon_Area: .1f}"})
|
||||
to_template.update({'ENERGY_EFFICIENCY': q.sEnergy_Efficiency})
|
||||
if q.iEntrances_Porchs < 0:
|
||||
to_template.update({'NUM_ENTERANCES': "Нет"})
|
||||
else:
|
||||
to_template.update({'NUM_ENTERANCES': q.iEntrances_Porchs})
|
||||
if q.fUninhabited_Area < 0:
|
||||
to_template.update({'UNINHABITED_AREA': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'UNINHABITED_AREA': f"{q.fUninhabited_Area: .1f}"})
|
||||
if q.sManagement_Co == u"N/A":
|
||||
to_template.update({'MANAGEMENT_CO': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'MANAGEMENT_CO': q.sManagement_Co})
|
||||
if q.iElevators < 0:
|
||||
to_template.update({'NUM_ELEVATORS': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'NUM_ELEVATORS': q.iElevators})
|
||||
if q.fResidential_Area < 0:
|
||||
to_template.update({'RESIDENTIAL_AREA': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'RESIDENTIAL_AREA': f"{q.fResidential_Area: .1f}"})
|
||||
if q.iNum_Residents < 0:
|
||||
to_template.update({'NUM_RESIDENTS': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'NUM_RESIDENTS': q.iNum_Residents})
|
||||
if q.fPrivate_Area < 0:
|
||||
to_template.update({'PRIVATE_AREA': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'PRIVATE_AREA': f"{q.fPrivate_Area:.1f}"})
|
||||
if q.iNum_Accounts < 0:
|
||||
to_template.update({'NUM_ACCOUNTS': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'NUM_ACCOUNTS': q.iNum_Accounts})
|
||||
if q.iCommissioning_year == "N/A":
|
||||
to_template.update({'COMMISSIONING_YEAR': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'COMMISSIONING_YEAR': q.iCommissioning_year})
|
||||
if q.fGovernment_Area < 0:
|
||||
to_template.update({'GOVERNMENT_AREA': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'GOVERNMENT_AREA': f"{q.fGovernment_Area: .1f}"})
|
||||
if q.fCondition_House < 0:
|
||||
to_template.update({'CONDITION_HOUSE': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'CONDITION_HOUSE': f"{q.fCondition_House: .0f}%"})
|
||||
if q.fCondition_Foundation < 0:
|
||||
to_template.update({'CONDITION_FOUNDATION': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'CONDITION_FOUNDATION': f"{q.fCondition_Foundation: .0f}%"})
|
||||
if q.fCondition_Walls < 0:
|
||||
to_template.update({'CONDITION_WALL': u"Нет данных"})
|
||||
else:
|
||||
to_template.update({'CONDITION_WALL': f"{q.fCondition_Walls: .0f}%"})
|
||||
if q.fCondition_Overlap < 0:
|
||||
to_template.update({'CONDITION_OVERLAP': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'CONDITION_OVERLAP': f"{q.fCondition_Overlap: .0f}%"})
|
||||
if q.fMunicipal_Area < 0:
|
||||
to_template.update({'MUNICIPAL_AREA': "Нет данных"})
|
||||
else:
|
||||
to_template.update({'MUNICIPAL_AREA': f"{q.fMunicipal_Area: .1f}"})
|
||||
to_template.update({'URL2REFOEMAGKH': q.sURL})
|
||||
to_template.update({
|
||||
'ADDRESS_ID': q.id,
|
||||
'SERIA': q.sSerias_Project,
|
||||
'TOTAL_AREA': _fmt(q.fTotal_Area),
|
||||
'CADASTRE_NUM': q.sCadastre_Num_Area,
|
||||
'LAND': _fmt(q.fLand_Area),
|
||||
'INVENTORY_NUM': q.sInventory_Num,
|
||||
'NUM_APARTMENTS': q.iNum_Apartments if q.iNum_Apartments >= 0 else "Нет данных",
|
||||
'TYPE_BUILDING': q.sType,
|
||||
'STOREYS': q.iStoreys if q.iNum_Apartments >= 0 else "Нет данных",
|
||||
'COMMON_AREA': _fmt(q.fCommon_Area),
|
||||
'ENERGY_EFFICIENCY': q.sEnergy_Efficiency,
|
||||
'NUM_ENTERANCES': q.iEntrances_Porchs if q.iEntrances_Porchs >= 0 else "Нет",
|
||||
'UNINHABITED_AREA': _fmt(q.fUninhabited_Area),
|
||||
'MANAGEMENT_CO': q.sManagement_Co if q.sManagement_Co != "N/A" else "Нет данных",
|
||||
'NUM_ELEVATORS': q.iElevators if q.iElevators >= 0 else "Нет данных",
|
||||
'RESIDENTIAL_AREA': _fmt(q.fResidential_Area),
|
||||
'NUM_RESIDENTS': q.iNum_Residents if q.iNum_Residents >= 0 else "Нет данных",
|
||||
'PRIVATE_AREA': _fmt(q.fPrivate_Area),
|
||||
'NUM_ACCOUNTS': q.iNum_Accounts if q.iNum_Accounts >= 0 else "Нет данных",
|
||||
'COMMISSIONING_YEAR': q.iCommissioning_year if q.iCommissioning_year != "N/A" else "Нет данных",
|
||||
'GOVERNMENT_AREA': _fmt(q.fGovernment_Area),
|
||||
'CONDITION_HOUSE': _fmt(q.fCondition_House, fmt=".0f", default="Нет данных") + "%" if q.fCondition_House >= 0 else "Нет данных",
|
||||
'CONDITION_FOUNDATION': _fmt(q.fCondition_Foundation, fmt=".0f", default="Нет данных") + "%" if q.fCondition_Foundation >= 0 else "Нет данных",
|
||||
'CONDITION_WALL': _fmt(q.fCondition_Walls, fmt=".0f", default="Нет данных") + "%" if q.fCondition_Walls >= 0 else "Нет данных",
|
||||
'CONDITION_OVERLAP': _fmt(q.fCondition_Overlap, fmt=".0f", default="Нет данных") + "%" if q.fCondition_Overlap >= 0 else "Нет данных",
|
||||
'MUNICIPAL_AREA': _fmt(q.fMunicipal_Area),
|
||||
'URL2REFOEMAGKH': q.sURL,
|
||||
})
|
||||
# Пробуем получить базовую серию дома. Для этого рекурсивно раскручиваем записи в таблице Seria_Info
|
||||
idd = q.kSeria_Link_id
|
||||
all_apartment_in_seria = False
|
||||
q1 = None # страховка: если у здания нет серии, q1 остаётся None
|
||||
while idd is not None:
|
||||
# рекурсивно движемся по дерву потомок→предок серий домов.
|
||||
q1 = Seria_Info.objects.get(id=idd)
|
||||
q1 = Seria_Info.objects.select_related('kRoot').get(id=idd)
|
||||
# получаем список типовых квартир для серии дома с id == idd
|
||||
all_apartment_in_seria = Apartment_Type.objects.filter(kSeria_id=idd).order_by("iSort")
|
||||
# проверяем есть-ли что-то в списке типовых квартир.
|
||||
@@ -258,39 +203,105 @@ def get_address(request: HttpRequest) -> HttpResponse:
|
||||
# проверяем, был ли получен список квартир
|
||||
if not bool(all_apartment_in_seria):
|
||||
# Если списка квартир нет, нужно получить список ближайших адресов, для которых есть цены.
|
||||
q = Building_Info.objects.raw(
|
||||
f"SELECT"
|
||||
f" oknardia_building_info.sAddress, oknardia_building_info.id,"
|
||||
f" oknardia_building_info.fGeoCode_Longitude, oknardia_building_info.fGeoCode_Latitude,"
|
||||
f" oknardia_seria_info.kRoot_id, oknardia_seria_info.sName,"
|
||||
f" COUNT(oknardia_apartment_type.sNameApartment) AS NumApart,"
|
||||
f" ABS({geocode[0]} - oknardia_building_info.fGeoCode_Latitude)"
|
||||
f" + ABS({geocode[1]} - oknardia_building_info.fGeoCode_Longitude) AS R2 "
|
||||
f"FROM oknardia_building_info"
|
||||
f" INNER JOIN oknardia_seria_info"
|
||||
f" ON oknardia_building_info.kSeria_Link_id = oknardia_seria_info.id"
|
||||
f" INNER JOIN oknardia_apartment_type"
|
||||
f" ON oknardia_seria_info.kRoot_id = oknardia_apartment_type.kSeria_id "
|
||||
f"WHERE oknardia_building_info.fGeoCode_Longitude <> 0.0"
|
||||
f" AND oknardia_building_info.fGeoCode_Latitude <> 0.0 "
|
||||
f"GROUP BY oknardia_seria_info.sName,"
|
||||
f" oknardia_seria_info.kRoot_id,"
|
||||
f" oknardia_building_info.id,"
|
||||
f" oknardia_building_info.sAddress,"
|
||||
f" oknardia_building_info.fGeoCode_Longitude,"
|
||||
f" oknardia_building_info.fGeoCode_Latitude "
|
||||
f"ORDER BY R2 "
|
||||
f"LIMIT 5;")
|
||||
q = list(q)
|
||||
# Ищем здания с ненулевыми координатами и у которых через серию есть типовые квартиры.
|
||||
q = list(
|
||||
Building_Info.objects
|
||||
.exclude(fGeoCode_Longitude=0.0)
|
||||
.exclude(fGeoCode_Latitude=0.0)
|
||||
.filter(kSeria_Link__kRoot__apartment_type__isnull=False)
|
||||
.select_related('kSeria_Link', 'kSeria_Link__kRoot')
|
||||
.annotate(
|
||||
R2=ExpressionWrapper(
|
||||
Abs(float(geocode[0]) - F('fGeoCode_Latitude'))
|
||||
+ Abs(float(geocode[1]) - F('fGeoCode_Longitude')),
|
||||
output_field=FloatField()
|
||||
),
|
||||
NumApart=Count('kSeria_Link__kRoot__apartment_type', distinct=True),
|
||||
sName=F('kSeria_Link__sName'),
|
||||
kRoot_id=F('kSeria_Link__kRoot_id'),
|
||||
)
|
||||
.order_by('R2')[:5]
|
||||
)
|
||||
for i in q:
|
||||
# Пересчитываем на реальное геодезическое расстояние (км)
|
||||
i.R2 = get_geo_distance(i.fGeoCode_Longitude, i.fGeoCode_Latitude, geocode[0], geocode[1])
|
||||
# print i.id, i.sAddress, i.sName, i.R2
|
||||
# сортируем список по R2 (дистанция от текущего адреса, до домов по которым данные известны)
|
||||
sorted(q, key=lambda item: item.R2)
|
||||
q = sorted(q, key=lambda item: item.R2) # NOTE: sorted() возвращает новый список
|
||||
to_template.update({'NEAR_KNOWN_ADDRESS': q})
|
||||
# print q
|
||||
to_template.update({'SERIA_BASE': q1.sName,
|
||||
'addr': addr,
|
||||
'addr_T': pytils.translit.slugify(addr),
|
||||
'ticks': float(time.time()-time_start)})
|
||||
# Определяем корневую серию для формирования канонического URL
|
||||
# Если у серии есть kRoot — берём его, иначе сама q1 является корневой
|
||||
seria_root = (q1.kRoot if (q1 and q1.kRoot_id) else q1)
|
||||
to_template.update({
|
||||
'SERIA_BASE': q1.sName if q1 else "",
|
||||
'BASE_SERIA_ID': seria_root.id if seria_root else "",
|
||||
'BASE_SERIA_LAT': sanitize_slug((seria_root.sName or "").strip()) if seria_root else "",
|
||||
'addr': addr,
|
||||
'addr_T': sanitize_slug(addr),
|
||||
'ticks': float(time.perf_counter() - time_start),
|
||||
})
|
||||
return render(request, "popup/popup_show_apartment_variants.html", to_template)
|
||||
|
||||
|
||||
# ============================================================================
|
||||
# ОБРАБОТЧИКИ ОШИБОК ДЛЯ ТЕСТИРОВАНИЯ В DEBUG РЕЖИМЕ
|
||||
# ============================================================================
|
||||
# Используется в urls.py при DEBUG=True для тестирования верстки страниц ошибок.
|
||||
# Позволяет визуально проверить, как выглядят страницы 400, 403, 404 и 500
|
||||
# без необходимости искусственно вызывать реальные ошибки.
|
||||
|
||||
|
||||
def handler400(request: HttpRequest, exception=None) -> HttpResponse:
|
||||
"""Отображает страницу ошибки 400 (Bad Request).
|
||||
|
||||
Используется только в DEBUG режиме для тестирования верстки.
|
||||
|
||||
:param request: входящий http-запрос
|
||||
:param exception: исключение (если есть)
|
||||
:return response: исходящий http-ответ с кодом 400
|
||||
"""
|
||||
response = render(request, "error/400.html", {})
|
||||
response.status_code = 400
|
||||
return response
|
||||
|
||||
|
||||
def handler403(request: HttpRequest, exception=None) -> HttpResponse:
|
||||
"""Отображает страницу ошибки 403 (Forbidden).
|
||||
|
||||
Используется только в DEBUG режиме для тестирования верстки.
|
||||
|
||||
:param request: входящий http-запрос
|
||||
:param exception: исключение (если есть)
|
||||
:return response: исходящий http-ответ с кодом 403
|
||||
"""
|
||||
response = render(request, "error/403.html", {})
|
||||
response.status_code = 403
|
||||
return response
|
||||
|
||||
|
||||
def handler404(request: HttpRequest, exception=None) -> HttpResponse:
|
||||
"""Отображает страницу ошибки 404 (Not Found).
|
||||
|
||||
Используется только в DEBUG режиме для тестирования верстки.
|
||||
|
||||
:param request: входящий http-запрос
|
||||
:param exception: исключение (если есть)
|
||||
:return response: исходящий http-ответ с кодом 404
|
||||
"""
|
||||
response = render(request, "error/404.html", {})
|
||||
response.status_code = 404
|
||||
return response
|
||||
|
||||
|
||||
def handler500(request: HttpRequest) -> HttpResponse:
|
||||
"""Отображает страницу ошибки 500 (Internal Server Error).
|
||||
|
||||
Используется только в DEBUG режиме для тестирования верстки.
|
||||
|
||||
:param request: входящий http-запрос
|
||||
:return response: исходящий http-ответ с кодом 500
|
||||
"""
|
||||
response = render(request, "error/500.html", {})
|
||||
response.status_code = 500
|
||||
return response
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user