feat(integrations): архитектура включения/выключения интеграций

- Удалена лишняя модель IntegrationConfig из system_settings
- Singleton-паттерн: одна запись на интеграцию с is_active тумблером
- Добавлено шифрование токенов (EncryptedCharField с Fernet AES-128)
- UI: тумблеры слева, форма настроек справа
- API endpoints: toggle, settings, form_data
- Модель Recommerce: store_url + api_token (x-auth-token)
- Модель WooCommerce: store_url + consumer_key/secret

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
2026-01-12 00:29:04 +03:00
parent 4629369823
commit 37394121e1
14 changed files with 804 additions and 200 deletions

View File

@@ -1,59 +1,36 @@
from django.db import models
from .base import MarketplaceIntegration
from ...fields import EncryptedCharField
class RecommerceIntegration(MarketplaceIntegration):
"""
Интеграция с Recommerce.
Recommerce - сервис для управления товарами на маркетплейсах.
Recommerce - платформа для управления интернет-магазином.
API документация: запросы отправляются на домен магазина с заголовком x-auth-token.
Обязательные настройки:
- store_url: URL магазина (домен для API запросов)
- api_token: токен авторизации (передаётся в заголовке x-auth-token)
"""
# API endpoint (может отличаться от store_url)
api_endpoint = models.URLField(
blank=True,
verbose_name="API Endpoint",
help_text="URL API Recommerce (если отличается от URL магазина)"
)
# API токен (основной метод авторизации)
api_token = models.CharField(
# API токен (x-auth-token) - ЗАШИФРОВАН
api_token = EncryptedCharField(
max_length=500,
blank=True,
verbose_name="API Токен",
help_text="Токен авторизации Recommerce API"
)
# ID магазина в системе Recommerce
merchant_id = models.CharField(
max_length=100,
blank=True,
verbose_name="ID магазина",
help_text="Идентификатор магазина в Recommerce"
)
# Синхронизация цен
sync_prices = models.BooleanField(
default=True,
verbose_name="Синхронизировать цены",
help_text="Обновлять цены на маркетплейсе"
)
# Синхронизация остатков
sync_stock = models.BooleanField(
default=True,
verbose_name="Синхронизировать остатки",
help_text="Обновлять остатки на маркетплейсе"
verbose_name="API Токен (x-auth-token)",
help_text="Токен авторизации из панели управления Recommerce"
)
class Meta:
verbose_name = "Recommerce"
verbose_name_plural = "Recommerce"
managed = False # Пока заготовка - без создания таблицы
def __str__(self):
return f"Recommerce: {self.name or self.merchant_id}"
return f"Recommerce: {self.name or self.store_url or 'не настроен'}"
@property
def is_configured(self) -> bool:
"""Recommerce требует api_token"""
return bool(self.api_token)
"""Recommerce требует store_url и api_token"""
return bool(self.store_url and self.api_token)