Merge branch 'main' into storefront-nuxt

This commit is contained in:
Egor Pavlovich Gorbunov 2025-10-07 15:53:29 +03:00
commit 02c402c6de
252 changed files with 25684 additions and 13015 deletions

View file

@ -7,7 +7,7 @@ from .models import Post, PostTag
@register(Post)
class PostAdmin(SummernoteModelAdminMixin, FieldsetsMixin, ActivationActionsMixin, ModelAdmin): # type: ignore [misc]
class PostAdmin(SummernoteModelAdminMixin, FieldsetsMixin, ActivationActionsMixin, ModelAdmin): # type: ignore [misc, type-arg]
list_display = ("title", "author", "slug", "created", "modified")
list_filter = ("author", "tags", "created", "modified")
search_fields = ("title", "content", "slug")
@ -34,7 +34,7 @@ class PostAdmin(SummernoteModelAdminMixin, FieldsetsMixin, ActivationActionsMixi
@register(PostTag)
class PostTagAdmin(ModelAdmin):
class PostTagAdmin(ModelAdmin): # type: ignore [misc, type-arg]
list_display = ("tag_name", "name")
search_fields = ("tag_name", "name")
ordering = ("tag_name",)

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,44 +25,44 @@ msgstr "عنوان المنشور"
msgid "title"
msgstr "العنوان"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "المنشور"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "المنشورات"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"ملفات تخفيض السعر غير مدعومة Yer - استخدم محتوى تخفيض السعر بدلاً من ذلك!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr "يجب توفير ملف ترميز أو محتوى ترميز مخفض - متنافيان"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "معرّف العلامة الداخلي لعلامة المنشور"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "اسم العلامة"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "اسم سهل الاستخدام لعلامة المنشور"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "اسم عرض العلامة"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "علامة المشاركة"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "علامات المشاركة"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,46 +25,46 @@ msgstr "Název příspěvku"
msgid "title"
msgstr "Název"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Příspěvek"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Příspěvky"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"Soubory Markdown nejsou podporovány - místo toho použijte obsah Markdown!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"musí být poskytnut soubor markdown nebo obsah markdown - vzájemně se "
"vylučují."
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "interní identifikátor tagu pro tag příspěvku"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Název štítku"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Uživatelsky přívětivý název pro značku příspěvku"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Zobrazení názvu štítku"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Označení příspěvku"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Štítky příspěvků"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,44 +25,44 @@ msgstr "Indlæggets titel"
msgid "title"
msgstr "Titel"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Indlæg"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Indlæg"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr "Markdown-filer understøttes ikke - brug markdown-indhold i stedet!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"en markdown-fil eller markdown-indhold skal leveres - gensidigt udelukkende"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "intern tag-identifikator for indlægs-tagget"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Tag-navn"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Brugervenligt navn til posttagget"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Navn på tag-visning"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Tag til indlæg"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Tags til indlæg"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,47 +25,47 @@ msgstr "Titel des Beitrags"
msgid "title"
msgstr "Titel"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Beitrag"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Beiträge"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"Markdown-Dateien werden nicht unterstützt - verwenden Sie stattdessen "
"Markdown-Inhalte!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"eine Markdown-Datei oder ein Markdown-Inhalt muss bereitgestellt werden - "
"beide schließen sich gegenseitig aus"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "interner Tag-Bezeichner für den Post-Tag"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Tag name"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Benutzerfreundlicher Name für das Post-Tag"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Tag-Anzeigename"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Tag eintragen"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Tags eintragen"

View file

@ -2,12 +2,12 @@
# Copyright (C) 2025 EGOR <FUREUNOIR> GORBUNOV
# This file is distributed under the same license as the EVIBES package.
# EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>, 2025.
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -29,44 +29,44 @@ msgstr "Post's title"
msgid "title"
msgstr "Title"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Post"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Posts"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr "Markdown files are not supported yer - use markdown content instead!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"a markdown file or markdown content must be provided - mutually exclusive"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "internal tag identifier for the post tag"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Tag name"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "User-friendly name for the post tag"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Tag display name"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Post tag"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Post tags"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,44 +25,44 @@ msgstr "Post's title"
msgid "title"
msgstr "Title"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Post"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Posts"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr "Markdown files are not supported yer - use markdown content instead!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"a markdown file or markdown content must be provided - mutually exclusive"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "internal tag identifier for the post tag"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Tag name"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "User-friendly name for the post tag"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Tag display name"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Post tag"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Post tags"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,46 +25,46 @@ msgstr "Título del mensaje"
msgid "title"
msgstr "Título"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Publicar en"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Puestos"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"No se admiten archivos Markdown - ¡utiliza contenido Markdown en su lugar!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"se debe proporcionar un archivo markdown o contenido markdown - mutuamente "
"excluyentes"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "identificador interno de la etiqueta post"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Nombre de la etiqueta"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Nombre fácil de usar para la etiqueta de la entrada"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Nombre de la etiqueta"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Etiqueta postal"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Etiquetas"

View file

@ -7,7 +7,7 @@ msgid ""
msgstr ""
"Project-Id-Version: PACKAGE VERSION\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
"Language-Team: LANGUAGE <LL@li.org>\n"
@ -28,43 +28,43 @@ msgstr ""
msgid "title"
msgstr ""
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr ""
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr ""
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr ""
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr ""
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr ""
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr ""
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr ""
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr ""

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,47 +25,47 @@ msgstr "Titre du message"
msgid "title"
msgstr "Titre"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Poste"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Postes"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"Les fichiers Markdown ne sont pas pris en charge - utilisez plutôt du "
"contenu Markdown !"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"un fichier markdown ou un contenu markdown doit être fourni - ils s'excluent "
"mutuellement"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "identifiant interne de la balise post"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Nom du jour"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Nom convivial pour la balise post"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Nom d'affichage de l'étiquette"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Tag de poste"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Tags de la poste"

View file

@ -3,7 +3,7 @@ msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,43 +25,43 @@ msgstr "כותרת הפוסט"
msgid "title"
msgstr "כותרת"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "פוסט"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "פוסטים"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr "קובצי Markdown אינם נתמכים עדיין - השתמש בתוכן Markdown במקום!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr "יש לספק קובץ markdown או תוכן markdown - באופן בלעדי"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "מזהה תגיות פנימי עבור תגיות הפוסט"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "שם היום"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "שם ידידותי למשתמש עבור תגיות הפוסט"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "שם תצוגה של התג"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "תגית פוסט"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "תגיות פוסט"

View file

@ -2,12 +2,12 @@
# Copyright (C) 2025 EGOR <FUREUNOIR> GORBUNOV
# This file is distributed under the same license as the EVIBES package.
# EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>, 2025.
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -28,43 +28,43 @@ msgstr ""
msgid "title"
msgstr ""
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr ""
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr ""
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr ""
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr ""
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr ""
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr ""
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr ""
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr ""

View file

@ -7,7 +7,7 @@ msgid ""
msgstr ""
"Project-Id-Version: PACKAGE VERSION\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
"Language-Team: LANGUAGE <LL@li.org>\n"
@ -28,43 +28,43 @@ msgstr ""
msgid "title"
msgstr ""
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr ""
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr ""
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr ""
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr ""
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr ""
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr ""
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr ""
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr ""

View file

@ -3,7 +3,7 @@ msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,47 +25,47 @@ msgstr "Judul postingan"
msgid "title"
msgstr "Judul"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Pos"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Posting"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"File penurunan harga tidak didukung - gunakan konten penurunan harga sebagai"
" gantinya!"
"File penurunan harga tidak didukung - gunakan konten penurunan harga sebagai "
"gantinya!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"file penurunan harga atau konten penurunan harga harus disediakan - tidak "
"boleh ada yang sama"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "pengidentifikasi tag internal untuk tag pos"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Nama tag"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Nama yang mudah digunakan untuk tag postingan"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Nama tampilan tag"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Tag pos"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Tag pos"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,45 +25,45 @@ msgstr "Titolo del post"
msgid "title"
msgstr "Titolo"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Posta"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Messaggi"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr "I file Markdown non sono supportati: usa invece i contenuti Markdown!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"deve essere fornito un file markdown o un contenuto markdown - si escludono "
"a vicenda"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "identificatore interno del tag post"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Nome del tag"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Nome intuitivo per il tag del post"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Nome del tag"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Post tag"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Tag dei post"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,47 +25,47 @@ msgstr "投稿タイトル"
msgid "title"
msgstr "タイトル"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "ポスト"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "投稿"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"マークダウン・ファイルはサポートされていません - 代わりにマークダウン・コンテ"
"ンツを使用してください!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"マークダウン・ファイルまたはマークダウン・コンテンツを提供しなければならな"
"い。"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "投稿タグの内部タグ識別子"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "タグ名"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "投稿タグのユーザーフレンドリーな名前"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "タグ表示名"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "投稿タグ"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "投稿タグ"

View file

@ -2,12 +2,12 @@
# Copyright (C) 2025 EGOR <FUREUNOIR> GORBUNOV
# This file is distributed under the same license as the EVIBES package.
# EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>, 2025.
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -28,43 +28,43 @@ msgstr ""
msgid "title"
msgstr ""
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr ""
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr ""
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr ""
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr ""
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr ""
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr ""
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr ""
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr ""

View file

@ -3,7 +3,7 @@ msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,43 +25,44 @@ msgstr "게시물 제목"
msgid "title"
msgstr "제목"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "게시물"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "게시물"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr "마크다운 파일은 지원되지 않습니다 예 - 대신 마크다운 콘텐츠를 사용하세요!"
msgstr ""
"마크다운 파일은 지원되지 않습니다 예 - 대신 마크다운 콘텐츠를 사용하세요!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr "마크다운 파일 또는 마크다운 콘텐츠가 제공되어야 합니다."
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "게시물 태그의 내부 태그 식별자"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "태그 이름"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "게시물 태그의 사용자 친화적인 이름"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "태그 표시 이름"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "게시물 태그"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "게시물 태그"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,47 +25,47 @@ msgstr "Titel van de post"
msgid "title"
msgstr "Titel"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Plaats"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Berichten"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"Markdown-bestanden worden niet ondersteund - gebruik in plaats daarvan "
"markdown-inhoud!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"er moet een markdown-bestand of markdown-inhoud worden geleverd - wederzijds "
"exclusief"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "interne tagidentifier voor de posttag"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Tag naam"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Gebruiksvriendelijke naam voor de posttag"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Tag weergavenaam"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Post tag"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Post tags"

View file

@ -3,7 +3,7 @@ msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,44 +25,44 @@ msgstr "Innleggets tittel"
msgid "title"
msgstr "Title"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Post"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Innlegg"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr "Markdown-filer støttes ikke - bruk markdown-innhold i stedet!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"en markdown-fil eller markdown-innhold må oppgis - gjensidig utelukkende"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "intern tagg-identifikator for innleggstaggen"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Tagg navn"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Brukervennlig navn for innleggstaggen"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Visningsnavn for taggen"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Post tag"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Tagger for innlegg"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,46 +25,46 @@ msgstr "Tytuł postu"
msgid "title"
msgstr "Tytuł"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Post"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Posty"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"Pliki Markdown nie są obsługiwane - zamiast tego użyj zawartości Markdown!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"należy dostarczyć plik markdown lub zawartość markdown - wzajemnie się "
"wykluczające"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "wewnętrzny identyfikator tagu posta"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Nazwa tagu"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Przyjazna dla użytkownika nazwa tagu posta"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Wyświetlana nazwa znacznika"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Tag posta"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Tagi postów"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,45 +25,45 @@ msgstr "Título da postagem"
msgid "title"
msgstr "Título"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Postar"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Publicações"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"Os arquivos markdown não são suportados - use conteúdo markdown em vez disso!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"um arquivo ou conteúdo de markdown deve ser fornecido - mutuamente exclusivo"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "identificador de tag interno para a tag de postagem"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Nome da etiqueta"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Nome de fácil utilização para a tag de postagem"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Nome de exibição da tag"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Etiqueta de postagem"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Tags de postagem"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,46 +25,46 @@ msgstr "Titlul postului"
msgid "title"
msgstr "Titlul"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Post"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Mesaje"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"Fișierele Markdown nu sunt acceptate - utilizați în schimb conținut Markdown!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"trebuie furnizat un fișier markdown sau conținut markdown - se exclud "
"reciproc"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "identificator intern de etichetă pentru eticheta postului"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Nume etichetă"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Nume ușor de utilizat pentru eticheta postului"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Nume afișare etichetă"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Etichetă post"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Etichete poștale"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,47 +25,47 @@ msgstr "Заголовок сообщения"
msgid "title"
msgstr "Название"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Пост"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Посты"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"Файлы в формате Markdown не поддерживаются - используйте вместо них "
"содержимое в формате Markdown!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"необходимо предоставить файл разметки или содержимое разметки - "
"взаимоисключающие варианты"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "внутренний идентификатор тега для тега post"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Название тега"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Удобное для пользователя название тега поста"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Отображаемое имя тега"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Тэг поста"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Тэги постов"

View file

@ -3,7 +3,7 @@ msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,45 +25,45 @@ msgstr "Inläggets titel"
msgid "title"
msgstr "Titel"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Post"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Inlägg"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr "Markdown-filer stöds inte - använd markdown-innehåll istället!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"en markdown-fil eller markdown-innehåll måste tillhandahållas - ömsesidigt "
"uteslutande"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "intern taggidentifierare för inläggstaggen"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Tagg namn"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Användarvänligt namn för inläggstaggen"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Taggens visningsnamn"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Post tagg"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Taggar för inlägg"

View file

@ -3,7 +3,7 @@ msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,45 +25,43 @@ msgstr "ชื่อโพสต์"
msgid "title"
msgstr "ชื่อเรื่อง"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "โพสต์"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "โพสต์"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr "ไฟล์มาร์กดาวน์ยังไม่รองรับในตอนนี้ - กรุณาใช้เนื้อหาแบบมาร์กดาวน์แทน!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"ไฟล์มาร์กดาวน์หรือเนื้อหาแบบมาร์กดาวน์ต้องได้รับการจัดเตรียมไว้ - "
"ไม่สามารถใช้ร่วมกันได้"
msgstr "ไฟล์มาร์กดาวน์หรือเนื้อหาแบบมาร์กดาวน์ต้องได้รับการจัดเตรียมไว้ - ไม่สามารถใช้ร่วมกันได้"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "ตัวระบุแท็กภายในสำหรับแท็กโพสต์"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "ชื่อวัน"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "ชื่อที่ใช้งานได้ง่ายสำหรับแท็กโพสต์"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "แสดงชื่อแท็ก"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "โพสต์แท็ก"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "ป้ายกำกับโพสต์"

View file

@ -3,7 +3,7 @@ msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,47 +25,46 @@ msgstr "Gönderinin başlığı"
msgid "title"
msgstr "Başlık"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Posta"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Mesajlar"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"Markdown dosyaları desteklenmiyor yer - bunun yerine markdown içeriği "
"kullanın!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"bir markdown dosyası veya markdown içeriği sağlanmalıdır - birbirini "
"dışlayan"
"bir markdown dosyası veya markdown içeriği sağlanmalıdır - birbirini dışlayan"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "gönderi etiketi için dahili etiket tanımlayıcısı"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Etiket adı"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Gönderi etiketi için kullanıcı dostu ad"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Etiket görünen adı"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Mesaj etiketi"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Gönderi etiketleri"

View file

@ -3,7 +3,7 @@ msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,46 +25,46 @@ msgstr "Tiêu đề bài đăng"
msgid "title"
msgstr "Tiêu đề"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "Bài đăng"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "Bài đăng"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr ""
"Tệp Markdown hiện chưa được hỗ trợ - hãy sử dụng nội dung Markdown thay thế!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr ""
"Phải cung cấp tệp Markdown hoặc nội dung Markdown - hai tùy chọn này là "
"tương phản nhau."
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "Mã định danh thẻ nội bộ cho thẻ bài viết"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "Tên ngày"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "Tên thân thiện với người dùng cho thẻ bài viết"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "Hiển thị tên thẻ"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "Thẻ bài viết"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "Thẻ bài viết"

View file

@ -1,9 +1,9 @@
#
#
msgid ""
msgstr ""
"Project-Id-Version: EVIBES 3.0.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-09-19 15:04+0300\n"
"POT-Creation-Date: 2025-10-07 15:47+0300\n"
"PO-Revision-Date: 2025-06-16 08:59+0100\n"
"Last-Translator: EGOR GORBUNOV <CONTACT@FUREUNOIR.COM>\n"
"Language-Team: LANGUAGE <CONTACT@FUREUNOIR.COM>\n"
@ -25,43 +25,43 @@ msgstr "帖子标题"
msgid "title"
msgstr "标题"
#: blog/models.py:83
#: blog/models.py:84
msgid "post"
msgstr "职位"
#: blog/models.py:84
#: blog/models.py:85
msgid "posts"
msgstr "职位"
#: blog/models.py:88
#: blog/models.py:89
msgid "markdown files are not supported yet - use markdown content instead"
msgstr "不支持 Markdown 文件,请使用 Markdown 内容!"
#: blog/models.py:90
#: blog/models.py:91
msgid ""
"a markdown file or markdown content must be provided - mutually exclusive"
msgstr "必须提供标记符文件或标记符内容 - 相互排斥"
#: blog/models.py:122
#: blog/models.py:123
msgid "internal tag identifier for the post tag"
msgstr "职位标签的内部标签标识符"
#: blog/models.py:123
#: blog/models.py:124
msgid "tag name"
msgstr "标签名称"
#: blog/models.py:127
#: blog/models.py:128
msgid "user-friendly name for the post tag"
msgstr "方便用户使用的帖子标签名称"
#: blog/models.py:128
#: blog/models.py:129
msgid "tag display name"
msgstr "标签显示名称"
#: blog/models.py:136
#: blog/models.py:137
msgid "post tag"
msgstr "职位标签"
#: blog/models.py:137
#: blog/models.py:138
msgid "post tags"
msgstr "帖子标签"

View file

@ -0,0 +1,158 @@
# Generated by Django 5.2 on 2025-10-07 12:46
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
("blog", "0005_post_content_fa_ir_post_content_he_il_and_more"),
]
operations = [
migrations.AddField(
model_name="post",
name="meta_description",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_ar_ar",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_cs_cz",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_da_dk",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_de_de",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_en_gb",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_en_us",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_es_es",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_fa_ir",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_fr_fr",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_he_il",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_hi_in",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_hr_hr",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_id_id",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_it_it",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_ja_jp",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_kk_kz",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_ko_kr",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_nl_nl",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_no_no",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_pl_pl",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_pt_br",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_ro_ro",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_ru_ru",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_sv_se",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_th_th",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_tr_tr",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_vi_vn",
field=models.CharField(blank=True, max_length=150, null=True),
),
migrations.AddField(
model_name="post",
name="meta_description_zh_hans",
field=models.CharField(blank=True, max_length=150, null=True),
),
]

View file

@ -75,6 +75,7 @@ class Post(NiceModel): # type: ignore [django-manager-missing]
file = FileField(upload_to="posts/", blank=True, null=True)
slug = AutoSlugField(populate_from="title", allow_unicode=True, unique=True, editable=False)
tags = ManyToManyField(to="blog.PostTag", blank=True, related_name="posts")
meta_description = CharField(max_length=150, blank=True, null=True)
def __str__(self):
return f"{self.title} | {self.author.first_name} {self.author.last_name}"

View file

@ -6,4 +6,4 @@ from blog.models import Post
@register(Post)
class PostOptions(TranslationOptions):
fields = ("title", "content")
fields = ("title", "content", "meta_description")

View file

@ -1,4 +1,5 @@
import uuid
from typing import Collection
from django.db.models import BooleanField, Model, UUIDField
from django.utils.translation import gettext_lazy as _
@ -22,9 +23,19 @@ class NiceModel(Model):
created = CreationDateTimeField(_("created"), help_text=_("when the object first appeared on the database"))
modified = ModificationDateTimeField(_("modified"), help_text=_("when the object was last modified"))
def save(self, **kwargs):
self.update_modified = kwargs.pop("update_modified", getattr(self, "update_modified", True))
super().save(**kwargs)
def save(
self,
*,
force_insert: bool = False,
force_update: bool = False,
using: str | None = None,
update_fields: Collection | None = None,
update_modified: bool = True,
) -> None:
self.update_modified = update_modified
return super().save(
force_insert=force_insert, force_update=force_update, using=using, update_fields=update_fields
)
class Meta:
abstract = True

View file

@ -1,5 +1,5 @@
from core.models import CustomerRelationshipManagementProvider
def any_crm_integrations():
def any_crm_integrations() -> bool:
return CustomerRelationshipManagementProvider.objects.exists()

View file

@ -89,15 +89,12 @@ class AmoCRM:
if type(order.attributes) is not dict:
raise ValueError("order.attributes must be a dict")
business_identificator = (
order.attributes.get("business_identificator")
or order.attributes.get("businessIdentificator")
or order.user.attributes.get("business_identificator")
or order.user.attributes.get("businessIdentificator")
or ""
)
if order.user:
if type(order.user.attributes) is not dict:
order.user.attributes = {}
order.user.save()
if not business_identificator:
if not order.business_identificator:
return (
order.user.get_full_name()
if order.user
@ -109,7 +106,7 @@ class AmoCRM:
)
try:
r = requests.get(
f"https://api-fns.ru/api/egr?req={business_identificator}&key={self.fns_api_key}", timeout=15
f"https://api-fns.ru/api/egr?req={order.business_identificator}&key={self.fns_api_key}", timeout=15
)
r.raise_for_status()
body = r.json()
@ -122,9 +119,9 @@ class AmoCRM:
ip = body.get("ИП")
if ul and not ip:
return f"{ul.get('НаимСокрЮЛ')} | {business_identificator}"
return f"{ul.get('НаимСокрЮЛ')} | {order.business_identificator}"
if ip and not ul:
return f"ИП {ip.get('ФИОПолн')} | {business_identificator}"
return f"ИП {ip.get('ФИОПолн')} | {order.business_identificator}"
return ""

View file

@ -1,6 +1,9 @@
import re
from typing import Any
from blib2to3.pgen2.parse import Callable
from django.conf import settings
from django.db.models import QuerySet
from django.http import Http404
from django.utils.text import slugify
from django.utils.translation import gettext_lazy as _
@ -182,7 +185,7 @@ def process_query(
minimum_should_match=1,
)
def build_search(idxs, size):
def build_search(idxs: list[str], size: int) -> Search:
return (
Search(index=idxs)
.query(query_base)
@ -221,9 +224,9 @@ def process_query(
search_products = build_search(["products"], size=44)
resp_products = search_products.execute()
results: dict = {"products": [], "categories": [], "brands": [], "posts": []}
uuids_by_index: dict[str, list] = {"products": [], "categories": [], "brands": []}
hit_cache: list = []
results: dict[str, list[dict[str, Any]]] = {"products": [], "categories": [], "brands": [], "posts": []}
uuids_by_index: dict[str, list[dict[str, Any]]] = {"products": [], "categories": [], "brands": []}
hit_cache: list[Any] = []
for h in (
list(resp_cats.hits[:12] if resp_cats else [])
@ -325,10 +328,10 @@ def _lang_analyzer(lang_code: str) -> str:
class ActiveOnlyMixin:
def get_queryset(self):
def get_queryset(self) -> QuerySet[Any]:
return super().get_queryset().filter(is_active=True)
def should_index_object(self, obj):
def should_index_object(self, obj) -> bool:
return getattr(obj, "is_active", False)
@ -433,7 +436,7 @@ COMMON_ANALYSIS = {
}
def add_multilang_fields(cls):
def add_multilang_fields(cls) -> None:
for code, _lang in settings.LANGUAGES:
lc = code.replace("-", "_").lower()
name_field = f"name_{lc}"
@ -453,7 +456,7 @@ def add_multilang_fields(cls):
),
)
def make_prepare(attr):
def make_prepare(attr: str) -> Callable[[Any, Any], str]:
return lambda self, instance: getattr(instance, attr, "") or ""
setattr(cls, f"prepare_{name_field}", make_prepare(name_field))

View file

@ -277,19 +277,15 @@ class ProductFilter(FilterSet):
ordering_param = self.data.get("order_by", "")
qs = qs.annotate(
has_stock=Max(
Case(
When(stocks__quantity__gt=0, then=Value(True)),
default=Value(False),
output_field=BooleanField(),
)
has_stock=Case(
When(stocks__quantity__gt=0, then=Value(True)),
default=Value(False),
output_field=BooleanField(),
),
has_price=Max(
Case(
When(stocks__price__gt=0, then=Value(True)),
default=Value(False),
output_field=BooleanField(),
)
has_price=Case(
When(stocks__price__gt=0, then=Value(True)),
default=Value(False),
output_field=BooleanField(),
),
).annotate(
personal_orders_only=Case(

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

View file

@ -2,7 +2,7 @@ from django.conf import settings
class RootDirectory:
def __init__(self):
def __init__(self) -> None:
self.label = "root"
self.path = settings.BASE_DIR / "evibes"

View file

@ -1,5 +1,6 @@
import importlib
import os
from argparse import ArgumentParser
import requests
from django.core.management.base import BaseCommand, CommandError
@ -16,7 +17,7 @@ class Command(BaseCommand):
"in the translated_<lang> field created by django-modeltranslation."
)
def add_arguments(self, parser):
def add_arguments(self, parser: ArgumentParser) -> None:
parser.add_argument(
"-t",
"--target",
@ -30,7 +31,7 @@ class Command(BaseCommand):
help="Modeltranslation language code to translate into, e.g. de-de, fr-fr, zh-hans",
)
def handle(self, *args, **options):
def handle(self, *args, **options) -> None:
target = options["target"]
lang = options["language"].lower()

View file

@ -51,9 +51,9 @@ from core.errors import DisabledCommerceError, NotEnoughMoneyError
from core.managers import AddressManager, ProductManager
from core.utils import (
generate_human_readable_id,
generate_human_readable_token,
get_product_uuid_as_path,
get_random_code,
generate_human_readable_token,
)
from core.utils.db import TweakedAutoSlugField, unicode_slugify_function
from core.utils.lists import FAILED_STATUSES
@ -66,7 +66,7 @@ logger = logging.getLogger("django")
class AttributeGroup(ExportModelOperationsMixin("attribute_group"), NiceModel): # type: ignore [misc, django-manager-missing]
__doc__ = _(
__doc__ = _( # type: ignore
"Represents a group of attributes, which can be hierarchical."
" This class is used to manage and organize attribute groups."
" An attribute group can have a parent group, forming a hierarchical structure."
@ -102,7 +102,7 @@ class AttributeGroup(ExportModelOperationsMixin("attribute_group"), NiceModel):
class Vendor(ExportModelOperationsMixin("vendor"), NiceModel): # type: ignore [misc, django-manager-missing]
__doc__ = _(
__doc__ = _( # type: ignore
"Represents a vendor entity capable of storing information about external vendors and their interaction requirements."
" The Vendor class is used to define and manage information related to an external vendor."
" It stores the vendor's name, authentication details required for communication,"
@ -139,7 +139,15 @@ class Vendor(ExportModelOperationsMixin("vendor"), NiceModel): # type: ignore [
def __str__(self) -> str:
return self.name
def save(self, **kwargs) -> None:
def save(
self,
*,
force_insert=False,
force_update=False,
using=None,
update_fields=None,
update_modified: bool = True,
) -> None:
users = self.users.filter(is_active=True)
users = users.exclude(attributes__icontains="is_business")
if users.count() > 0:
@ -148,7 +156,13 @@ class Vendor(ExportModelOperationsMixin("vendor"), NiceModel): # type: ignore [
user.attributes = {}
user.attributes.update({"is_business": True})
user.save()
return super().save(**kwargs)
return super().save(
force_insert=force_insert,
force_update=force_update,
using=using,
update_fields=update_fields,
update_modified=update_modified,
)
class Meta:
verbose_name = _("vendor")
@ -159,7 +173,7 @@ class Vendor(ExportModelOperationsMixin("vendor"), NiceModel): # type: ignore [
class ProductTag(ExportModelOperationsMixin("product_tag"), NiceModel): # type: ignore [misc, django-manager-missing]
__doc__ = _(
__doc__ = _( # type: ignore
"Represents a product tag used for classifying or identifying products."
" The ProductTag class is designed to uniquely identify and classify products through a combination"
" of an internal tag identifier and a user-friendly display name."
@ -191,7 +205,7 @@ class ProductTag(ExportModelOperationsMixin("product_tag"), NiceModel): # type:
class CategoryTag(ExportModelOperationsMixin("category_tag"), NiceModel): # type: ignore [misc, django-manager-missing]
__doc__ = _(
__doc__ = _( # type: ignore
"Represents a category tag used for products."
" This class models a category tag that can be used to associate and classify products."
" It includes attributes for an internal tag identifier and a user-friendly display name."
@ -222,7 +236,7 @@ class CategoryTag(ExportModelOperationsMixin("category_tag"), NiceModel): # typ
class Category(ExportModelOperationsMixin("category"), NiceModel, MPTTModel): # type: ignore [misc, django-manager-missing]
__doc__ = _(
__doc__ = _( # type: ignore
"Represents a category entity to organize and group related items in a hierarchical structure."
" Categories may have hierarchical relationships with other categories, supporting parent-child relationships."
" The class includes fields for metadata and visual representation,"
@ -314,24 +328,12 @@ class Category(ExportModelOperationsMixin("category"), NiceModel, MPTTModel): #
class Brand(ExportModelOperationsMixin("brand"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Represents a Brand object in the system.
This class handles information and attributes related to a brand, including its name, logos,
description, associated categories, a unique slug, and priority order.
It allows for the organization and representation of brand-related data within the application.
Attributes:
is_publicly_visible (bool): Indicates if the brand is visible publicly.
name (str): The name of the brand.
small_logo (ImageField): An optional small logo image file representing the brand.
big_logo (ImageField): An optional large logo image file representing the brand.
description (str): An optional textual description providing details about the brand.
categories (Category): Optional categories associated with this brand.
slug (str): A unique auto-generated slug used for SEO-friendly URLs.
priority (int): Specifies the priority ranking of the brand.
"""
__doc__ = _( # type: ignore
"Represents a Brand object in the system. "
"This class handles information and attributes related to a brand, including its name, logos, "
"description, associated categories, a unique slug, and priority order. "
"It allows for the organization and representation of brand-related data within the application."
)
is_publicly_visible = True
@ -396,7 +398,7 @@ class Brand(ExportModelOperationsMixin("brand"), NiceModel): # type: ignore [mi
class Stock(ExportModelOperationsMixin("stock"), NiceModel): # type: ignore [misc, django-manager-missing]
__doc__ = _(
__doc__ = _( # type: ignore
"Represents the stock of a product managed in the system."
" This class provides details about the relationship between vendors, products, and their stock information, "
"as well as inventory-related properties like price, purchase price, quantity, SKU, and digital assets."
@ -459,7 +461,7 @@ class Stock(ExportModelOperationsMixin("stock"), NiceModel): # type: ignore [mi
class Product(ExportModelOperationsMixin("product"), NiceModel): # type: ignore [misc, django-manager-missing]
__doc__ = _(
__doc__ = _( # type: ignore
"Represents a product with attributes such as category, brand, tags, digital status, name, description, part number, and slug."
" Provides related utility properties to retrieve ratings, feedback counts, price, quantity, and total orders."
" Designed for use in a system that handles e-commerce or inventory management."
@ -557,7 +559,7 @@ class Product(ExportModelOperationsMixin("product"), NiceModel): # type: ignore
if rating is None:
feedbacks = Feedback.objects.filter(order_product__product_id=self.pk)
rating = feedbacks.aggregate(Avg("rating"))["rating__avg"] or 0
cache.set(cache_key, rating, 604800)
cache.set(cache_key, rating, 86400)
return round(rating, 2)
@rating.setter
@ -601,9 +603,13 @@ class Product(ExportModelOperationsMixin("product"), NiceModel): # type: ignore
def personal_orders_only(self) -> bool:
return not (self.quantity > 0 and self.price > 0.0)
@personal_orders_only.setter
def personal_orders_only(self, value):
self.__dict__["personal_orders_only"] = value
class Attribute(ExportModelOperationsMixin("attribute"), NiceModel): # type: ignore [misc, django-manager-missing]
__doc__ = _(
__doc__ = _( # type: ignore
"Represents an attribute in the system."
" This class is used to define and manage attributes,"
" which are customizable pieces of data that can be associated with other entities."
@ -667,28 +673,11 @@ class Attribute(ExportModelOperationsMixin("attribute"), NiceModel): # type: ig
class AttributeValue(ExportModelOperationsMixin("attribute_value"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Represents a specific value for an attribute that is linked to a product.
This class serves the purpose of mapping a value to an attribute for a
specific product. It links the 'attribute' to a unique 'value', allowing
better organization and dynamic representation of product characteristics.
It also defines whether the attribute value is public through the
'is_publicly_visible' attribute.
Attributes
----------
is_publicly_visible
Determines if the attribute value is visible publicly. Defaults to True.
attribute : core.Attribute
The 'Attribute' object this value is linked to. Foreign key relationship
with 'core.Attribute'.
product : core.Product
The specific 'Product' this attribute's value is associated with.
Foreign key relationship with 'core.Product'.
value
Holds the specific value for this attribute as a text field.
"""
__doc__ = _( # type: ignore
"Represents a specific value for an attribute that is linked to a product. "
"It links the 'attribute' to a unique 'value', allowing "
"better organization and dynamic representation of product characteristics."
)
is_publicly_visible = True
@ -722,23 +711,13 @@ class AttributeValue(ExportModelOperationsMixin("attribute_value"), NiceModel):
class ProductImage(ExportModelOperationsMixin("product_image"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Represents a product image associated with a product in the system.
This class is designed to manage images for products, including functionality
for uploading image files, associating them with specific products, and
determining their display order. It also includes an accessibility feature
with alternative text for the images.
Attributes:
is_publicly_visible (bool): A flag indicating whether the image is
visible publicly.
alt (str): Alternative text for the image to support accessibility.
image (ImageField): The image file associated with the product.
priority (int): The display priority of the image. Images with lower
priority values are displayed first.
product (ForeignKey): The product associated with this image.
"""
__doc__ = _( # type: ignore
"Represents a product image associated with a product in the system. "
"This class is designed to manage images for products, including functionality "
"for uploading image files, associating them with specific products, and "
"determining their display order. It also includes an accessibility feature "
"with alternative text for the images."
)
is_publicly_visible = True
@ -779,38 +758,14 @@ class ProductImage(ExportModelOperationsMixin("product_image"), NiceModel): # t
class Promotion(ExportModelOperationsMixin("promotion"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Represents a promotional campaign for products with a discount.
This class is used to define and manage promotional campaigns that offer a
percentage-based discount for products. The class includes attributes for
setting the discount rate, providing details about the promotion, and linking
it to the applicable products. It integrates with the product catalog to
determine the affected items in the campaign.
Attributes:
is_publicly_visible: A class-level attribute indicating whether the promotion
is publicly visible.
discount_percent: IntegerField. Specifies the percentage discount for the
selected products. Must be between 1 and 100 inclusive.
name: CharField. A unique name for the promotion, required for promoting
distinguishable campaigns. The maximum length is 256 characters.
description: TextField, optional. Provides a detailed description of the
promotion. Can be left blank or null.
products. Links the promotion to the products that are included
in its scope. Can be left blank.
Meta:
verbose_name: The singular name for the promotion in database and UI contexts.
verbose_name_plural: The pluralized name for multiple promotions in database and
UI contexts.
Methods:
__str__():
Returns a string representation of the promotion. If the name is
provided, it returns the name; otherwise, it returns the ID of the
promotion as a string.
"""
__doc__ = _( # type: ignore
"Represents a promotional campaign for products with a discount. "
"This class is used to define and manage promotional campaigns that offer a "
"percentage-based discount for products. The class includes attributes for "
"setting the discount rate, providing details about the promotion, and linking "
"it to the applicable products. It integrates with the product catalog to "
"determine the affected items in the campaign."
)
is_publicly_visible = True
@ -849,15 +804,13 @@ class Promotion(ExportModelOperationsMixin("promotion"), NiceModel): # type: ig
class Wishlist(ExportModelOperationsMixin("wishlist"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Represents a user's wishlist for storing and managing desired products.
The Wishlist class provides functionality to manage a collection of products
that a user wishes to save. It supports operations such as adding products,
removing products, adding multiple products in bulk, and removing multiple
products in bulk. The wishlist is associated with a specific user and is
stored with optional public visibility status.
"""
__doc__ = _( # type: ignore
"Represents a user's wishlist for storing and managing desired products. "
"The class provides functionality to manage a collection of products, "
"supporting operations such as adding and removing products, "
"as well as supporting operations for adding and removing multiple "
"products at once."
)
is_publicly_visible = False
@ -922,25 +875,13 @@ class Wishlist(ExportModelOperationsMixin("wishlist"), NiceModel): # type: igno
class Documentary(ExportModelOperationsMixin("attribute_group"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Model representing a documentary record tied to a product.
This class is used to store information about documentaries related to specific
products, including file uploads and their metadata. It contains methods and
properties to handle the file type and storage path for the documentary files.
It extends functionality from specific mixins and provides additional custom
features.
Attributes:
is_publicly_visible: A boolean indicating if the documentary is
publicly visible.
product linking the documentary to a product.
document: FileField used to store the file associated with the documentary.
Meta:
verbose_name: Singular name for the documentary model.
verbose_name_plural: Plural name for the documentary model.
"""
__doc__ = _( # type: ignore
"Represents a documentary record tied to a product. "
"This class is used to store information about documentaries related to specific "
"products, including file uploads and their metadata. It contains methods and "
"properties to handle the file type and storage path for the documentary files. "
"It extends functionality from specific mixins and provides additional custom features."
)
is_publicly_visible = True
@ -963,42 +904,16 @@ class Documentary(ExportModelOperationsMixin("attribute_group"), NiceModel): #
class Address(ExportModelOperationsMixin("address"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Represents an address entity that includes location details and associations with
a user. Provides functionality for geographic and address data storage, as well
as integration with geocoding services.
This class is designed to store detailed address information including components
like street, city, region, country, and geolocation (longitude and latitude).
It supports integration with geocoding APIs, enabling the storage of raw API
responses for further processing or inspection. The class also allows associating
an address with a user, facilitating personalized data handling.
Attributes:
is_publicly_visible (bool): Indicates whether the address is visible publicly.
address_line (str): A general address line containing information about the
customer's location. Optional.
street (str): The street name or number in the address. Optional.
district (str): The district related to the address. Optional.
city (str): The name of the city where the address is located. Optional.
region (str): The name of the region associated with the address. Optional.
postal_code (str): The postal code corresponding to the address. Optional.
country (str): The country where the address resides. Optional.
location (PointField): A geolocation represented as (longitude, latitude).
Allows geospatial searches. Optional.
raw_data (dict): The full JSON response directly from the geocoding service,
containing detailed information about the address. Optional.
api_response (dict): Stores a processed version or subset of the JSON
response from the geocoding service. Optional.
user (ForeignKey): Reference to a User entity who owns this address. Optional.
Meta:
verbose_name (str): Human-readable singular name for the address.
verbose_name_plural (str): Human-readable plural name for addresses.
indexes (list): Database indexes defined for improving query performance
on specific fields like 'location'.
"""
__doc__ = _( # type: ignore
"Represents an address entity that includes location details and associations with a user. "
"Provides functionality for geographic and address data storage, as well "
"as integration with geocoding services. "
"This class is designed to store detailed address information including components "
"like street, city, region, country, and geolocation (longitude and latitude). "
"It supports integration with geocoding APIs, enabling the storage of raw API "
"responses for further processing or inspection. The class also allows associating "
"an address with a user, facilitating personalized data handling."
)
is_publicly_visible = False
@ -1048,33 +963,14 @@ class Address(ExportModelOperationsMixin("address"), NiceModel): # type: ignore
class PromoCode(ExportModelOperationsMixin("promocode"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Represents a promotional code that can be used for discounts, managing its validity,
type of discount, and application.
The PromoCode class stores details about a promotional code, including its unique
identifier, discount properties (amount or percentage), validity period, associated
user (if any), and status of its usage. It includes functionality to validate and
apply the promo code to an order while ensuring constraints are met.
Attributes:
code (str): The unique identifier for the promo code.
discount_amount (Decimal): The fixed discount amount applied, if defined.
discount_percent (int): The percentage discount applied, if defined.
end_time (datetime): The expiration timestamp of the promo code.
start_time (datetime): The timestamp from when the promo code is valid.
used_on (datetime): The timestamp when the promo code was used (if applicable).
user (ForeignKey): The user associated with the promo code, if any.
Methods:
save(**kwargs): Ensures only one type of discount (amount or percent) is defined.
__str__(): Returns the promo code identifier as its string representation.
use(order): Applies the promo code to the given order and calculates the final price.
Meta:
verbose_name: Display name for the promo code model.
verbose_name_plural: Plural display name for the promo code model.
"""
__doc__ = _( # type: ignore
"Represents a promotional code that can be used for discounts, managing its validity, "
"type of discount, and application. "
"The PromoCode class stores details about a promotional code, including its unique "
"identifier, discount properties (amount or percentage), validity period, associated "
"user (if any), and status of its usage. It includes functionality to validate and "
"apply the promo code to an order while ensuring constraints are met."
)
is_publicly_visible = False
@ -1132,14 +1028,28 @@ class PromoCode(ExportModelOperationsMixin("promocode"), NiceModel): # type: ig
verbose_name = _("promo code")
verbose_name_plural = _("promo codes")
def save(self, **kwargs):
def save(
self,
*args,
force_insert=False,
force_update=False,
using=None,
update_fields=None,
update_modified: bool = True,
) -> None:
if (self.discount_amount is not None and self.discount_percent is not None) or (
self.discount_amount is None and self.discount_percent is None
):
raise ValidationError(
_("only one type of discount should be defined (amount or percent), but not both or neither.")
)
super().save(**kwargs)
return super().save(
force_insert=force_insert,
force_update=force_update,
using=using,
update_fields=update_fields,
update_modified=update_modified,
)
def __str__(self) -> str:
return self.code
@ -1176,7 +1086,7 @@ class PromoCode(ExportModelOperationsMixin("promocode"), NiceModel): # type: ig
class Order(ExportModelOperationsMixin("order"), NiceModel): # type: ignore [misc, django-manager-missing]
__doc__ = _(
__doc__ = _( # type: ignore
"Represents an order placed by a user."
" This class models an order within the application,"
" including its various attributes such as billing and shipping information,"
@ -1270,21 +1180,40 @@ class Order(ExportModelOperationsMixin("order"), NiceModel): # type: ignore [mi
self.attributes = {}
self.save()
return False
if self.user:
if type(self.user.attributes) is not dict:
self.user.attributes = {}
self.user.save()
return False
with suppress(Exception):
return (self.attributes.get("is_business", False) if self.attributes else False) or (
(self.user.attributes.get("is_business", False) and self.user.attributes.get("business_identificator"))
(self.user.attributes.get("is_business", False) and self.user.attributes.get("business_identificator")) # type: ignore [union-attr]
if self.user
else False
)
return False
def save(self, **kwargs) -> Self:
def save(
self,
*args,
force_insert=False,
force_update=False,
using=None,
update_fields=None,
update_modified: bool = True,
) -> None:
pending_orders = 0
if self.user:
pending_orders = self.user.orders.filter(status="PENDING").count()
if self.status == "PENDING" and pending_orders > 1:
raise ValueError(_("a user must have only one pending order at a time"))
return super().save(**kwargs)
return super().save(
force_insert=force_insert,
force_update=force_update,
using=using,
update_fields=update_fields,
update_modified=update_modified,
)
@property
def total_price(self) -> float:
@ -1431,8 +1360,8 @@ class Order(ExportModelOperationsMixin("order"), NiceModel): # type: ignore [mi
shipping_address = billing_address
else:
billing_address = Address.objects.get(uuid=billing_address_uuid)
shipping_address = Address.objects.get(uuid=shipping_address_uuid)
billing_address = Address.objects.get(uuid=str(billing_address_uuid))
shipping_address = Address.objects.get(uuid=str(shipping_address_uuid))
self.billing_address = billing_address
self.shipping_address = shipping_address
@ -1645,30 +1574,29 @@ class Order(ExportModelOperationsMixin("order"), NiceModel): # type: ignore [mi
logger.error(traceback.format_exc())
return False
@property
def business_identificator(self) -> str | None:
if self.attributes:
return self.attributes.get("business_identificator") or self.attributes.get("businessIdentificator")
if self.user:
if self.user.attributes:
return self.user.attributes.get("business_identificator") or self.user.attributes.get(
"businessIdentificator"
)
return None
class OrderProduct(ExportModelOperationsMixin("order_product"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Represents a product associated with an order.
The OrderProduct model maintains information about a product that is part of an order,
including details such as purchase price, quantity, product attributes, and status. It
manages notifications for the user and administrators and handles operations such as
returning the product balance or adding feedback. This model also provides methods and
properties that support business logic, such as calculating the total price or generating
a download URL for digital products. The model integrates with the Order and Product models
and stores a reference to them.
Attributes:
is_publicly_visible (bool): Indicates whether this model is visible publicly.
buy_price (float): The price paid by the customer for this product at purchase time.
comments (str): Internal comments entered by admins regarding this ordered product.
notifications (dict): JSON structure containing notifications relevant to the product.
attributes (dict): JSON representation of the product's attributes as part of the order.
order (Order): Reference to the parent order that contains this product.
product (Product): Reference to the specific product associated with the order line.
quantity (int): Represents the quantity of this product ordered.
status (str): The current status of the product in the order.
"""
__doc__ = _( # type: ignore
"Represents products associated with orders and their attributes. "
"The OrderProduct model maintains information about a product that is part of an order, "
"including details such as purchase price, quantity, product attributes, and status. It "
"manages notifications for the user and administrators and handles operations such as "
"returning the product balance or adding feedback. This model also provides methods and "
"properties that support business logic, such as calculating the total price or generating "
"a download URL for digital products. The model integrates with the Order and Product models "
"and stores a reference to them."
)
is_publicly_visible = False
@ -1780,9 +1708,9 @@ class OrderProduct(ExportModelOperationsMixin("order_product"), NiceModel): # t
def download_url(self: Self) -> str:
if self.product and self.product.stocks:
if self.product.is_digital and self.product.stocks.first().digital_asset: # type: ignore [union-attr]
try:
if hasattr(self, "download"):
return self.download.url
except self.download.RelatedObjectDoesNotExist:
else:
return DigitalAssetDownload.objects.create(order_product=self).url
return ""
@ -1809,7 +1737,7 @@ class OrderProduct(ExportModelOperationsMixin("order_product"), NiceModel): # t
return None
class CustomerRelationshipManagementProvider(ExportModelOperationsMixin("crm_provider"), NiceModel):
class CustomerRelationshipManagementProvider(ExportModelOperationsMixin("crm_provider"), NiceModel): # type: ignore [misc, django-manager-missing]
name = CharField(max_length=128, unique=True, verbose_name=_("name"))
integration_url = URLField(blank=True, null=True, help_text=_("URL of the integration"))
authentication = JSONField(blank=True, null=True, help_text=_("authentication credentials"))
@ -1820,14 +1748,28 @@ class CustomerRelationshipManagementProvider(ExportModelOperationsMixin("crm_pro
def __str__(self) -> str:
return self.name
def save(self, **kwargs):
def save(
self,
*args,
force_insert=False,
force_update=False,
using=None,
update_fields=None,
update_modified: bool = True,
) -> None:
if self.default:
qs = type(self).objects.all()
if self.pk:
qs = qs.exclude(pk=self.pk)
if qs.filter(default=True).exists():
raise ValueError(_("you can only have one default CRM provider"))
super().save(**kwargs)
super().save(
force_insert=force_insert,
force_update=force_update,
using=using,
update_fields=update_fields,
update_modified=update_modified,
)
class Meta:
verbose_name = _("CRM")
@ -1848,29 +1790,14 @@ class OrderCrmLink(ExportModelOperationsMixin("order_crm_link"), NiceModel): #
class DigitalAssetDownload(ExportModelOperationsMixin("attribute_group"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Represents the downloading functionality for digital assets associated
with orders.
The DigitalAssetDownload class provides the ability to manage and access
downloads related to order products. It maintains information about the
associated order product, the number of downloads, and whether the asset
is publicly visible. It includes a method to generate a URL for downloading
the asset when the associated order is in a completed status.
Attributes:
is_publicly_visible (bool): Indicates whether the digital asset is
publicly visible. Always set to False for this class.
order_product (OneToOneField): Reference to the associated order product.
It has a one-to-one relationship with the OrderProduct model, and
deleting the OrderProduct will delete the associated download.
num_downloads (int): Indicates the number of times the digital asset
has been downloaded.
Methods:
url: Property to generate the download URL for the digital asset
if the associated order is in a finished status.
"""
__doc__ = _( # type: ignore
"Represents the downloading functionality for digital assets associated with orders. "
"The DigitalAssetDownload class provides the ability to manage and access "
"downloads related to order products. It maintains information about the "
"associated order product, the number of downloads, and whether the asset "
"is publicly visible. It includes a method to generate a URL for downloading "
"the asset when the associated order is in a completed status."
)
is_publicly_visible = False
@ -1886,31 +1813,19 @@ class DigitalAssetDownload(ExportModelOperationsMixin("attribute_group"), NiceMo
@property
def url(self):
if self.order_product.status != "FINISHED":
raise ValueError(_("you can not download a digital asset for a non-finished order"))
return (
f"https://api.{config.BASE_DOMAIN}/download/{urlsafe_base64_encode(force_bytes(self.order_product.uuid))}"
)
class Feedback(ExportModelOperationsMixin("feedback"), NiceModel): # type: ignore [misc, django-manager-missing]
"""
Manages user feedback for products.
This class is designed to capture and store user feedback for specific products
that they have purchased. It contains attributes to store user comments,
a reference to the related product in the order, and a user-assigned rating. The
class uses database fields to effectively model and manage feedback data.
Attributes:
is_publicly_visible (bool): Indicates whether the feedback is visible to the public.
comment (str): User-provided comments about their experience with the product.
order_product (OrderProduct): Reference to the specific product in an order that this
feedback is about.
rating (float): User-assigned rating for the product, validated to be within the range
of 0 to 10.
"""
__doc__ = _( # type: ignore
"Manages user feedback for products. "
"This class is designed to capture and store user feedback for specific products "
"that they have purchased. It contains attributes to store user comments, "
"a reference to the related product in the order, and a user-assigned rating. The "
"class uses database fields to effectively model and manage feedback data."
)
is_publicly_visible = True

View file

@ -84,15 +84,18 @@ def process_order_changes(instance, created, **_kwargs):
except IntegrityError:
human_readable_id = generate_human_readable_id()
while True:
if Order.objects.filter(human_readable_id=human_readable_id).exists():
human_readable_id = generate_human_readable_id()
try:
if Order.objects.filter(human_readable_id=human_readable_id).exists():
human_readable_id = generate_human_readable_id()
continue
Order.objects.create(
user=instance,
status="PENDING",
human_readable_id=human_readable_id,
)
break
except IntegrityError:
continue
Order.objects.create(
user=instance,
status="PENDING",
human_readable_id=human_readable_id,
)
break
if instance.status in ["CREATED", "PAYMENT"]:
if not instance.is_whole_digital:
@ -110,12 +113,15 @@ def process_order_changes(instance, created, **_kwargs):
if has_file:
order_product.status = "FINISHED"
download = DigitalAssetDownload.objects.create(order_product=order_product)
order_product.download = download
order_product.save()
order_product.order.user.payments_balance.amount -= order_product.buy_price
order_product.order.user.payments_balance.save()
continue
if not order_product.download:
DigitalAssetDownload.objects.create(order_product=order_product)
order_product.order.user.payments_balance.amount -= order_product.buy_price
order_product.order.user.payments_balance.save()
order_product.save()
continue
order_product.save()
try:
vendor_name = (
order_product.product.stocks.filter(price=order_product.buy_price).first().vendor.name.lower()

View file

@ -13,7 +13,7 @@ from django.core.cache import cache
from core.models import Product, Promotion
from core.utils.caching import set_default_cache
from core.vendors import delete_stale, VendorInactiveError
from core.vendors import VendorInactiveError, delete_stale
from evibes.settings import MEDIA_ROOT
logger = get_task_logger(__name__)
@ -38,7 +38,7 @@ def update_products_task() -> tuple[bool, str]:
if not update_products_task_running:
cache.set("update_products_task_running", True, 86400)
vendors_classes = []
vendors_classes: list = []
for vendor_class in vendors_classes:
vendor = vendor_class()
@ -69,7 +69,7 @@ def update_orderproducts_task() -> tuple[bool, str]:
message confirming the successful execution of the task.
:rtype: Tuple[bool, str]
"""
vendors_classes = []
vendors_classes: list = []
for vendor_class in vendors_classes:
vendor = vendor_class()
@ -95,7 +95,7 @@ def set_default_caches_task() -> tuple[bool, str]:
@shared_task(queue="default")
def remove_stale_product_images() -> tuple[bool, str] | None:
def remove_stale_product_images() -> tuple[bool, str]:
"""
Removes stale product images from the products directory by identifying directories
whose names do not match any UUIDs currently present in the database.
@ -114,7 +114,7 @@ def remove_stale_product_images() -> tuple[bool, str] | None:
products_dir = os.path.join(MEDIA_ROOT, "products")
if not os.path.isdir(products_dir):
logger.info("The products directory does not exist: %s", products_dir)
return
return True, "The products directory does not exist."
# Load all current product UUIDs into a set.
# This query returns all product UUIDs (as strings or UUID objects).
@ -139,6 +139,7 @@ def remove_stale_product_images() -> tuple[bool, str] | None:
logger.info("Removed stale product images directory: %s", entry_path)
except Exception as e:
logger.error("Error removing directory %s: %s", entry_path, e)
return True, "Successfully removed stale product images."
@shared_task(queue="default")

View file

@ -37,7 +37,7 @@
</table>
<button type="button" class="add-row-button" data-table-id="json-fields-{{ widget.attrs.id }}">
Add Row
{% blocktrans %}Add Row{% endblocktrans %}
</button>
<script>

View file

@ -15,7 +15,7 @@ from evibes.settings import DEBUG, EXPOSABLE_KEYS, LANGUAGE_CODE
logger = logging.getLogger("django")
def graphene_current_lang():
def graphene_current_lang() -> str:
"""
Determines the currently active language code.
@ -45,7 +45,7 @@ def graphene_abs(request, path_or_url: str) -> str:
Returns:
str: The absolute URI corresponding to the provided path or URL.
"""
return request.build_absolute_uri(path_or_url)
return str(request.build_absolute_uri(path_or_url))
def get_random_code() -> str:
@ -64,40 +64,10 @@ def get_random_code() -> str:
def get_product_uuid_as_path(instance, filename: str = "") -> str:
"""
Generates a file path for a product using its UUID.
This function constructs a standardized file path where an uploaded file
is saved for a product. The path includes a `products` directory, followed
by the product's UUID, and concludes with the original filename. It can be
utilized in file storage applications to ensure unique and organized file
storage based on the product's identity.
Args:
instance: The object instance that contains a reference to the product.
filename: str, optional. The name of the file being uploaded. Default is an
empty string.
Returns:
str: A string that represents the constructed file path.
"""
return "products" + "/" + str(instance.product.uuid) + "/" + filename
def get_brand_name_as_path(instance, filename: str = "") -> str:
"""
Generates a file path for a brand based on its name and the provided filename.
This function constructs a unique file path within the 'brands/' directory using
the name of the given instance and appends the supplied filename.
Parameters:
instance: An object containing a 'name' attribute.
filename: str, optional. The name of the file to be appended to the path.
Returns:
str: A string representing the constructed file path.
"""
return "brands/" + str(instance.name) + "/" + filename
@ -111,9 +81,6 @@ def atomic_if_not_debug():
database transaction, preventing partial updates to the database in case of
an exception. If the DEBUG setting is enabled, no transaction is enforced,
allowing for easier debugging.
Yields:
None: This context manager does not return any values.
"""
if not DEBUG:
with transaction.atomic():
@ -123,18 +90,6 @@ def atomic_if_not_debug():
def is_url_safe(url: str) -> bool:
"""
Determines if a given URL starts with "https://" indicating it is a secure URL.
This function checks if the provided URL adheres to secure HTTPS protocol.
It uses a regular expression to validate the URL prefix.
Parameters:
url (str): The URL string to validate.
Returns:
bool: True if the URL starts with "https://", False otherwise.
"""
return bool(re.match(r"^https://", url, re.IGNORECASE))
@ -146,14 +101,6 @@ def format_attributes(attributes: str | None = None) -> dict:
formatted as `key=value` pairs separated by commas, and converts it into a
dictionary. It returns an empty dictionary if the input is `None` or invalid.
Invalid key-value pairs within the input string are skipped.
Parameters:
attributes (str | None): A comma-separated string of key-value pairs in the
format `key=value`, or None.
Returns:
dict: A dictionary where keys are the attribute names and values are their
corresponding values.
"""
if not attributes:
return {}
@ -182,9 +129,6 @@ def get_project_parameters() -> dict:
If they are not cached, it collects the parameters from a designated
configuration source, formats their keys to lowercase, and then stores
them in the cache for a limited period.
Returns:
dict: A dictionary containing the project parameters with lowercase keys.
"""
parameters = cache.get("parameters", {})
@ -197,19 +141,11 @@ def get_project_parameters() -> dict:
return parameters
def resolve_translations_for_elasticsearch(instance, field_name) -> None:
def resolve_translations_for_elasticsearch(instance, field_name: str) -> None:
"""
Resolves translations for a given field in an Elasticsearch-compatible
format. It checks if the localized version of the field contains data,
and if not, sets it to the value of the default field.
Parameters:
instance: The object instance containing the field to resolve.
field_name (str): The base name of the field for which translations
are being resolved.
Returns:
None
"""
field = getattr(instance, f"{field_name}_{LANGUAGE_CODE}", "")
filled_field = getattr(instance, field_name, "")

View file

@ -1,10 +1,15 @@
import json
import logging
from pathlib import Path
from typing import Any
from django.contrib.auth.base_user import AbstractBaseUser
from django.contrib.auth.models import AnonymousUser
from django.core.cache import cache
from django.core.exceptions import BadRequest
from django.utils.translation import gettext_lazy as _
from graphene import Context
from rest_framework.request import Request
from evibes.settings import UNSAFE_CACHE_KEYS
from vibes_auth.models import User
@ -26,7 +31,9 @@ def get_cached_value(user: User, key: str, default=None) -> None | object:
return None
def set_cached_value(user: User, key: str, value: object, timeout: int = 3600) -> None | object:
def set_cached_value(
user: User | AbstractBaseUser | AnonymousUser, key: str, value: object, timeout: int = 3600
) -> None | object:
if user.is_staff or user.is_superuser:
cache.set(key, value, timeout)
return value
@ -34,7 +41,7 @@ def set_cached_value(user: User, key: str, value: object, timeout: int = 3600) -
return None
def web_cache(request, key, data, timeout):
def web_cache(request: Request | Context, key: str, data: dict[str, Any], timeout: int):
if not data and not timeout:
return {"data": get_cached_value(request.user, key)}
if (data and not timeout) or (timeout and not data):
@ -44,7 +51,7 @@ def web_cache(request, key, data, timeout):
return {"data": set_cached_value(request.user, key, data, timeout)}
def set_default_cache():
def set_default_cache() -> None:
data_dir = Path(__file__).resolve().parent.parent / "data"
for json_file in data_dir.glob("*.json"):
with json_file.open("r", encoding="utf-8") as f:

View file

@ -1,11 +1,9 @@
import re
from django.core.exceptions import ValidationError
from django.core.files.images import get_image_dimensions
from django.core.files.images import ImageFile, get_image_dimensions
from django.utils.translation import gettext_lazy as _
def validate_category_image_dimensions(image):
def validate_category_image_dimensions(image: ImageFile) -> None:
max_width = 99999
max_height = 99999
@ -14,9 +12,3 @@ def validate_category_image_dimensions(image):
if width > max_width or height > max_height:
raise ValidationError(_(f"image dimensions should not exceed w{max_width} x h{max_height} pixels"))
def validate_phone_number(value, **_kwargs):
phone_regex = re.compile(r"^\+?1?\d{9,15}$")
if not phone_regex.match(value):
raise ValidationError(_("invalid phone number format"))

View file

@ -5,6 +5,7 @@ from math import ceil, log10
from typing import Any
from django.db import IntegrityError, transaction
from django.db.models import QuerySet
from core.elasticsearch import process_system_query
from core.models import (
@ -235,7 +236,7 @@ class AbstractVendor:
if not rate:
raise RatesError(f"No rate found for {currency or self.currency} in {rates} with probider {provider}...")
return float(round(price / rate, 2)) if rate else round(price, 2)
return float(round(price / rate, 2)) if rate else float(round(price, 2)) # type: ignore [arg-type]
@staticmethod
def round_price_marketologically(price: float) -> float:
@ -282,13 +283,13 @@ class AbstractVendor:
def get_products(self) -> None:
pass
def get_products_queryset(self):
def get_products_queryset(self) -> QuerySet[Product]:
return Product.objects.filter(stocks__vendor=self.get_vendor_instance(), orderproduct__isnull=True)
def get_stocks_queryset(self):
def get_stocks_queryset(self) -> QuerySet[Stock]:
return Stock.objects.filter(product__in=self.get_products_queryset(), product__orderproduct__isnull=True)
def get_attribute_values_queryset(self):
def get_attribute_values_queryset(self) -> QuerySet[AttributeValue]:
return AttributeValue.objects.filter(
product__in=self.get_products_queryset(), product__orderproduct__isnull=True
)

View file

@ -8,7 +8,7 @@ from django.contrib.sitemaps.views import index as _sitemap_index_view
from django.contrib.sitemaps.views import sitemap as _sitemap_detail_view
from django.core.cache import cache
from django.core.exceptions import BadRequest
from django.http import FileResponse, Http404, JsonResponse
from django.http import FileResponse, Http404, JsonResponse, HttpRequest, HttpResponseRedirect
from django.shortcuts import redirect
from django.utils.decorators import method_decorator
from django.utils.http import urlsafe_base64_decode
@ -24,6 +24,7 @@ from graphene_file_upload.django import FileUploadGraphQLView
from rest_framework import status
from rest_framework.permissions import AllowAny
from rest_framework.renderers import MultiPartRenderer
from rest_framework.request import Request
from rest_framework.response import Response
from rest_framework.views import APIView
from rest_framework_xml.renderers import XMLRenderer
@ -61,74 +62,40 @@ logger = logging.getLogger("django")
@cache_page(60 * 60 * 12)
@vary_on_headers("Host")
def sitemap_index(request, *args, **kwargs):
"""
Handles the request for the sitemap index and returns an XML response. It ensures the response includes
the appropriate content type header for XML.
Args:
request: The HTTP request object.
*args: Additional positional arguments passed to the view.
**kwargs: Additional keyword arguments passed to the view.
Returns:
A response object containing the sitemap index in XML format, with the proper content type set as
"application/xml; charset=utf-8".
"""
response = _sitemap_index_view(request, *args, **kwargs)
response["Content-Type"] = "application/xml; charset=utf-8"
return response
# noinspection PyTypeChecker
sitemap_index.__doc__ = _( # type: ignore [assignment]
"Handles the request for the sitemap index and returns an XML response. "
"It ensures the response includes the appropriate content type header for XML."
)
@cache_page(60 * 60 * 24)
@vary_on_headers("Host")
def sitemap_detail(request, *args, **kwargs):
"""
Handles the detailed view response for a sitemap. This function processes
the request, fetches the appropriate sitemap detail response, and sets the
Content-Type header for XML responses.
Args:
request: An HTTP request object containing request metadata, such as
headers and HTTP method.
*args: Additional positional arguments provided dynamically to the
underlying sitemap detail view function.
**kwargs: Additional keyword arguments provided dynamically to the
underlying sitemap detail view function.
Returns:
HttpResponse: A response object with content representing the requested
sitemap details. The Content-Type header is explicitly set to
"application/xml; charset=utf-8".
"""
response = _sitemap_detail_view(request, *args, **kwargs)
response["Content-Type"] = "application/xml; charset=utf-8"
return response
# noinspection PyTypeChecker
sitemap_detail.__doc__ = _( # type: ignore [assignment]
"Handles the detailed view response for a sitemap. "
"This function processes the request, fetches the appropriate "
"sitemap detail response, and sets the Content-Type header for XML."
)
class CustomGraphQLView(FileUploadGraphQLView):
"""
A custom GraphQL view class that extends the functionality of FileUploadGraphQLView.
This class serves as a customization extension of FileUploadGraphQLView that allows modification
or enhancement of specific behaviors, particularly the context handling for GraphQL requests.
"""
def get_context(self, request):
return request
class CustomSwaggerView(SpectacularSwaggerView):
"""
CustomSwaggerView is a subclass of SpectacularSwaggerView.
This class overrides the `get_context_data` method to
add extra context to the response. It modifies the context by
including the absolute URI of the current request as the `script_url`.
This can be useful in scenarios where the script or reference
URL needs to be dynamically generated and included in the context.
"""
def get_context_data(self, **kwargs):
# noinspection PyUnresolvedReferences
context = super().get_context_data(**kwargs)
@ -137,16 +104,6 @@ class CustomSwaggerView(SpectacularSwaggerView):
class CustomRedocView(SpectacularRedocView):
"""
CustomRedocView provides a customized version of the SpectacularRedocView.
This class extends the SpectacularRedocView to include additional
functionality, such as dynamically setting the `script_url` in the
context data. It is designed to be used where customized behavior
for rendering ReDoc UI is required, specifically adapting the script
URL for the current request environment.
"""
def get_context_data(self, **kwargs):
# noinspection PyUnresolvedReferences
context = super().get_context_data(**kwargs)
@ -156,24 +113,7 @@ class CustomRedocView(SpectacularRedocView):
@extend_schema_view(**LANGUAGE_SCHEMA)
class SupportedLanguagesView(APIView):
"""
Handles retrieving the list of supported languages.
This class provides an endpoint to return information about available languages.
It is configured with relevant serializers, permission classes, and renderers
for flexibility in response formats and access permissions. The endpoint
supports retrieving the list of languages with their respective codes, names,
and flags.
Attributes:
serializer_class (Serializer): Serializer used for formatting the response data.
permission_classes (list): Permissions applied to restrict the endpoint access.
renderer_classes (list): Renderers available for formatting response output.
Methods:
get(self, request): Retrieves the list of supported languages.
"""
__doc__ = _("Returns a list of supported languages and their corresponding information.") # type: ignore [assignment]
serializer_class = LanguageSerializer
permission_classes = [
@ -186,7 +126,7 @@ class SupportedLanguagesView(APIView):
YAMLRenderer,
]
def get(self, request):
def get(self, request: Request, *args, **kwargs) -> Response:
return Response(
data=self.serializer_class(
[
@ -205,30 +145,7 @@ class SupportedLanguagesView(APIView):
@extend_schema_view(**PARAMETERS_SCHEMA)
class WebsiteParametersView(APIView):
"""
Handles operations related to website parameters.
This class is a Django Rest Framework view that allows clients to retrieve
the parameters of a website. It uses different renderers to present the data
in various formats. The view is publicly accessible.
Attributes
----------
serializer_class
A placeholder for a DRF serializer, it is set to None since no serializer
is explicitly used in this view.
permission_classes
A list indicating the permissions required to access this view. In this case,
`AllowAny`, meaning the view is open to everyone.
renderer_classes
A list of renderers available for this view, supporting CamelCase JSON,
multipart forms, XML, and YAML formats.
Methods
-------
get(request)
Handles HTTP GET requests to fetch website parameters.
"""
__doc__ = _("Returns the parameters of the website as a JSON object.") # type: ignore [assignment]
serializer_class = None
permission_classes = [
@ -241,30 +158,13 @@ class WebsiteParametersView(APIView):
YAMLRenderer,
]
def get(self, request):
def get(self, request: Request, *args, **kwargs) -> Response:
return Response(data=camelize(get_project_parameters()), status=status.HTTP_200_OK)
@extend_schema_view(**CACHE_SCHEMA)
class CacheOperatorView(APIView):
"""
View for managing cache operations.
This class provides an API view for handling cache operations such as setting cache
data with a specified key and timeout. It leverages multiple renderer classes for
serializing outputs in various formats.
Attributes:
serializer_class (type): Serializer to validate and deserialize input data.
permission_classes (list): List of permission classes to apply access
restrictions.
renderer_classes (list): List of renderer classes to serialize the output
in desired formats.
Methods:
post(request, *args, **kwargs): Handles HTTP POST requests to set cache
data based on the provided key and timeout.
"""
__doc__ = _("Handles cache operations such as reading and setting cache data with a specified key and timeout.") # type: ignore [assignment]
serializer_class = CacheOperatorSerializer
permission_classes = [
@ -277,7 +177,7 @@ class CacheOperatorView(APIView):
YAMLRenderer,
]
def post(self, request, *args, **kwargs):
def post(self, request: Request, *args, **kwargs) -> Response:
return Response(
data=web_cache(
request,
@ -291,22 +191,7 @@ class CacheOperatorView(APIView):
@extend_schema_view(**CONTACT_US_SCHEMA)
class ContactUsView(APIView):
"""
Handles contact us form submissions via a REST API.
This view processes user submissions for a "Contact Us" form. It validates the received
data using a serializer, applies rate limiting for IP-based requests, and sends emails
asynchronously. The view is prepared to handle multiple response formats using configured
renderers.
Attributes:
serializer_class: The serializer class used to validate incoming data.
renderer_classes: A list of renderers to support multiple response formats.
Methods:
post: Handles POST requests to process form submissions.
"""
__doc__ = _("Handles `contact us` form submissions.") # type: ignore [assignment]
serializer_class = ContactUsSerializer
renderer_classes = [
@ -317,7 +202,7 @@ class ContactUsView(APIView):
]
@method_decorator(ratelimit(key="ip", rate="2/h", method="POST", block=True))
def post(self, request, *args, **kwargs):
def post(self, request: Request, *args, **kwargs) -> Response:
serializer = self.serializer_class(data=request.data)
serializer.is_valid(raise_exception=True)
contact_us_email.delay(serializer.validated_data)
@ -327,20 +212,7 @@ class ContactUsView(APIView):
@extend_schema_view(**REQUEST_CURSED_URL_SCHEMA)
class RequestCursedURLView(APIView):
"""
Handles requests for processing and validating URLs from incoming POST requests.
Particularly intended for validating and processing URLs provided by clients. It uses rate-limiting, caching, and
various response format renderers to optimize performance and ensure safe handling of external data.
Attributes:
permission_classes (list): Specifies the permissions required to access this view.
renderer_classes (list): Configures the response format renderers available for this view.
Methods:
post: Handles the POST request to validate the URL, fetches its data if valid,
and returns the processed response.
"""
__doc__ = _("Handles requests for processing and validating URLs from incoming POST requests.") # type: ignore [assignment]
permission_classes = [
AllowAny,
@ -353,7 +225,7 @@ class RequestCursedURLView(APIView):
]
@method_decorator(ratelimit(key="ip", rate="10/h"))
def post(self, request, *args, **kwargs):
def post(self, request: Request, *args, **kwargs) -> Response:
url = request.data.get("url")
if not is_url_safe(url):
return Response(
@ -380,23 +252,7 @@ class RequestCursedURLView(APIView):
@extend_schema_view(**SEARCH_SCHEMA)
class GlobalSearchView(APIView):
"""
Class-based view for handling global search functionality.
This class is designed to process search queries from HTTP GET requests. It is
capable of rendering results in multiple formats including CamelCase JSON,
MultiPart, XML, and YAML. The class uses a custom schema for API documentation
and processes search queries passed as parameters in the request.
Attributes:
renderer_classes (list): List of renderer classes used to serialize responses
into various formats such as CamelCase JSON, MultiPart, XML, and YAML.
Methods:
get: Handles HTTP GET requests by processing the search query and returning
formatted search results.
"""
__doc__ = _("Handles global search queries.") # type: ignore [assignment]
renderer_classes = [
CamelCaseJSONRenderer,
@ -405,31 +261,17 @@ class GlobalSearchView(APIView):
YAMLRenderer,
]
def get(self, request, *args, **kwargs):
def get(self, request: Request, *args, **kwargs) -> Response:
return Response(camelize({"results": process_query(query=request.GET.get("q", "").strip(), request=request)}))
@extend_schema_view(**BUY_AS_BUSINESS_SCHEMA)
class BuyAsBusinessView(APIView):
"""
View for buying as a business.
This view handles the logic of creating orders and processing transactions when a
business makes a purchase. It ensures that the request data is properly validated
and processed, handles the creation of the order, and starts the transaction process.
The view also restricts the rate of requests based on the client's IP address
to prevent abuse.
Attributes:
schema (class): Extended schema for API documentation.
Methods:
post(request, *_args, **kwargs):
Handles the "POST" request to process a business purchase.
"""
__doc__ = _("Handles the logic of buying as a business without registration.") # type: ignore [assignment]
# noinspection PyUnusedLocal
@method_decorator(ratelimit(key="ip", rate="10/h" if not settings.DEBUG else "888/h"))
def post(self, request, *_args, **kwargs):
def post(self, request: Request, *args, **kwargs) -> Response:
serializer = BuyAsBusinessOrderSerializer(data=request.data)
serializer.is_valid(raise_exception=True)
order = Order.objects.create(status="MOMENTAL")
@ -459,26 +301,7 @@ class BuyAsBusinessView(APIView):
)
def download_digital_asset_view(request, *args, **kwargs):
"""
Handles the downloading of a digital asset associated with an order. Ensures that users
are permitted to download the asset only once. Validates the request, retrieves the file,
and serves it as a downloadable response. Returns appropriate error responses for different
failure scenarios.
Args:
request: The HTTP request object containing information about the client request.
*args: Additional positional arguments.
**kwargs: Additional keyword arguments.
Raises:
BadRequest: If the digital asset has already been downloaded.
DigitalAssetDownload.DoesNotExist: If the requested digital asset cannot be found.
Returns:
A FileResponse containing the digital asset file if the request is valid. Returns
a JsonResponse with an error message if an error occurs during the process.
"""
def download_digital_asset_view(request: HttpRequest, *args, **kwargs) -> FileResponse | JsonResponse:
try:
logger.debug(f"download_digital_asset_view: {kwargs}")
uuid = urlsafe_base64_decode(str(kwargs.get("order_product_uuid"))).decode("utf-8")
@ -488,6 +311,9 @@ def download_digital_asset_view(request, *args, **kwargs):
if download.num_downloads >= 1:
raise BadRequest(_("you can only download the digital asset once"))
if download.order_product.status != "FINISHED":
raise BadRequest(_("the order must be paid before downloading the digital asset"))
download.num_downloads += 1
download.save()
@ -522,24 +348,16 @@ def download_digital_asset_view(request, *args, **kwargs):
)
def favicon_view(request, *args, **kwargs):
"""
Handles requests for the favicon of a website. This function attempts to serve the favicon
file located in the static directory of the project. If the favicon file is not found,
an HTTP 404 error is raised to indicate the resource is unavailable.
# noinspection PyTypeChecker
download_digital_asset_view.__doc__ = _( # type: ignore [assignment]
"Handles the downloading of a digital asset associated with an order.\n"
"This function attempts to serve the digital asset file located in the "
"storage directory of the project. If the file is not found, an HTTP 404 "
"error is raised to indicate the resource is unavailable."
)
Args:
request: The HTTP request object.
*args: Additional positional arguments that are ignored in this function.
**kwargs: Additional keyword arguments that are ignored in this function.
Returns:
FileResponse: A file response containing the favicon image with the content-type
"image/x-icon".
Raises:
Http404: Raised if the favicon file is not found.
"""
def favicon_view(request: HttpRequest, *args, **kwargs) -> FileResponse | Http404:
try:
favicon_path = os.path.join(settings.BASE_DIR, "static/favicon.png")
return FileResponse(open(favicon_path, "rb"), content_type="image/x-icon")
@ -547,20 +365,22 @@ def favicon_view(request, *args, **kwargs):
raise Http404(_("favicon not found")) from fnfe
def index(request, *args, **kwargs):
"""
Redirects the request to the admin index page.
# noinspection PyTypeChecker
favicon_view.__doc__ = _( # type: ignore [assignment]
"Handles requests for the favicon of a website.\n"
"This function attempts to serve the favicon file located in the static directory of the project. "
"If the favicon file is not found, an HTTP 404 error is raised to indicate the resource is unavailable."
)
The function handles incoming HTTP requests and redirects them to the Django
admin interface index page. It uses Django's `redirect` function for handling
the HTTP redirection.
Args:
request: The HttpRequest object representing the incoming request.
*args: Additional positional arguments, if any.
**kwargs: Additional keyword arguments, if any.
Returns:
HttpResponseRedirect: An HTTP response that redirects to the admin index.
"""
def index(request: HttpRequest, *args, **kwargs) -> HttpResponseRedirect:
return redirect("admin:index")
# noinspection PyTypeChecker
index.__doc__ = _( # type: ignore [assignment]
"Redirects the request to the admin index page. "
"The function handles incoming HTTP requests and redirects them to the Django "
"admin interface index page. It uses Django's `redirect` function for handling "
"the HTTP redirection."
)

Some files were not shown because too many files have changed in this diff Show more