diff --git a/lemarche/conversations/admin.py b/lemarche/conversations/admin.py index 1573cbaea..af18c87d0 100644 --- a/lemarche/conversations/admin.py +++ b/lemarche/conversations/admin.py @@ -1,9 +1,11 @@ from django.contrib import admin from django.http import HttpResponseRedirect +from django.urls import reverse +from django.utils.html import format_html from lemarche.conversations.models import Conversation, TemplateTransactional, TemplateTransactionalSendLog from lemarche.utils.admin.admin_site import admin_site -from lemarche.utils.fields import pretty_print_readonly_jsonfield_to_table +from lemarche.utils.fields import pretty_print_readonly_jsonfield, pretty_print_readonly_jsonfield_to_table from lemarche.www.conversations.tasks import send_first_email_from_conversation @@ -134,27 +136,66 @@ def data_display(self, conversation: Conversation = None): @admin.register(TemplateTransactional, site=admin_site) class TemplateTransactionalAdmin(admin.ModelAdmin): - list_display = ["id", "name", "code", "mailjet_id", "brevo_id", "source", "is_active", "created_at", "updated_at"] + list_display = [ + "id", + "name", + "code", + "mailjet_id", + "brevo_id", + "source", + "is_active", + "template_transactional_send_log_count_with_link", + "created_at", + "updated_at", + ] search_fields = ["id", "name", "code", "mailjet_id", "brevo_id"] - readonly_fields = ["code", "created_at", "updated_at"] + readonly_fields = ["code", "template_transactional_send_log_count_with_link", "created_at", "updated_at"] fieldsets = ( (None, {"fields": ("name", "code", "description")}), ("Paramètres d'envoi", {"fields": ("mailjet_id", "brevo_id", "source", "is_active")}), + ("Stats", {"fields": ("template_transactional_send_log_count_with_link",)}), ("Dates", {"fields": ("created_at", "updated_at")}), ) + def get_queryset(self, request): + qs = super().get_queryset(request) + qs = qs.with_stats() + return qs + + def template_transactional_send_log_count_with_link(self, obj): + url = ( + reverse("admin:conversations_templatetransactionalsendlog_changelist") + + f"?template_transactionals__in={obj.id}" + ) + return format_html(f'{obj.send_log_count}') + + template_transactional_send_log_count_with_link.short_description = "Logs d'envois" + template_transactional_send_log_count_with_link.admin_order_field = "send_log_count" + @admin.register(TemplateTransactionalSendLog, site=admin_site) class TemplateTransactionalSendLogAdmin(admin.ModelAdmin): - list_display = ["id", "template_transactional", "content_type", "created_at"] - list_filter = [("content_type", admin.RelatedOnlyFieldListFilter)] + list_display = ["id", "template_transactional", "recipient_content_type", "parent_content_type", "created_at"] + list_filter = [ + "template_transactional", + ("recipient_content_type", admin.RelatedOnlyFieldListFilter), + ("parent_content_type", admin.RelatedOnlyFieldListFilter), + ] search_fields = ["id", "template_transactional"] search_help_text = "Cherche sur les champs : ID, Template transactionnel" readonly_fields = [field.name for field in TemplateTransactionalSendLog._meta.fields] + fieldsets = ( + (None, {"fields": ("template_transactional",)}), + ("Envoyé à…", {"fields": ("recipient_content_type", "recipient_object_id_with_link")}), + ("Contexte…", {"fields": ("parent_content_type", "parent_object_id_with_link")}), + ("Logs", {"fields": ("extra_data_display",)}), + ("Dates", {"fields": ("created_at", "updated_at")}), + ) + def has_add_permission(self, request): return False @@ -163,3 +204,36 @@ def has_change_permission(self, request, obj=None): def has_delete_permission(self, request, obj=None): return False + + def recipient_object_id_with_link(self, obj): + if obj.recipient_content_type and obj.recipient_object_id: + if obj.recipient_content_type.model == "tender": + url = reverse("admin:tenders_tender_change", args=[obj.recipient_object_id]) + return format_html(f'{obj.recipient_object_id}') + elif obj.recipient_content_type.model == "siae": + url = reverse("admin:siaes_siae_change", args=[obj.recipient_object_id]) + return format_html(f'{obj.recipient_object_id}') + elif obj.recipient_content_type.model == "user": + url = reverse("admin:users_user_change", args=[obj.recipient_object_id]) + return format_html(f'{obj.recipient_object_id}') + return obj.recipient_object.id + + def parent_object_id_with_link(self, obj): + if obj.parent_content_type and obj.parent_object_id: + if obj.parent_content_type.model == "tender": + url = reverse("admin:tenders_tender_change", args=[obj.parent_object_id]) + return format_html(f'{obj.parent_object_id}') + if obj.parent_content_type.model == "tendersiae": + url = reverse("admin:tenders_tendersiae_change", args=[obj.parent_object_id]) + return format_html(f'{obj.parent_object_id}') + elif obj.parent_content_type.model == "siaeuserrequest": + url = reverse("admin:siaes_siaeuserrequest_change", args=[obj.parent_object_id]) + return format_html(f'{obj.parent_object_id}') + return obj.context_object.id + + def extra_data_display(self, instance: TemplateTransactionalSendLog = None): + if instance: + return pretty_print_readonly_jsonfield(instance.extra_data) + return "-" + + extra_data_display.short_description = TemplateTransactionalSendLog._meta.get_field("extra_data").verbose_name diff --git a/lemarche/conversations/migrations/0016_templatetransactionalsendlog.py b/lemarche/conversations/migrations/0016_templatetransactionalsendlog.py index 27db331ed..cf6413edb 100644 --- a/lemarche/conversations/migrations/0016_templatetransactionalsendlog.py +++ b/lemarche/conversations/migrations/0016_templatetransactionalsendlog.py @@ -1,4 +1,4 @@ -# Generated by Django 4.2.13 on 2024-07-15 10:43 +# Generated by Django 4.2.13 on 2024-07-22 10:30 import django.db.models.deletion import django.utils.timezone @@ -16,7 +16,8 @@ class Migration(migrations.Migration): name="TemplateTransactionalSendLog", fields=[ ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("object_id", models.PositiveBigIntegerField(blank=True, null=True)), + ("recipient_object_id", models.PositiveBigIntegerField(blank=True, null=True)), + ("parent_object_id", models.PositiveBigIntegerField(blank=True, null=True)), ("extra_data", models.JSONField(default=dict, editable=False, verbose_name="Données complémentaires")), ( "created_at", @@ -24,11 +25,22 @@ class Migration(migrations.Migration): ), ("updated_at", models.DateTimeField(auto_now=True, verbose_name="Date de modification")), ( - "content_type", + "parent_content_type", models.ForeignKey( blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, + related_name="parent_send_logs", + to="contenttypes.contenttype", + ), + ), + ( + "recipient_content_type", + models.ForeignKey( + blank=True, + null=True, + on_delete=django.db.models.deletion.CASCADE, + related_name="recipient_send_logs", to="contenttypes.contenttype", ), ), diff --git a/lemarche/conversations/models.py b/lemarche/conversations/models.py index bdadea13f..0114706a5 100644 --- a/lemarche/conversations/models.py +++ b/lemarche/conversations/models.py @@ -5,7 +5,7 @@ from django.contrib.contenttypes.fields import GenericForeignKey from django.contrib.contenttypes.models import ContentType from django.db import IntegrityError, models -from django.db.models import Func, IntegerField, Q +from django.db.models import Count, Func, IntegerField, Q from django.utils import timezone from django.utils.text import slugify from django_extensions.db.fields import ShortUUIDField @@ -198,6 +198,13 @@ def set_validated(self): self.save() +class TemplateTransactionalQuerySet(models.QuerySet): + def with_stats(self): + return self.annotate( + send_log_count=Count("send_logs"), + ) + + class TemplateTransactional(models.Model): name = models.CharField(verbose_name="Nom", max_length=255) code = models.CharField( @@ -238,6 +245,8 @@ class TemplateTransactional(models.Model): created_at = models.DateTimeField(verbose_name="Date de création", default=timezone.now) updated_at = models.DateTimeField(verbose_name="Date de modification", auto_now=True) + objects = models.Manager.from_queryset(TemplateTransactionalQuerySet)() + class Meta: verbose_name = "Template transactionnel" verbose_name_plural = "Templates transactionnels" @@ -267,6 +276,8 @@ def send_transactional_email( subject=None, from_email=settings.DEFAULT_FROM_EMAIL, from_name=settings.DEFAULT_FROM_NAME, + recipient_content_object=None, + parent_content_object=None, ): if self.is_active: args = { @@ -283,7 +294,11 @@ def send_transactional_email( elif self.source == conversation_constants.SOURCE_BREVO: result = api_brevo.send_transactional_email_with_template(**args) # create log - self.create_send_log(extra_data={"source": self.source, "args": args, "response": result()}) + self.create_send_log( + recipient_content_object=recipient_content_object, + parent_content_object=parent_content_object, + extra_data={"source": self.source, "args": args, "response": result()}, + ) class TemplateTransactionalSendLog(models.Model): @@ -295,9 +310,19 @@ class TemplateTransactionalSendLog(models.Model): related_name="send_logs", ) - content_type = models.ForeignKey(ContentType, blank=True, null=True, on_delete=models.CASCADE) - object_id = models.PositiveBigIntegerField(blank=True, null=True) - content_object = GenericForeignKey("content_type", "object_id") + # the object that is the recipient of the email (User, Siae...) + recipient_content_type = models.ForeignKey( + ContentType, blank=True, null=True, on_delete=models.CASCADE, related_name="recipient_send_logs" + ) + recipient_object_id = models.PositiveBigIntegerField(blank=True, null=True) + recipient_content_object = GenericForeignKey("recipient_content_type", "recipient_object_id") + + # the object that is the parent of the email (TenderSiae, SiaeUserRequest...) + parent_content_type = models.ForeignKey( + ContentType, blank=True, null=True, on_delete=models.CASCADE, related_name="parent_send_logs" + ) + parent_object_id = models.PositiveBigIntegerField(blank=True, null=True) + parent_content_object = GenericForeignKey("parent_content_type", "parent_object_id") extra_data = models.JSONField(verbose_name="Données complémentaires", editable=False, default=dict) diff --git a/lemarche/notes/admin.py b/lemarche/notes/admin.py index 052db5306..4a52876c3 100644 --- a/lemarche/notes/admin.py +++ b/lemarche/notes/admin.py @@ -26,12 +26,7 @@ class NoteAdmin(admin.ModelAdmin): formfield_overrides = {models.TextField: {"widget": CKEditorWidget(config_name="admin_note_text")}} fieldsets = ( - ( - None, - { - "fields": ("text", "author"), - }, - ), + (None, {"fields": ("text", "author")}), ("Rattachée à…", {"fields": ("content_type", "object_id_with_link")}), ("Dates", {"fields": ("created_at", "updated_at")}), ) @@ -41,15 +36,15 @@ def save_model(self, request, obj: Note, form, change): obj.author = request.user obj.save() - def object_id_with_link(self, note): - if note.content_type and note.object_id: - if note.content_type.model == "tender": - url = reverse("admin:tenders_tender_change", args=[note.object_id]) - return format_html(f'{note.object_id}') - elif note.content_type.model == "siae": - url = reverse("admin:siaes_siae_change", args=[note.object_id]) - return format_html(f'{note.object_id}') - elif note.content_type.model == "user": - url = reverse("admin:users_user_change", args=[note.object_id]) - return format_html(f'{note.object_id}') - return note.object.id + def object_id_with_link(self, obj): + if obj.content_type and obj.object_id: + if obj.content_type.model == "tender": + url = reverse("admin:tenders_tender_change", args=[obj.object_id]) + return format_html(f'{obj.object_id}') + elif obj.content_type.model == "siae": + url = reverse("admin:siaes_siae_change", args=[obj.object_id]) + return format_html(f'{obj.object_id}') + elif obj.content_type.model == "user": + url = reverse("admin:users_user_change", args=[obj.object_id]) + return format_html(f'{obj.object_id}') + return obj.object.id diff --git a/lemarche/siaes/admin.py b/lemarche/siaes/admin.py index 4fbdc2709..15f91c2c8 100644 --- a/lemarche/siaes/admin.py +++ b/lemarche/siaes/admin.py @@ -9,7 +9,7 @@ from fieldsets_with_inlines import FieldsetsInlineMixin from simple_history.admin import SimpleHistoryAdmin -from lemarche.conversations.models import Conversation +from lemarche.conversations.models import Conversation, TemplateTransactionalSendLog from lemarche.labels.models import Label from lemarche.networks.models import Network from lemarche.notes.models import Note @@ -212,6 +212,7 @@ class SiaeAdmin(FieldsetsInlineMixin, gis_admin.OSMGeoAdmin, SimpleHistoryAdmin) "tender_detail_display_count_annotated_with_link", "tender_detail_contact_click_count_annotated_with_link", "tender_detail_not_interested_count_annotated_with_link", + "recipient_transactional_send_logs_count_with_link", "brevo_company_id", "extra_data_display", "import_raw_object_display", @@ -362,6 +363,7 @@ class SiaeAdmin(FieldsetsInlineMixin, gis_admin.OSMGeoAdmin, SimpleHistoryAdmin) "fields": ( "signup_date", "content_filled_basic_date", + "recipient_transactional_send_logs_count_with_link", "brevo_company_id", "extra_data_display", ), @@ -665,6 +667,14 @@ def tender_detail_not_interested_count_annotated_with_link(self, siae): "tender_detail_not_interested_count_annotated" ) + def recipient_transactional_send_logs_count_with_link(self, obj): + url = reverse("admin:conversations_templatetransactionalsendlog_changelist") + f"?siae__id__exact={obj.id}" + return format_html(f'{obj.recipient_transactional_send_logs.count()}') + + recipient_transactional_send_logs_count_with_link.short_description = ( + TemplateTransactionalSendLog._meta.verbose_name + ) + def logs_display(self, siae=None): if siae: return pretty_print_readonly_jsonfield(siae.logs) @@ -700,13 +710,6 @@ class SiaeUserRequestAdmin(admin.ModelAdmin): readonly_fields = [field.name for field in SiaeUserRequest._meta.fields] fields = ["logs_display" if field_name == "logs" else field_name for field_name in readonly_fields] - def logs_display(self, siaeuserrequest=None): - if siaeuserrequest: - return pretty_print_readonly_jsonfield(siaeuserrequest.logs) - return "-" - - logs_display.short_description = SiaeUserRequest._meta.get_field("logs").verbose_name - def has_add_permission(self, request): return False @@ -716,6 +719,22 @@ def has_change_permission(self, request, obj=None): def has_delete_permission(self, request, obj=None): return False + def parent_transactional_send_logs_count_with_link(self, obj): + url = ( + reverse("admin:conversations_templatetransactionalsendlog_changelist") + + f"?siaeuserrequest__id__exact={obj.id}" + ) + return format_html(f'{obj.parent_transactional_send_logs.count()}') + + parent_transactional_send_logs_count_with_link.short_description = TemplateTransactionalSendLog._meta.verbose_name + + def logs_display(self, siaeuserrequest=None): + if siaeuserrequest: + return pretty_print_readonly_jsonfield(siaeuserrequest.logs) + return "-" + + logs_display.short_description = SiaeUserRequest._meta.get_field("logs").verbose_name + @admin.register(SiaeActivity, site=admin_site) class SiaeActivityAdmin(admin.ModelAdmin): diff --git a/lemarche/siaes/models.py b/lemarche/siaes/models.py index 8929fd1ae..83d8c8230 100644 --- a/lemarche/siaes/models.py +++ b/lemarche/siaes/models.py @@ -871,7 +871,12 @@ class Siae(models.Model): "Nombre de besoins intéressés", help_text=RECALCULATED_FIELD_HELP_TEXT, default=0 ) logs = models.JSONField(verbose_name="Logs historiques", editable=False, default=list) - transactional_send_logs = GenericRelation("conversations.TemplateTransactionalSendLog", related_query_name="siae") + recipient_transactional_send_logs = GenericRelation( + "conversations.TemplateTransactionalSendLog", + related_query_name="siae", + content_type_field="recipient_content_type", + object_id_field="recipient_object_id", + ) source = models.CharField( max_length=20, choices=siae_constants.SOURCE_CHOICES, default=siae_constants.SOURCE_STAFF_C4_CREATED ) @@ -1381,6 +1386,12 @@ class SiaeUserRequest(models.Model): response = models.BooleanField(verbose_name="Réponse", blank=True, null=True) response_date = models.DateTimeField("Date de la réponse", blank=True, null=True) + parent_transactional_send_logs = GenericRelation( + "conversations.TemplateTransactionalSendLog", + related_query_name="siaeuserrequest", + content_type_field="parent_content_type", + object_id_field="parent_object_id", + ) logs = models.JSONField(verbose_name="Logs des échanges", editable=False, default=list) created_at = models.DateTimeField(verbose_name="Date de création", default=timezone.now) diff --git a/lemarche/siaes/tasks.py b/lemarche/siaes/tasks.py index 2779fe1b4..2413b4ca6 100644 --- a/lemarche/siaes/tasks.py +++ b/lemarche/siaes/tasks.py @@ -43,7 +43,9 @@ def send_completion_reminder_email_to_siae(siae): recipient_name = siae_user.full_name variables = { + "SIAE_USER_ID": siae_user.id, "SIAE_USER_FIRST_NAME": siae_user.first_name, + "SIAE_ID": siae.id, "SIAE_NAME": siae.name_display, "SIAE_URL": get_object_share_url(siae), "SIAE_EDIT_URL": f"https://{get_domain_url()}{reverse_lazy('dashboard_siaes:siae_edit_contact', args=[siae.slug])}", # noqa @@ -53,6 +55,8 @@ def send_completion_reminder_email_to_siae(siae): recipient_email=recipient_email, recipient_name=recipient_name, variables=variables, + recipient_content_object=siae_user, + parent_content_object=siae, ) # log email diff --git a/lemarche/tenders/admin.py b/lemarche/tenders/admin.py index 2f1d5179c..c5c06b594 100644 --- a/lemarche/tenders/admin.py +++ b/lemarche/tenders/admin.py @@ -13,6 +13,7 @@ from django_better_admin_arrayfield.admin.mixins import DynamicArrayMixin from fieldsets_with_inlines import FieldsetsInlineMixin +from lemarche.conversations.models import TemplateTransactionalSendLog from lemarche.notes.models import Note from lemarche.perimeters.admin import PerimeterRegionFilter from lemarche.perimeters.models import Perimeter @@ -339,6 +340,7 @@ class TenderAdmin(FieldsetsInlineMixin, admin.ModelAdmin): "siae_transactioned_source", "siae_transactioned_last_updated", "source", + "parent_transactional_send_logs_count_with_link", "brevo_deal_id", "extra_data_display", "import_raw_object_display", @@ -511,6 +513,7 @@ class TenderAdmin(FieldsetsInlineMixin, admin.ModelAdmin): "marche_benefits", "siae_list_last_seen_date", "source", + "parent_transactional_send_logs_count_with_link", "brevo_deal_id", "extra_data_display", ), @@ -759,6 +762,12 @@ def siae_detail_not_interested_click_count_annotated_with_link(self, tender): "siae_detail_not_interested_click_count_annotated" ) + def parent_transactional_send_logs_count_with_link(self, obj): + url = reverse("admin:conversations_templatetransactionalsendlog_changelist") + f"?tender__id__exact={obj.id}" + return format_html(f'{obj.parent_transactional_send_logs.count()}') + + parent_transactional_send_logs_count_with_link.short_description = TemplateTransactionalSendLog._meta.verbose_name + def logs_display(self, tender=None): if tender: return pretty_print_readonly_jsonfield(tender.logs) @@ -950,6 +959,7 @@ class TenderSiaeAdmin(admin.ModelAdmin): "tender", "tender_with_link", "transactioned_source", + "parent_transactional_send_logs_count_with_link", "logs_display", ] @@ -964,7 +974,15 @@ class TenderSiaeAdmin(admin.ModelAdmin): {"fields": (*TenderSiae.FIELDS_SURVEY_TRANSACTIONED, "transactioned", "transactioned_source")}, ), ("Utilisateur", {"fields": ("user",)}), - ("Stats", {"fields": ("logs_display",)}), + ( + "Stats", + { + "fields": ( + "parent_transactional_send_logs_count_with_link", + "logs_display", + ) + }, + ), ("Dates", {"fields": ("created_at", "updated_at")}), ) @@ -1000,6 +1018,14 @@ def status(self, tendersiae): status.short_description = "Status" status.admin_order_field = "status" + def parent_transactional_send_logs_count_with_link(self, obj): + url = ( + reverse("admin:conversations_templatetransactionalsendlog_changelist") + f"?tendersiae__id__exact={obj.id}" + ) + return format_html(f'{obj.parent_transactional_send_logs.count()}') + + parent_transactional_send_logs_count_with_link.short_description = TemplateTransactionalSendLog._meta.verbose_name + def logs_display(self, tender=None): if tender: return pretty_print_readonly_jsonfield(tender.logs) diff --git a/lemarche/tenders/models.py b/lemarche/tenders/models.py index 0e471aa42..3960b74f3 100644 --- a/lemarche/tenders/models.py +++ b/lemarche/tenders/models.py @@ -676,6 +676,13 @@ class Tender(models.Model): related_name="tenders_admins", blank=True, ) + + parent_transactional_send_logs = GenericRelation( + "conversations.TemplateTransactionalSendLog", + related_query_name="tender", + content_type_field="parent_content_type", + object_id_field="parent_object_id", + ) logs = models.JSONField(verbose_name="Logs historiques", editable=False, default=list) source = models.CharField( verbose_name="Source", @@ -1189,6 +1196,12 @@ class TenderSiae(models.Model): null=True, ) + parent_transactional_send_logs = GenericRelation( + "conversations.TemplateTransactionalSendLog", + related_query_name="tendersiae", + content_type_field="parent_content_type", + object_id_field="parent_object_id", + ) logs = models.JSONField(verbose_name="Logs historiques", editable=False, default=list) created_at = models.DateTimeField(verbose_name="Date de création", default=timezone.now) diff --git a/lemarche/users/admin.py b/lemarche/users/admin.py index b1307fc0e..9f9f71343 100644 --- a/lemarche/users/admin.py +++ b/lemarche/users/admin.py @@ -9,11 +9,13 @@ from django.utils.html import format_html, mark_safe from fieldsets_with_inlines import FieldsetsInlineMixin +from lemarche.conversations.models import TemplateTransactionalSendLog from lemarche.notes.models import Note from lemarche.siaes.models import Siae, SiaeUser from lemarche.users.forms import UserChangeForm, UserCreationForm from lemarche.users.models import User from lemarche.utils.admin.admin_site import admin_site +from lemarche.utils.fields import pretty_print_readonly_jsonfield class HasCompanyFilter(admin.SimpleListFilter): @@ -194,8 +196,9 @@ class UserAdmin(FieldsetsInlineMixin, UserAdmin): "favorite_list_count_with_link", "image_url", "image_url_display", + "recipient_transactional_send_logs_count_with_link", "brevo_contact_id", - "extra_data", + "extra_data_display", ] ) @@ -283,7 +286,13 @@ class UserAdmin(FieldsetsInlineMixin, UserAdmin): "Stats", { "classes": ["collapse"], - "fields": ("dashboard_last_seen_date", "tender_list_last_seen_date", "brevo_contact_id", "extra_data"), + "fields": ( + "dashboard_last_seen_date", + "tender_list_last_seen_date", + "recipient_transactional_send_logs_count_with_link", + "brevo_contact_id", + "extra_data_display", + ), }, ), ( @@ -384,3 +393,18 @@ def image_url_display(self, user): return mark_safe("