From 5d2076c1c3b84795dae409112c07c45cb933d7cb Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Wed, 30 Oct 2024 10:48:03 +0100 Subject: [PATCH 01/32] add notification with eventmanager --- .../proposals_controller_override.rb | 27 ++++++++++++++++ .../proposals/proposal_published_event.rb | 31 +++++++++++++++++++ config/locales/en.yml | 5 +++ 3 files changed, 63 insertions(+) create mode 100644 app/events/decidim/proposals/proposal_published_event.rb diff --git a/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb b/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb index b627ce8e14..5181a1f412 100644 --- a/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb +++ b/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb @@ -103,6 +103,23 @@ def create end end end + def publish + enforce_permission_to :edit, :proposal, proposal: @proposal + + # Publier la proposition et envoyer la notification si succès + Decidim::Proposals::PublishProposal.call(@proposal, current_user) do + on(:ok) do + flash[:notice] = I18n.t("proposals.publish.success", scope: "decidim") + send_publication_notification # Appel de la méthode de notification + redirect_to proposal_path(@proposal) + end + + on(:invalid) do + flash.now[:alert] = I18n.t("proposals.publish.error", scope: "decidim") + render :edit_draft + end + end + end # Overridden because of a core bug when the command posts the "invalid" # signal and when rendering the form. @@ -203,6 +220,16 @@ def proposal_limit_reached?(form = form_proposal_params) def current_user_proposals(form) Decidim::Proposals::Proposal.from_author(current_user).where(component: form.current_component).except_withdrawn end + + def send_publication_notification + Decidim::EventsManager.publish( + event: "decidim.proposals.proposal_published", + event_class: Decidim::Proposals::ProposalPublishedEvent, + resource: @proposal, + affected_users: [@proposal.creator_identity], + extra: { participatory_space: true } + ) + end end end end diff --git a/app/events/decidim/proposals/proposal_published_event.rb b/app/events/decidim/proposals/proposal_published_event.rb new file mode 100644 index 0000000000..5eaf0fa74e --- /dev/null +++ b/app/events/decidim/proposals/proposal_published_event.rb @@ -0,0 +1,31 @@ +# app/events/decidim/proposals/proposal_published_event.rb +module Decidim + module Proposals + class ProposalPublishedEvent < Decidim::Events::BaseEvent + include Decidim::Events::NotificationEvent + + def initialize(resource:, **options) + @resource = resource + super(**options) + end + + def notification_title + I18n.t("decidim.proposals.notifications.proposal_published.title", proposal_title: resource_title) + end + + def notification_body + I18n.t("decidim.proposals.notifications.proposal_published.body", proposal_link: resource_path) + end + + private + + def resource_title + translated_attribute(@resource.title) + end + + def resource_path + Decidim::Engine.routes.url_helpers.proposal_path(@resource) + end + end + end +end diff --git a/config/locales/en.yml b/config/locales/en.yml index 477839a9f7..f64f878fc5 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -147,6 +147,11 @@ en: exports: awesome_private_proposals: Proposals with private fields proposal_comments: Comments + notifications: + proposal_published: + title: "Your proposal %{proposal_title} has been published!" + body: "Your proposal is now live. View it here: %{proposal_link}" + subject: "Your proposal has been published!" collaborative_drafts: new: add_file: Add file From b4ef5667df7d8a0b18ce3c0efca9a8387d65e57f Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Mon, 4 Nov 2024 16:19:01 +0100 Subject: [PATCH 02/32] base to watch the CI and see files on github --- .../proposals_controller_override.rb | 27 ------------------- .../proposals/proposal_published_event.rb | 9 ++----- 2 files changed, 2 insertions(+), 34 deletions(-) diff --git a/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb b/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb index 5181a1f412..51fcfe5cb3 100644 --- a/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb +++ b/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb @@ -103,24 +103,6 @@ def create end end end - def publish - enforce_permission_to :edit, :proposal, proposal: @proposal - - # Publier la proposition et envoyer la notification si succès - Decidim::Proposals::PublishProposal.call(@proposal, current_user) do - on(:ok) do - flash[:notice] = I18n.t("proposals.publish.success", scope: "decidim") - send_publication_notification # Appel de la méthode de notification - redirect_to proposal_path(@proposal) - end - - on(:invalid) do - flash.now[:alert] = I18n.t("proposals.publish.error", scope: "decidim") - render :edit_draft - end - end - end - # Overridden because of a core bug when the command posts the "invalid" # signal and when rendering the form. def update_draft @@ -221,15 +203,6 @@ def current_user_proposals(form) Decidim::Proposals::Proposal.from_author(current_user).where(component: form.current_component).except_withdrawn end - def send_publication_notification - Decidim::EventsManager.publish( - event: "decidim.proposals.proposal_published", - event_class: Decidim::Proposals::ProposalPublishedEvent, - resource: @proposal, - affected_users: [@proposal.creator_identity], - extra: { participatory_space: true } - ) - end end end end diff --git a/app/events/decidim/proposals/proposal_published_event.rb b/app/events/decidim/proposals/proposal_published_event.rb index 5eaf0fa74e..4f68c2ca6e 100644 --- a/app/events/decidim/proposals/proposal_published_event.rb +++ b/app/events/decidim/proposals/proposal_published_event.rb @@ -4,11 +4,6 @@ module Proposals class ProposalPublishedEvent < Decidim::Events::BaseEvent include Decidim::Events::NotificationEvent - def initialize(resource:, **options) - @resource = resource - super(**options) - end - def notification_title I18n.t("decidim.proposals.notifications.proposal_published.title", proposal_title: resource_title) end @@ -20,11 +15,11 @@ def notification_body private def resource_title - translated_attribute(@resource.title) + translated_attribute(resource.title) end def resource_path - Decidim::Engine.routes.url_helpers.proposal_path(@resource) + Decidim::Engine.routes.url_helpers.proposal_path(resource) end end end From 490cc23eb9ea1c0465ec6ba17006973ec609b1c8 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Wed, 6 Nov 2024 13:29:11 +0100 Subject: [PATCH 03/32] fix translation key & notififaction displaying --- .../decidim/proposals/proposal_published_event.rb | 10 +++++++++- config/locales/fr.yml | 5 +++++ 2 files changed, 14 insertions(+), 1 deletion(-) diff --git a/app/events/decidim/proposals/proposal_published_event.rb b/app/events/decidim/proposals/proposal_published_event.rb index 4f68c2ca6e..2fe8cf8bb7 100644 --- a/app/events/decidim/proposals/proposal_published_event.rb +++ b/app/events/decidim/proposals/proposal_published_event.rb @@ -4,6 +4,10 @@ module Proposals class ProposalPublishedEvent < Decidim::Events::BaseEvent include Decidim::Events::NotificationEvent + def self.model_name + ActiveModel::Name.new(self, nil, I18n.t('decidim.proposals.notifications.proposal_published.subject')) + end + def notification_title I18n.t("decidim.proposals.notifications.proposal_published.title", proposal_title: resource_title) end @@ -19,7 +23,11 @@ def resource_title end def resource_path - Decidim::Engine.routes.url_helpers.proposal_path(resource) + Decidim::Proposals::Engine.routes.url_helpers.proposal_path( + resource, + component_id: resource.component.id, + initiative_slug: resource.component.participatory_space.slug + ) end end end diff --git a/config/locales/fr.yml b/config/locales/fr.yml index a82163e17b..58f4874a10 100644 --- a/config/locales/fr.yml +++ b/config/locales/fr.yml @@ -158,6 +158,11 @@ fr: collaborative_drafts_list: Accéder aux brouillons collaboratifs new_proposal: Nouvelle proposition view_proposal: Voir la proposition + notifications: + proposal_published: + title: "Votre proposition %{proposal_title} a bien été publiée!" + body: "Votre proposition est en ligne. Lien ici: %{proposal_link}" + subject: "Votre proposition est publiée!" update: error: Il y a eu une erreur lors de la mise à jour de la proposition. success: Proposition mise à jour avec succès. From f8b938bf2f04ed1fb1ddea93a7edf3c011e3effd Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Wed, 6 Nov 2024 13:45:02 +0100 Subject: [PATCH 04/32] add send_pubication_notification to right file --- .../decidim/proposals/publish_proposal.rb | 118 ++++++++++++++++++ 1 file changed, 118 insertions(+) create mode 100644 app/commands/decidim/proposals/publish_proposal.rb diff --git a/app/commands/decidim/proposals/publish_proposal.rb b/app/commands/decidim/proposals/publish_proposal.rb new file mode 100644 index 0000000000..4e6f2860f1 --- /dev/null +++ b/app/commands/decidim/proposals/publish_proposal.rb @@ -0,0 +1,118 @@ +# frozen_string_literal: true + +module Decidim + module Proposals + # A command with all the business logic when a user publishes a draft proposal. + class PublishProposal < Decidim::Command + # Public: Initializes the command. + # + # proposal - The proposal to publish. + # current_user - The current user. + def initialize(proposal, current_user) + @proposal = proposal + @current_user = current_user + end + + # Executes the command. Broadcasts these events: + # + # - :ok when everything is valid and the proposal is published. + # - :invalid if the proposal's author is not the current user. + # + # Returns nothing. + def call + return broadcast(:invalid) unless @proposal.authored_by?(@current_user) + + transaction do + publish_proposal + increment_scores + send_notification + send_notification_to_participatory_space + send_publication_notification + end + + broadcast(:ok, @proposal) + end + + private + + # This will be the PaperTrail version that is + # shown in the version control feature (1 of 1) + # + # For an attribute to appear in the new version it has to be reset + # and reassigned, as PaperTrail only keeps track of object CHANGES. + def publish_proposal + title = reset(:title) + body = reset(:body) + + Decidim.traceability.perform_action!( + "publish", + @proposal, + @current_user, + visibility: "public-only" + ) do + @proposal.update title: title, body: body, published_at: Time.current + end + end + + # Reset the attribute to an empty string and return the old value + def reset(attribute) + attribute_value = @proposal[attribute] + PaperTrail.request(enabled: false) do + # rubocop:disable Rails/SkipsModelValidations + @proposal.update_attribute attribute, "" + # rubocop:enable Rails/SkipsModelValidations + end + attribute_value + end + + def send_notification + return if @proposal.coauthorships.empty? + + Decidim::EventsManager.publish( + event: "decidim.events.proposals.proposal_published", + event_class: Decidim::Proposals::PublishProposalEvent, + resource: @proposal, + followers: coauthors_followers + ) + end + + def send_publication_notification + puts "send notification from proposal #{@proposal.id}" + # debug message + Decidim::EventsManager.publish( + event: "decidim.proposals.proposal_published", + event_class: Decidim::Proposals::ProposalPublishedEvent, + resource: @proposal, + affected_users: [@proposal.creator_identity], + extra: { participatory_space: true } + ) + end + + def send_notification_to_participatory_space + Decidim::EventsManager.publish( + event: "decidim.events.proposals.proposal_published", + event_class: Decidim::Proposals::PublishProposalEvent, + resource: @proposal, + followers: @proposal.participatory_space.followers - coauthors_followers, + extra: { + participatory_space: true + } + ) + end + + def coauthors_followers + @coauthors_followers ||= @proposal.authors.flat_map(&:followers) + end + + def increment_scores + @proposal.coauthorships.find_each do |coauthorship| + if coauthorship.user_group + Decidim::Gamification.increment_score(coauthorship.user_group, :proposals) + else + Decidim::Gamification.increment_score(coauthorship.author, :proposals) + end + end + end + end + end +end From a28f39a0678348faa237a343fafa5b56bbe87aa1 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Wed, 6 Nov 2024 13:46:25 +0100 Subject: [PATCH 05/32] start test rspec --- .../proposal_published_event_spec.rb | 73 +++++++++++++++++++ 1 file changed, 73 insertions(+) create mode 100644 spec/events/decidim/proposals/proposal_published_event_spec.rb diff --git a/spec/events/decidim/proposals/proposal_published_event_spec.rb b/spec/events/decidim/proposals/proposal_published_event_spec.rb new file mode 100644 index 0000000000..4b5ddb61ee --- /dev/null +++ b/spec/events/decidim/proposals/proposal_published_event_spec.rb @@ -0,0 +1,73 @@ +# frozen_string_literal: true + +require "spec_helper" + +module Decidim::Proposals + describe "ProposalPublishedEvent", type: :job do + include MailerHelpers + + let(:organization) { create(:organization) } + let(:participatory_process) { create(:participatory_process, organization: organization) } + let(:component) { create(:component, manifest_name: :proposals, participatory_space: participatory_process) } + let!(:proposal_state) { create(:proposal_state, component: component) } + let(:proposal) { create(:proposal, component: component, proposal_state: proposal_state, title: { en: "Test proposition notification" }) } + let(:user) { create(:user, :admin, organization: organization, notifications_sending_frequency: "daily") } + + before do + clear_emails + end + + context "when proposal is published" do + it "sends a notification email with the correct content" do + Decidim::EventsManager.publish( + event: "decidim.proposals.proposal_published", + event_class: ProposalPublishedEvent, + resource: proposal, + affected_users: [user], + ) + + perform_enqueued_jobs do + Decidim::EmailNotificationsDigestGeneratorJob.perform_now(user.id, user.notifications_sending_frequency) + end + + expect(last_email).not_to be_nil + expect(last_email_body).not_to include("translation missing") + expect(last_email_body).to include("Test proposition notification") + + end + end + + end +end +# +# shared_context "when sends the notification digest" do +# context "when daily notification mail" do +# let(:user) { create(:user, :admin, organization:, notifications_sending_frequency: "daily") } +# +# it_behaves_like "notification digest mail" +# end +# +# context "when weekly notification mail" do +# let(:user) { create(:user, :admin, organization:, notifications_sending_frequency: "weekly") } +# +# it_behaves_like "notification digest mail" +# end +# end +# +# shared_examples_for "notification digest mail" do +# context "when a notifiable event takes place" do +# let!(:organization) { create(:organization) } +# let!(:participatory_space) { create(:participatory_process, organization:) } +# +# it "sends a notification to the user's email" do +# perform_enqueued_jobs do +# expect(command.call).to broadcast(:ok) +# Decidim::Notification.last.update(created_at: 1.day.ago) +# Decidim::EmailNotificationsDigestGeneratorJob.perform_now(user.id, user.notifications_sending_frequency) +# end +# +# expect(last_email_body.length).to be_positive +# expect(last_email_body).not_to include("translation missing") +# end +# end +# end From 4c9be63015037e9573659e7c230874fb46995cc2 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Wed, 6 Nov 2024 17:05:02 +0100 Subject: [PATCH 06/32] continuing rspec --- .../proposal_published_event_spec.rb | 25 ++++++++++--------- 1 file changed, 13 insertions(+), 12 deletions(-) diff --git a/spec/events/decidim/proposals/proposal_published_event_spec.rb b/spec/events/decidim/proposals/proposal_published_event_spec.rb index 4b5ddb61ee..8ec2b28d2c 100644 --- a/spec/events/decidim/proposals/proposal_published_event_spec.rb +++ b/spec/events/decidim/proposals/proposal_published_event_spec.rb @@ -1,6 +1,7 @@ # frozen_string_literal: true require "spec_helper" +require "nokogiri" module Decidim::Proposals describe "ProposalPublishedEvent", type: :job do @@ -18,25 +19,25 @@ module Decidim::Proposals end context "when proposal is published" do - it "sends a notification email with the correct content" do - Decidim::EventsManager.publish( - event: "decidim.proposals.proposal_published", - event_class: ProposalPublishedEvent, - resource: proposal, - affected_users: [user], - ) - - perform_enqueued_jobs do - Decidim::EmailNotificationsDigestGeneratorJob.perform_now(user.id, user.notifications_sending_frequency) - end + it "sends a notification email with the correct content" do + subject { ProposalPublishedEvent.new(resource: proposal, event_name: "decidim.proposals.proposal_published", user: proposal.creator_identity) } expect(last_email).not_to be_nil expect(last_email_body).not_to include("translation missing") expect(last_email_body).to include("Test proposition notification") + end + + it "sends a notification with the correct content" do + subject { ProposalPublishedEvent.new(resource: proposal, event_name: "decidim.proposals.proposal_published", user: proposal.creator_identity) } + notification = Decidim::Notification.last + # Parse avec Nokogiri + html_body = Nokogiri::HTML(notification.extra["body"]) + text = html_body.text + expect(text).to eq("Your proposal Test proposition notification has been published") + expect(notification).not_to be_nil end end - end end # From eb00e654b56787bcb3729b57aacf229f9ae8f9dc Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Wed, 6 Nov 2024 17:21:34 +0100 Subject: [PATCH 07/32] potential final test file --- .../proposal_published_event_spec.rb | 43 +------------------ 1 file changed, 2 insertions(+), 41 deletions(-) diff --git a/spec/events/decidim/proposals/proposal_published_event_spec.rb b/spec/events/decidim/proposals/proposal_published_event_spec.rb index 8ec2b28d2c..8dfd763e38 100644 --- a/spec/events/decidim/proposals/proposal_published_event_spec.rb +++ b/spec/events/decidim/proposals/proposal_published_event_spec.rb @@ -1,7 +1,6 @@ # frozen_string_literal: true require "spec_helper" -require "nokogiri" module Decidim::Proposals describe "ProposalPublishedEvent", type: :job do @@ -19,56 +18,18 @@ module Decidim::Proposals end context "when proposal is published" do - + subject { ProposalPublishedEvent.new(resource: proposal, event_name: "decidim.proposals.proposal_published", user: proposal.creator_identity) } it "sends a notification email with the correct content" do - subject { ProposalPublishedEvent.new(resource: proposal, event_name: "decidim.proposals.proposal_published", user: proposal.creator_identity) } + expect(last_email).not_to be_nil expect(last_email_body).not_to include("translation missing") expect(last_email_body).to include("Test proposition notification") end it "sends a notification with the correct content" do - subject { ProposalPublishedEvent.new(resource: proposal, event_name: "decidim.proposals.proposal_published", user: proposal.creator_identity) } notification = Decidim::Notification.last - # Parse avec Nokogiri - html_body = Nokogiri::HTML(notification.extra["body"]) - text = html_body.text - expect(text).to eq("Your proposal Test proposition notification has been published") - expect(notification).not_to be_nil end end end end -# -# shared_context "when sends the notification digest" do -# context "when daily notification mail" do -# let(:user) { create(:user, :admin, organization:, notifications_sending_frequency: "daily") } -# -# it_behaves_like "notification digest mail" -# end -# -# context "when weekly notification mail" do -# let(:user) { create(:user, :admin, organization:, notifications_sending_frequency: "weekly") } -# -# it_behaves_like "notification digest mail" -# end -# end -# -# shared_examples_for "notification digest mail" do -# context "when a notifiable event takes place" do -# let!(:organization) { create(:organization) } -# let!(:participatory_space) { create(:participatory_process, organization:) } -# -# it "sends a notification to the user's email" do -# perform_enqueued_jobs do -# expect(command.call).to broadcast(:ok) -# Decidim::Notification.last.update(created_at: 1.day.ago) -# Decidim::EmailNotificationsDigestGeneratorJob.perform_now(user.id, user.notifications_sending_frequency) -# end -# -# expect(last_email_body.length).to be_positive -# expect(last_email_body).not_to include("translation missing") -# end -# end -# end From df4935d7f6a62f55fd262cefbf74dbea96bfb2f0 Mon Sep 17 00:00:00 2001 From: Quentin Champenois Date: Thu, 7 Nov 2024 16:55:07 +0100 Subject: [PATCH 08/32] fix: Merge proposal command and anonymous proposals --- .../decidim/proposals/publish_proposal.rb | 18 +++++++++++++----- 1 file changed, 13 insertions(+), 5 deletions(-) diff --git a/app/commands/decidim/proposals/publish_proposal.rb b/app/commands/decidim/proposals/publish_proposal.rb index 4e6f2860f1..398f2b88fb 100644 --- a/app/commands/decidim/proposals/publish_proposal.rb +++ b/app/commands/decidim/proposals/publish_proposal.rb @@ -4,13 +4,17 @@ module Decidim module Proposals # A command with all the business logic when a user publishes a draft proposal. class PublishProposal < Decidim::Command + include Decidim::AnonymousProposals::AnonymousBehaviorCommandsConcern + # Public: Initializes the command. # # proposal - The proposal to publish. # current_user - The current user. + # override: decidim-module-anonymous_proposals/app/commands/decidim/anonymous_proposals/publish_proposal_command_overrides.rb def initialize(proposal, current_user) @proposal = proposal - @current_user = current_user + @is_anonymous = allow_anonymous_proposals? && (current_user.blank? || proposal.authored_by?(anonymous_group)) + set_current_user(current_user) end # Executes the command. Broadcasts these events: @@ -77,14 +81,13 @@ def send_notification end def send_publication_notification - puts "send notification from proposal #{@proposal.id}" - # debug message Decidim::EventsManager.publish( - event: "decidim.proposals.proposal_published", + event: "decidim.events.proposals.proposal_published", event_class: Decidim::Proposals::ProposalPublishedEvent, resource: @proposal, affected_users: [@proposal.creator_identity], - extra: { participatory_space: true } + extra: { force_email: true }, + force_send: true ) end @@ -113,6 +116,11 @@ def increment_scores end end end + + # override: decidim-module-anonymous_proposals/app/commands/decidim/anonymous_proposals/publish_proposal_command_overrides.rb + def component + @component ||= @proposal.component + end end end end From cb8772af1b552efdbf033f89742153272836922f Mon Sep 17 00:00:00 2001 From: Quentin Champenois Date: Thu, 7 Nov 2024 17:06:28 +0100 Subject: [PATCH 09/32] fix: Change ProposalPublishedEvent to SimpleEvent --- .../proposals/proposal_published_event.rb | 14 +- spec/shared/simple_event.rb | 202 ++++++++++++++++++ 2 files changed, 204 insertions(+), 12 deletions(-) create mode 100644 spec/shared/simple_event.rb diff --git a/app/events/decidim/proposals/proposal_published_event.rb b/app/events/decidim/proposals/proposal_published_event.rb index 2fe8cf8bb7..05456df290 100644 --- a/app/events/decidim/proposals/proposal_published_event.rb +++ b/app/events/decidim/proposals/proposal_published_event.rb @@ -1,13 +1,7 @@ # app/events/decidim/proposals/proposal_published_event.rb module Decidim module Proposals - class ProposalPublishedEvent < Decidim::Events::BaseEvent - include Decidim::Events::NotificationEvent - - def self.model_name - ActiveModel::Name.new(self, nil, I18n.t('decidim.proposals.notifications.proposal_published.subject')) - end - + class ProposalPublishedEvent < Decidim::Events::SimpleEvent def notification_title I18n.t("decidim.proposals.notifications.proposal_published.title", proposal_title: resource_title) end @@ -23,11 +17,7 @@ def resource_title end def resource_path - Decidim::Proposals::Engine.routes.url_helpers.proposal_path( - resource, - component_id: resource.component.id, - initiative_slug: resource.component.participatory_space.slug - ) + "" end end end diff --git a/spec/shared/simple_event.rb b/spec/shared/simple_event.rb new file mode 100644 index 0000000000..d65d354503 --- /dev/null +++ b/spec/shared/simple_event.rb @@ -0,0 +1,202 @@ +# frozen_string_literal: true + +require "spec_helper" + +shared_context "when a simple event" do + include Decidim::SanitizeHelper + subject { event_instance } + + let(:event_instance) do + described_class.new( + resource: resource, + event_name: event_name, + user: user, + user_role: user_role, + extra: extra + ) + end + + let(:organization) do + if resource.respond_to?(:organization) + resource.organization + else + create :organization + end + end + let(:user) { create :user, organization: organization } + let(:user_role) { :follower } + let(:extra) { {} } + let(:resource_path) { resource_locator(resource).path } + let(:resource_url) { resource_locator(resource).url } + let(:resource_title) { decidim_sanitize_translated(resource.title) } + # to be used when resource is a component resource, not a participatory space, in which case should be overriden + let(:participatory_space) { resource.participatory_space } + let(:participatory_space_title) { decidim_sanitize_translated(participatory_space.title) } + let(:participatory_space_path) { Decidim::ResourceLocatorPresenter.new(participatory_space).path } + let(:participatory_space_url) { Decidim::ResourceLocatorPresenter.new(participatory_space).url } + let(:author) do + if resource.respond_to?(:creator_author) + resource.creator_author + else + resource.author + end + end + let(:author_presenter) { Decidim::UserPresenter.new(author) } + let(:author_name) { decidim_html_escape author.name } + let(:author_path) { author_presenter&.profile_path.to_s } + let(:author_nickname) { author_presenter&.nickname.to_s } + let(:i18n_scope) { event_name } +end + +shared_examples_for "a simple event" do |skip_space_checks| + describe "types" do + subject { described_class } + + it "supports notifications" do + expect(subject.types).to include :notification + end + + it "supports emails" do + expect(subject.types).to include :email + end + end + + describe "email_subject" do + it "is generated correctly" do + expect(subject.email_subject).to be_kind_of(String) + expect(subject.email_subject).not_to include("translation missing") + expect(subject.email_subject).not_to include("script") + end + end + + describe "email_intro" do + it "is generated correctly" do + expect(subject.email_intro).to be_kind_of(String) + expect(subject.email_intro).not_to include("translation missing") + end + end + + describe "email_outro" do + it "is generated correctly" do + expect(subject.email_outro).to be_kind_of(String) + expect(subject.email_outro).not_to include("translation missing") + end + end + + describe "email_greeting" do + it "is generated correctly" do + expect(subject.email_greeting).to be_kind_of(String) + expect(subject.email_greeting).not_to include("translation missing") + end + end + + describe "safe_resource_text" do + it "is generated correctly" do + expect(subject.safe_resource_text).to be_kind_of(String) + expect(subject.safe_resource_text).to be_html_safe + end + end + + describe "notification_title" do + it "is generated correctly" do + expect(subject.notification_title).to be_kind_of(String) + expect(subject.notification_title).not_to include("translation missing") + expect(subject.notification_title).not_to include("script") + end + end + + describe "resource_path" do + it "is generated correctly" do + expect(subject.resource_path).to be_kind_of(String) + end + end + + describe "resource_url" do + it "is generated correctly" do + expect(subject.resource_url).to be_kind_of(String) + expect(subject.resource_url).to start_with("http") + end + end + + describe "resource_title" do + it "responds to the method" do + expect(subject).to respond_to(:resource_title) + end + end + + unless skip_space_checks + describe "participatory_space_url" do + it "is generated correctly" do + expect(subject.participatory_space_url).to be_kind_of(String) + expect(subject.participatory_space_url).to start_with("http") + end + end + + describe "participatory_space_title" do + it "is generated correctly" do + expect(translated(participatory_space.title)).to include("script") + end + end + end + + describe "i18n_options" do + subject { super().i18n_options } + + it { is_expected.to include(resource_path: satisfy(&:present?)) } + it { is_expected.to include(resource_title: satisfy(&:present?)) } + it { is_expected.to include(resource_url: start_with("http")) } + + it "includes the i18n scope" do + if event_instance.event_has_roles? + expect(subject).to include(scope: "#{i18n_scope}.#{user_role}") + else + expect(subject).to include(scope: i18n_scope) + end + end + + unless skip_space_checks + it { is_expected.to include(participatory_space_title: satisfy(&:present?)) } + it { is_expected.to include(participatory_space_url: start_with("http")) } + end + end +end + +shared_examples_for "a simple event email" do + describe "email_subject" do + it "is generated correctly" do + expect(subject.email_subject).to eq(email_subject) + end + + # it "is html safe" do + # # pending "Enable after #12547 is merged" + # expect(subject.email_subject).not_to include("script") + # end + end + + describe "email_intro" do + it "is generated correctly" do + expect(subject.email_intro).to eq(email_intro) + end + end + + describe "email_outro" do + it "is generated correctly" do + expect(subject.email_outro).to eq(email_outro) + end + end +end + +shared_examples_for "a simple event notification" do + describe "notification_title" do + it "is generated correctly" do + expect(subject.notification_title) + .to eq(notification_title) + end + # + # it "is html safe" do + # pp subject.notification_title + # pending "Enable after #12547 is merged" + # expect(subject.notification_title).not_to include("script") + # end + end +end \ No newline at end of file From d4beb619b21d89f73e25c35761172d1e01561ddf Mon Sep 17 00:00:00 2001 From: Quentin Champenois Date: Thu, 7 Nov 2024 17:18:01 +0100 Subject: [PATCH 10/32] fix: Proposal Published Event --- app/commands/decidim/proposals/publish_proposal.rb | 2 +- .../decidim/proposals/proposal_published_event.rb | 14 -------------- 2 files changed, 1 insertion(+), 15 deletions(-) diff --git a/app/commands/decidim/proposals/publish_proposal.rb b/app/commands/decidim/proposals/publish_proposal.rb index 398f2b88fb..81f58d3779 100644 --- a/app/commands/decidim/proposals/publish_proposal.rb +++ b/app/commands/decidim/proposals/publish_proposal.rb @@ -82,7 +82,7 @@ def send_notification def send_publication_notification Decidim::EventsManager.publish( - event: "decidim.events.proposals.proposal_published", + event: "decidim.events.proposals.proposal_published_event", event_class: Decidim::Proposals::ProposalPublishedEvent, resource: @proposal, affected_users: [@proposal.creator_identity], diff --git a/app/events/decidim/proposals/proposal_published_event.rb b/app/events/decidim/proposals/proposal_published_event.rb index 05456df290..27b8bba9ab 100644 --- a/app/events/decidim/proposals/proposal_published_event.rb +++ b/app/events/decidim/proposals/proposal_published_event.rb @@ -2,23 +2,9 @@ module Decidim module Proposals class ProposalPublishedEvent < Decidim::Events::SimpleEvent - def notification_title - I18n.t("decidim.proposals.notifications.proposal_published.title", proposal_title: resource_title) - end - - def notification_body - I18n.t("decidim.proposals.notifications.proposal_published.body", proposal_link: resource_path) - end - - private - def resource_title translated_attribute(resource.title) end - - def resource_path - "" - end end end end From 2fbbf92b6d8c96416e7bea692bf1a043cf6eca95 Mon Sep 17 00:00:00 2001 From: Quentin Champenois Date: Thu, 7 Nov 2024 17:18:49 +0100 Subject: [PATCH 11/32] fix: Push FR locales --- config/locales/fr.yml | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/config/locales/fr.yml b/config/locales/fr.yml index 58f4874a10..00052e5887 100644 --- a/config/locales/fr.yml +++ b/config/locales/fr.yml @@ -67,6 +67,12 @@ fr: new: sign_in_disabled: Vous pouvez accéder avec un compte externe events: + proposals: + proposal_published_event: + email_intro: 'que vous suivez, a publié une nouvelle proposition appelée "%{resource_title}". Découvrez-le et contribuez:' + email_outro: Vous avez reçu cette notification car vous suivez . Si vous souhaitez vous désabonner des notifications, connectez-vous à la plateforme, puis rendez-vous dans l'onglet “Mon compte” > “Paramètres des notifications”. + email_subject: Nouvelle proposition "%{resource_title}" publiée + notification_title: La proposition %{resource_title} a été publiée. budgets: pending_order: email_intro: Le vote sur le budget "%{resource_title}" n'est pas encore finalisé sur la concertation "%{participatory_space_title}". From 899f21481b58e6ecfa084ab68cfc70c1a58bb07d Mon Sep 17 00:00:00 2001 From: Quentin Champenois Date: Thu, 7 Nov 2024 17:37:41 +0100 Subject: [PATCH 12/32] test: Add specs for proposal_published_event --- .../proposal_published_event_spec.rb | 91 ++++++++---- spec/shared/translated_event_examples.rb | 131 ++++++++++++++++++ 2 files changed, 198 insertions(+), 24 deletions(-) create mode 100644 spec/shared/translated_event_examples.rb diff --git a/spec/events/decidim/proposals/proposal_published_event_spec.rb b/spec/events/decidim/proposals/proposal_published_event_spec.rb index 8dfd763e38..43fd5ea9db 100644 --- a/spec/events/decidim/proposals/proposal_published_event_spec.rb +++ b/spec/events/decidim/proposals/proposal_published_event_spec.rb @@ -2,34 +2,77 @@ require "spec_helper" -module Decidim::Proposals - describe "ProposalPublishedEvent", type: :job do - include MailerHelpers - - let(:organization) { create(:organization) } - let(:participatory_process) { create(:participatory_process, organization: organization) } - let(:component) { create(:component, manifest_name: :proposals, participatory_space: participatory_process) } - let!(:proposal_state) { create(:proposal_state, component: component) } - let(:proposal) { create(:proposal, component: component, proposal_state: proposal_state, title: { en: "Test proposition notification" }) } - let(:user) { create(:user, :admin, organization: organization, notifications_sending_frequency: "daily") } - - before do - clear_emails - end +module Decidim + module Proposals + describe ProposalPublishedEvent do + let(:resource) { create :extended_proposal } + let(:participatory_process) { create :participatory_process, organization: organization } + let(:proposal_component) { create(:extended_proposal_component, participatory_space: participatory_process) } + let(:resource_title) { decidim_sanitize_translated(resource.title) } + let(:event_name) { "decidim.events.proposals.proposal_published" } + + include_context "when a simple event" + + it_behaves_like "a simple event" + + describe "resource_text" do + it "returns the proposal body" do + expect(subject.resource_text).to eq(resource.body) + end + end + + describe "email_subject" do + context "when resource title contains apostrophes" do + it "is generated correctly" do + expect(subject.email_subject).to eq("New proposal \"#{resource_title}\" by @#{author.nickname}") + end + end + + it "is generated correctly" do + expect(subject.email_subject).to eq("New proposal \"#{resource_title}\" by @#{author.nickname}") + end + end - context "when proposal is published" do - subject { ProposalPublishedEvent.new(resource: proposal, event_name: "decidim.proposals.proposal_published", user: proposal.creator_identity) } - it "sends a notification email with the correct content" do + describe "email_intro" do + it "is generated correctly" do + expect(subject.email_intro) + .to eq("#{author.name} @#{author.nickname}, who you are following, has published a new proposal called \"#{resource_title}\". Check it out and contribute:") + end + end - expect(last_email).not_to be_nil - expect(last_email_body).not_to include("translation missing") - expect(last_email_body).to include("Test proposition notification") + describe "email_outro" do + it "is generated correctly" do + expect(subject.email_outro) + .to eq("You have received this notification because you are following @#{author.nickname}. You can stop receiving notifications following the previous link.") + end end - it "sends a notification with the correct content" do - notification = Decidim::Notification.last - expect(notification).not_to be_nil + describe "notification_title" do + it "is generated correctly" do + expect(subject.notification_title) + .to include("The #{resource_title} proposal was published by ") + + expect(subject.notification_title) + .to include("#{author.name} @#{author.nickname}.") + end + end + + describe "translated notifications" do + let(:en_body) { "A nice proposal" } + let(:body) { { en: en_body, machine_translations: { ca: "Une belle idee" } } } + let(:resource) do + create :extended_proposal, + component: proposal_component, + title: { en: "A nice proposal", machine_translations: { ca: "Une belle idee" } }, + body: body + end + + let(:en_version) { subject.resource_text["en"] } + let(:machine_translated) { subject.resource_text["machine_translations"]["ca"] } + let(:translatable) { true } + + it_behaves_like "a translated event" end end end -end +end \ No newline at end of file diff --git a/spec/shared/translated_event_examples.rb b/spec/shared/translated_event_examples.rb new file mode 100644 index 0000000000..47a9a6b6a7 --- /dev/null +++ b/spec/shared/translated_event_examples.rb @@ -0,0 +1,131 @@ +# frozen_string_literal: true + +shared_examples_for "a translated event" do + context "when it is not machine machine translated" do + let(:organization) { create(:organization, enable_machine_translations: false, machine_translation_display_priority: "original") } + + it "does not perform translation" do + expect(subject.perform_translation?).to eq(false) + end + + it "does not have a missing translation" do + expect(subject.translation_missing?).to eq(false) + end + + it "does have content available in multiple languages" do + expect(subject.content_in_same_language?).to eq(false) + end + + it "does return the original language" do + expect(subject.safe_resource_text).to eq(en_version) + end + + it "does not offer an alternate translation" do + expect(subject.safe_resource_translated_text).to eq(en_version) + end + end + + context "when is machine machine translated" do + let(:user) { create :user, organization: organization, locale: "ca" } + + around do |example| + I18n.with_locale(user.locale) { example.run } + end + + context "when priority is original" do + let(:organization) { create(:organization, enable_machine_translations: true, machine_translation_display_priority: "original") } + + it "does perform translation" do + expect(subject.perform_translation?).to eq(translatable) + end + + it "does not have a missing translation" do + expect(subject.translation_missing?).to eq(false) + end + + it "does have content available in multiple languages" do + expect(subject.content_in_same_language?).to eq(false) + end + + it "does return the original language" do + expect(subject.safe_resource_text).to eq(en_version) + end + + it "does not offer an alternate translation" do + expect(subject.safe_resource_translated_text).to eq(machine_translated) + end + + context "when translation is not available" do + let(:body) { { en: en_body } } + + it "does perform translation" do + expect(subject.perform_translation?).to eq(translatable) + end + + it "does have a missing translation" do + expect(subject.translation_missing?).to eq(translatable) + end + + it "does have content available in multiple languages" do + expect(subject.content_in_same_language?).to eq(false) + end + + it "does return the original language" do + expect(subject.safe_resource_text).to eq(en_version) + end + + it "does not offer an alternate translation" do + expect(subject.safe_resource_translated_text).to eq(en_version) + end + end + end + + context "when priority is translation" do + let(:organization) { create(:organization, enable_machine_translations: true, machine_translation_display_priority: "translation") } + + it "does perform translation" do + expect(subject.perform_translation?).to eq(translatable) + end + + it "does not have a missing translation" do + expect(subject.translation_missing?).to eq(false) + end + + it "does have content available in multiple languages" do + expect(subject.content_in_same_language?).to eq(false) + end + + it "does return the original language" do + expect(subject.safe_resource_text).to eq(en_version) + end + + it "does not offer an alternate translation" do + expect(subject.safe_resource_translated_text).to eq(machine_translated) + end + + context "when translation is not available" do + let(:body) { { en: en_body } } + + it "does perform translation" do + expect(subject.perform_translation?).to eq(translatable) + end + + it "does have a missing translation" do + expect(subject.translation_missing?).to eq(translatable) + end + + it "does have content available in multiple languages" do + expect(subject.content_in_same_language?).to eq(false) + end + + it "does return the original language" do + expect(subject.safe_resource_text).to eq(en_version) + end + + it "does not offer an alternate translation" do + expect(subject.safe_resource_translated_text).to eq(en_version) + end + end + end + end +end From 1b491c31b5dfd5811cbf8b3b257a8f56c2ba68c2 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Fri, 8 Nov 2024 16:27:05 +0100 Subject: [PATCH 13/32] update methode & file name --- .../decidim/proposals/publish_proposal.rb | 2 +- .../proposals_controller_override.rb | 2 +- ... => author_confirmation_proposal_event.rb} | 2 +- ...author_confirmation_proposal_event_spec.rb | 53 +++++++++++++++++++ .../proposal_published_event_spec.rb | 35 ------------ 5 files changed, 56 insertions(+), 38 deletions(-) rename app/events/decidim/proposals/{proposal_published_event.rb => author_confirmation_proposal_event.rb} (72%) create mode 100644 spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb delete mode 100644 spec/events/decidim/proposals/proposal_published_event_spec.rb diff --git a/app/commands/decidim/proposals/publish_proposal.rb b/app/commands/decidim/proposals/publish_proposal.rb index 81f58d3779..a1dcb62fd1 100644 --- a/app/commands/decidim/proposals/publish_proposal.rb +++ b/app/commands/decidim/proposals/publish_proposal.rb @@ -83,7 +83,7 @@ def send_notification def send_publication_notification Decidim::EventsManager.publish( event: "decidim.events.proposals.proposal_published_event", - event_class: Decidim::Proposals::ProposalPublishedEvent, + event_class: Decidim::Proposals::AuthorConfirmationProposalEvent, resource: @proposal, affected_users: [@proposal.creator_identity], extra: { force_email: true }, diff --git a/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb b/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb index 51fcfe5cb3..717d58555b 100644 --- a/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb +++ b/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb @@ -105,6 +105,7 @@ def create end # Overridden because of a core bug when the command posts the "invalid" # signal and when rendering the form. + def update_draft enforce_permission_to :edit, :proposal, proposal: @proposal @step = Decidim::Proposals::ProposalsController::STEP1 @@ -202,7 +203,6 @@ def proposal_limit_reached?(form = form_proposal_params) def current_user_proposals(form) Decidim::Proposals::Proposal.from_author(current_user).where(component: form.current_component).except_withdrawn end - end end end diff --git a/app/events/decidim/proposals/proposal_published_event.rb b/app/events/decidim/proposals/author_confirmation_proposal_event.rb similarity index 72% rename from app/events/decidim/proposals/proposal_published_event.rb rename to app/events/decidim/proposals/author_confirmation_proposal_event.rb index 27b8bba9ab..604f9b39ef 100644 --- a/app/events/decidim/proposals/proposal_published_event.rb +++ b/app/events/decidim/proposals/author_confirmation_proposal_event.rb @@ -1,7 +1,7 @@ # app/events/decidim/proposals/proposal_published_event.rb module Decidim module Proposals - class ProposalPublishedEvent < Decidim::Events::SimpleEvent + class AuthorConfirmationProposalEvent < Decidim::Events::SimpleEvent def resource_title translated_attribute(resource.title) end diff --git a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb new file mode 100644 index 0000000000..ac73368a96 --- /dev/null +++ b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb @@ -0,0 +1,53 @@ +# frozen_string_literal: true + +require "spec_helper" + +module Decidim + module Proposals + describe ProposalPublishedEvent do + let(:organization) { create(:organization) } + let(:author) { create(:user, organization: organization) } + let(:resource) { create(:proposal, component: proposal_component, title: { en: "Proposition de test" }, body: { en: "Corps de la proposition" }) } + let(:participatory_process) { create(:participatory_process, organization: organization) } + let(:proposal_component) { create(:component, manifest_name: :proposals, participatory_space: participatory_process) } + let(:resource_title) { decidim_sanitize_translated(resource.title) } + let(:resource_path) { Decidim::ResourceLocatorPresenter.new(resource).path } + let(:event_name) { "decidim.events.proposals.proposal_published" } + + subject { described_class.new(resource: resource, event_name: event_name, user: author) } + + describe "email_subject" do + it "matches the expected translation" do + expect(subject.email_subject).to eq(I18n.t("decidim.proposals.notifications.proposal_published.subject")) + end + end + + describe "email_intro" do + it "matches the expected translation" do + expected_intro = I18n.t("decidim.proposals.notifications.proposal_published.email_intro", resource_title: resource_title, resource_path: resource_path) + expect(subject.email_intro).to eq(expected_intro) + end + end + + describe "email_outro" do + it "matches the expected translation" do + expected_outro = I18n.t("decidim.proposals.notifications.proposal_published.email_outro", resource_title: resource_title) + expect(subject.email_outro).to eq(expected_outro) + end + end + + describe "notification_title" do + it "matches the expected translation" do + expected_title = I18n.t("decidim.proposals.notifications.proposal_published.notification_title", resource_title: resource_title, resource_path: resource_path) + expect(subject.notification_title).to eq(expected_title) + end + end + + describe "resource_text" do + it "returns the proposal body" do + expect(subject.resource_text).to eq(resource.body) + end + end + end + end +end diff --git a/spec/events/decidim/proposals/proposal_published_event_spec.rb b/spec/events/decidim/proposals/proposal_published_event_spec.rb deleted file mode 100644 index 8dfd763e38..0000000000 --- a/spec/events/decidim/proposals/proposal_published_event_spec.rb +++ /dev/null @@ -1,35 +0,0 @@ -# frozen_string_literal: true - -require "spec_helper" - -module Decidim::Proposals - describe "ProposalPublishedEvent", type: :job do - include MailerHelpers - - let(:organization) { create(:organization) } - let(:participatory_process) { create(:participatory_process, organization: organization) } - let(:component) { create(:component, manifest_name: :proposals, participatory_space: participatory_process) } - let!(:proposal_state) { create(:proposal_state, component: component) } - let(:proposal) { create(:proposal, component: component, proposal_state: proposal_state, title: { en: "Test proposition notification" }) } - let(:user) { create(:user, :admin, organization: organization, notifications_sending_frequency: "daily") } - - before do - clear_emails - end - - context "when proposal is published" do - subject { ProposalPublishedEvent.new(resource: proposal, event_name: "decidim.proposals.proposal_published", user: proposal.creator_identity) } - it "sends a notification email with the correct content" do - - expect(last_email).not_to be_nil - expect(last_email_body).not_to include("translation missing") - expect(last_email_body).to include("Test proposition notification") - end - - it "sends a notification with the correct content" do - notification = Decidim::Notification.last - expect(notification).not_to be_nil - end - end - end -end From 4f9de63bcc3c76d80424fd970096503cf7062368 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Fri, 8 Nov 2024 16:28:16 +0100 Subject: [PATCH 14/32] adjust trad key --- config/locales/en.yml | 10 +++++----- config/locales/fr.yml | 22 +++++++++++----------- 2 files changed, 16 insertions(+), 16 deletions(-) diff --git a/config/locales/en.yml b/config/locales/en.yml index f64f878fc5..a54145f7e8 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -147,11 +147,11 @@ en: exports: awesome_private_proposals: Proposals with private fields proposal_comments: Comments - notifications: - proposal_published: - title: "Your proposal %{proposal_title} has been published!" - body: "Your proposal is now live. View it here: %{proposal_link}" - subject: "Your proposal has been published!" + author_confirmation_proposal_event: + email_intro: 'Your proposal "%{resource_title}" was successfully received and is now public. Thank you for participating! You can view it here: %{resource_title}.' + email_outro: You received this notification because you are the author of the proposal. You can unfollow it by visiting the proposal page ("%{resource_title}") and clicking on "Unfollow". + email_subject: Your proposal has been published! + notification_title: Your proposal %{resource_title} is now live. collaborative_drafts: new: add_file: Add file diff --git a/config/locales/fr.yml b/config/locales/fr.yml index 00052e5887..2064fcb182 100644 --- a/config/locales/fr.yml +++ b/config/locales/fr.yml @@ -67,12 +67,6 @@ fr: new: sign_in_disabled: Vous pouvez accéder avec un compte externe events: - proposals: - proposal_published_event: - email_intro: 'que vous suivez, a publié une nouvelle proposition appelée "%{resource_title}". Découvrez-le et contribuez:' - email_outro: Vous avez reçu cette notification car vous suivez . Si vous souhaitez vous désabonner des notifications, connectez-vous à la plateforme, puis rendez-vous dans l'onglet “Mon compte” > “Paramètres des notifications”. - email_subject: Nouvelle proposition "%{resource_title}" publiée - notification_title: La proposition %{resource_title} a été publiée. budgets: pending_order: email_intro: Le vote sur le budget "%{resource_title}" n'est pas encore finalisé sur la concertation "%{participatory_space_title}". @@ -85,6 +79,12 @@ fr: email_outro: Vous avez reçu cette notification parce que vous suivez la pétition "%{resource_title}". email_subject: La pétition "%{resource_title}" a reçu une réponse. notification_title: La pétition %{resource_title} a reçu une réponse. + proposals: + proposal_published_event: + email_intro: 'que vous suivez, a publié une nouvelle proposition appelée "%{resource_title}". Découvrez-le et contribuez:' + email_outro: Vous avez reçu cette notification car vous suivez . Si vous souhaitez vous désabonner des notifications, connectez-vous à la plateforme, puis rendez-vous dans l'onglet “Mon compte” > “Paramètres des notifications”. + email_subject: Nouvelle proposition "%{resource_title}" publiée + notification_title: La proposition %{resource_title} a été publiée. users: user_officialized: email_intro: Le participant %{name} (%{nickname}) a été officialisé. @@ -155,6 +155,11 @@ fr: exports: awesome_private_proposals: Propositions avec champs privés proposal_comments: Commentaires + author_confirmation_proposal_event: + email_intro: 'Votre proposition "%{resource_title}" a été reçue avec succès et est maintenant publique. Merci pour votre participation! Vous pouvez la consulter ici : %{resource_title}.' + email_outro: Vous recevez cette notification car vous êtes l’auteur de la proposition. Vous pouvez vous désabonner en visitant la page de la proposition ("%{resource_title}") et en cliquant sur "Ne plus suivre". + email_subject: Votre proposition a été publiée ! + notification_title: Votre proposition %{resource_title} est maintenant en ligne. collaborative_drafts: new: add_file: Ajouter le fichier @@ -164,11 +169,6 @@ fr: collaborative_drafts_list: Accéder aux brouillons collaboratifs new_proposal: Nouvelle proposition view_proposal: Voir la proposition - notifications: - proposal_published: - title: "Votre proposition %{proposal_title} a bien été publiée!" - body: "Votre proposition est en ligne. Lien ici: %{proposal_link}" - subject: "Votre proposition est publiée!" update: error: Il y a eu une erreur lors de la mise à jour de la proposition. success: Proposition mise à jour avec succès. From 1c13724d406b1322e630c7b856e60d23cad4b571 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Fri, 8 Nov 2024 16:29:48 +0100 Subject: [PATCH 15/32] adjust test file & update name of test file --- ...author_confirmation_proposal_event_spec.rb | 46 +++++++++++++------ 1 file changed, 31 insertions(+), 15 deletions(-) diff --git a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb index ac73368a96..914631fe2c 100644 --- a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb +++ b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb @@ -4,49 +4,65 @@ module Decidim module Proposals - describe ProposalPublishedEvent do - let(:organization) { create(:organization) } - let(:author) { create(:user, organization: organization) } - let(:resource) { create(:proposal, component: proposal_component, title: { en: "Proposition de test" }, body: { en: "Corps de la proposition" }) } - let(:participatory_process) { create(:participatory_process, organization: organization) } - let(:proposal_component) { create(:component, manifest_name: :proposals, participatory_space: participatory_process) } + describe AuthorConfirmationProposalEvent do + let(:resource) { create :extended_proposal } + let(:participatory_process) { create :participatory_process, organization: organization } + let(:proposal_component) { create(:extended_proposal_component, participatory_space: participatory_process) } let(:resource_title) { decidim_sanitize_translated(resource.title) } - let(:resource_path) { Decidim::ResourceLocatorPresenter.new(resource).path } let(:event_name) { "decidim.events.proposals.proposal_published" } - subject { described_class.new(resource: resource, event_name: event_name, user: author) } + include_context "when a simple event" + + it_behaves_like "a simple event" + + describe "resource_text" do + it "returns the proposal body" do + expect(subject.resource_text).to eq(resource.body) + end + end describe "email_subject" do it "matches the expected translation" do - expect(subject.email_subject).to eq(I18n.t("decidim.proposals.notifications.proposal_published.subject")) + expect(subject.email_subject).to eq("Your proposal has been published!") end end describe "email_intro" do it "matches the expected translation" do - expected_intro = I18n.t("decidim.proposals.notifications.proposal_published.email_intro", resource_title: resource_title, resource_path: resource_path) + expected_intro = "Your proposal \"#{resource_title}\" was successfully received and is now public. Thank you for participating! Here it is: #{resource_title}." expect(subject.email_intro).to eq(expected_intro) end end describe "email_outro" do it "matches the expected translation" do - expected_outro = I18n.t("decidim.proposals.notifications.proposal_published.email_outro", resource_title: resource_title) + expected_outro = "You have received this notification because you are the author of the proposal. You can unfollow it by going to the proposal page (\"#{resource_title}\") and clicking on \"Unfollow\"." expect(subject.email_outro).to eq(expected_outro) end end describe "notification_title" do it "matches the expected translation" do - expected_title = I18n.t("decidim.proposals.notifications.proposal_published.notification_title", resource_title: resource_title, resource_path: resource_path) + expected_title = "Your proposal #{resource_title} is now live." expect(subject.notification_title).to eq(expected_title) end end - describe "resource_text" do - it "returns the proposal body" do - expect(subject.resource_text).to eq(resource.body) + describe "translated notifications" do + let(:en_body) { "A nice proposal" } + let(:body) { { en: en_body, machine_translations: { ca: "Une belle idee" } } } + let(:resource) do + create :extended_proposal, + component: proposal_component, + title: { en: "A nice proposal", machine_translations: { ca: "Une belle idee" } }, + body: body end + + let(:en_version) { subject.resource_text["en"] } + let(:machine_translated) { subject.resource_text["machine_translations"]["ca"] } + let(:translatable) { true } + + it_behaves_like "a translated event" end end end From 633159721f355b8129e82785c1cff4f557a3dc1b Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Fri, 8 Nov 2024 16:31:09 +0100 Subject: [PATCH 16/32] lint --- .../decidim/proposals/author_confirmation_proposal_event.rb | 3 ++- db/schema.rb | 1 + spec/shared/simple_event.rb | 2 +- 3 files changed, 4 insertions(+), 2 deletions(-) diff --git a/app/events/decidim/proposals/author_confirmation_proposal_event.rb b/app/events/decidim/proposals/author_confirmation_proposal_event.rb index 604f9b39ef..6bd2162ac8 100644 --- a/app/events/decidim/proposals/author_confirmation_proposal_event.rb +++ b/app/events/decidim/proposals/author_confirmation_proposal_event.rb @@ -1,4 +1,5 @@ -# app/events/decidim/proposals/proposal_published_event.rb +# frozen_string_literal: true + module Decidim module Proposals class AuthorConfirmationProposalEvent < Decidim::Events::SimpleEvent diff --git a/db/schema.rb b/db/schema.rb index 7b40047a8a..af643bb11c 100644 --- a/db/schema.rb +++ b/db/schema.rb @@ -11,6 +11,7 @@ # It's strongly recommended that you check this file into your version control system. ActiveRecord::Schema.define(version: 2024_10_10_092645) do + # These are extensions that must be enabled in order to support this database enable_extension "ltree" enable_extension "pg_trgm" diff --git a/spec/shared/simple_event.rb b/spec/shared/simple_event.rb index d65d354503..4e8b980527 100644 --- a/spec/shared/simple_event.rb +++ b/spec/shared/simple_event.rb @@ -199,4 +199,4 @@ # expect(subject.notification_title).not_to include("script") # end end -end \ No newline at end of file +end From 1f699187c162ce67069ae7e898d21f03abacb762 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Fri, 8 Nov 2024 17:42:30 +0100 Subject: [PATCH 17/32] lint fr trad key --- config/locales/fr.yml | 15 +++++---------- 1 file changed, 5 insertions(+), 10 deletions(-) diff --git a/config/locales/fr.yml b/config/locales/fr.yml index 2064fcb182..5de02a32a2 100644 --- a/config/locales/fr.yml +++ b/config/locales/fr.yml @@ -80,11 +80,11 @@ fr: email_subject: La pétition "%{resource_title}" a reçu une réponse. notification_title: La pétition %{resource_title} a reçu une réponse. proposals: - proposal_published_event: - email_intro: 'que vous suivez, a publié une nouvelle proposition appelée "%{resource_title}". Découvrez-le et contribuez:' - email_outro: Vous avez reçu cette notification car vous suivez . Si vous souhaitez vous désabonner des notifications, connectez-vous à la plateforme, puis rendez-vous dans l'onglet “Mon compte” > “Paramètres des notifications”. - email_subject: Nouvelle proposition "%{resource_title}" publiée - notification_title: La proposition %{resource_title} a été publiée. + author_confirmation_proposal_event: + email_intro: 'Votre proposition "%{resource_title}" a été reçue avec succès et est maintenant publique. Merci pour votre participation! Vous pouvez la consulter ici : %{resource_title}.' + email_outro: Vous recevez cette notification car vous êtes l’auteur de la proposition. Vous pouvez vous désabonner en visitant la page de la proposition ("%{resource_title}") et en cliquant sur "Ne plus suivre". + email_subject: Votre proposition a été publiée ! + notification_title: Votre proposition %{resource_title} est maintenant en ligne. users: user_officialized: email_intro: Le participant %{name} (%{nickname}) a été officialisé. @@ -155,11 +155,6 @@ fr: exports: awesome_private_proposals: Propositions avec champs privés proposal_comments: Commentaires - author_confirmation_proposal_event: - email_intro: 'Votre proposition "%{resource_title}" a été reçue avec succès et est maintenant publique. Merci pour votre participation! Vous pouvez la consulter ici : %{resource_title}.' - email_outro: Vous recevez cette notification car vous êtes l’auteur de la proposition. Vous pouvez vous désabonner en visitant la page de la proposition ("%{resource_title}") et en cliquant sur "Ne plus suivre". - email_subject: Votre proposition a été publiée ! - notification_title: Votre proposition %{resource_title} est maintenant en ligne. collaborative_drafts: new: add_file: Ajouter le fichier From 51260053c94586102dc66dcb28153c14b79de861 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Fri, 8 Nov 2024 17:43:17 +0100 Subject: [PATCH 18/32] lint --- .../author_confirmation_proposal_event_spec.rb | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb index 914631fe2c..6c6ccf17c0 100644 --- a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb +++ b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb @@ -17,34 +17,34 @@ module Proposals describe "resource_text" do it "returns the proposal body" do - expect(subject.resource_text).to eq(resource.body) + expect(subject.resource_text).to be(resource.body) end end describe "email_subject" do it "matches the expected translation" do - expect(subject.email_subject).to eq("Your proposal has been published!") + expect(subject.email_subject).to be("Your proposal has been published!") end end describe "email_intro" do it "matches the expected translation" do expected_intro = "Your proposal \"#{resource_title}\" was successfully received and is now public. Thank you for participating! Here it is: #{resource_title}." - expect(subject.email_intro).to eq(expected_intro) + expect(subject.email_intro).to be(expected_intro) end end describe "email_outro" do it "matches the expected translation" do expected_outro = "You have received this notification because you are the author of the proposal. You can unfollow it by going to the proposal page (\"#{resource_title}\") and clicking on \"Unfollow\"." - expect(subject.email_outro).to eq(expected_outro) + expect(subject.email_outro).to be(expected_outro) end end describe "notification_title" do it "matches the expected translation" do expected_title = "Your proposal #{resource_title} is now live." - expect(subject.notification_title).to eq(expected_title) + expect(subject.notification_title).to be(expected_title) end end From 19c986f022a7ff307e93aa4397c7384e0b5932be Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Fri, 8 Nov 2024 17:45:58 +0100 Subject: [PATCH 19/32] update trad key order --- config/locales/en.yml | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/config/locales/en.yml b/config/locales/en.yml index a54145f7e8..e64c501a41 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -77,6 +77,12 @@ en: email_outro: You have received this notification because you are following the initiative "%{resource_title}". email_subject: Initiative "%{resource_title}" has been answered notification_title: The initiative %{resource_title} has been answered. + proposals: + author_confirmation_proposal_event: + email_intro: 'Your proposal "%{resource_title}" was successfully received and is now public. Thank you for participating! You can view it here: %{resource_title}.' + email_outro: You received this notification because you are the author of the proposal. You can unfollow it by visiting the proposal page ("%{resource_title}") and clicking on "Unfollow". + email_subject: Your proposal has been published! + notification_title: Your proposal %{resource_title} is now live. users: user_officialized: email_intro: Participant %{name} (%{nickname}) has been officialized. @@ -147,11 +153,6 @@ en: exports: awesome_private_proposals: Proposals with private fields proposal_comments: Comments - author_confirmation_proposal_event: - email_intro: 'Your proposal "%{resource_title}" was successfully received and is now public. Thank you for participating! You can view it here: %{resource_title}.' - email_outro: You received this notification because you are the author of the proposal. You can unfollow it by visiting the proposal page ("%{resource_title}") and clicking on "Unfollow". - email_subject: Your proposal has been published! - notification_title: Your proposal %{resource_title} is now live. collaborative_drafts: new: add_file: Add file From 1e9bf1a72327a1d1f94c8cb07a4f953557157953 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Tue, 12 Nov 2024 09:41:53 +0100 Subject: [PATCH 20/32] lint --- spec/shared/translated_event_examples.rb | 50 ++++++++++++------------ 1 file changed, 25 insertions(+), 25 deletions(-) diff --git a/spec/shared/translated_event_examples.rb b/spec/shared/translated_event_examples.rb index 47a9a6b6a7..77938b01f4 100644 --- a/spec/shared/translated_event_examples.rb +++ b/spec/shared/translated_event_examples.rb @@ -5,23 +5,23 @@ let(:organization) { create(:organization, enable_machine_translations: false, machine_translation_display_priority: "original") } it "does not perform translation" do - expect(subject.perform_translation?).to eq(false) + expect(subject.perform_translation?).to be(false) end it "does not have a missing translation" do - expect(subject.translation_missing?).to eq(false) + expect(subject.translation_missing?).to be(false) end it "does have content available in multiple languages" do - expect(subject.content_in_same_language?).to eq(false) + expect(subject.content_in_same_language?).to be(false) end it "does return the original language" do - expect(subject.safe_resource_text).to eq(en_version) + expect(subject.safe_resource_text).to be(en_version) end it "does not offer an alternate translation" do - expect(subject.safe_resource_translated_text).to eq(en_version) + expect(subject.safe_resource_translated_text).to be(en_version) end end @@ -36,46 +36,46 @@ let(:organization) { create(:organization, enable_machine_translations: true, machine_translation_display_priority: "original") } it "does perform translation" do - expect(subject.perform_translation?).to eq(translatable) + expect(subject.perform_translation?).to be(translatable) end it "does not have a missing translation" do - expect(subject.translation_missing?).to eq(false) + expect(subject.translation_missing?).to be(false) end it "does have content available in multiple languages" do - expect(subject.content_in_same_language?).to eq(false) + expect(subject.content_in_same_language?).to be(false) end it "does return the original language" do - expect(subject.safe_resource_text).to eq(en_version) + expect(subject.safe_resource_text).to be(en_version) end it "does not offer an alternate translation" do - expect(subject.safe_resource_translated_text).to eq(machine_translated) + expect(subject.safe_resource_translated_text).to be(machine_translated) end context "when translation is not available" do let(:body) { { en: en_body } } it "does perform translation" do - expect(subject.perform_translation?).to eq(translatable) + expect(subject.perform_translation?).to be(translatable) end it "does have a missing translation" do - expect(subject.translation_missing?).to eq(translatable) + expect(subject.translation_missing?).to be(translatable) end it "does have content available in multiple languages" do - expect(subject.content_in_same_language?).to eq(false) + expect(subject.content_in_same_language?).to be(false) end it "does return the original language" do - expect(subject.safe_resource_text).to eq(en_version) + expect(subject.safe_resource_text).to be(en_version) end it "does not offer an alternate translation" do - expect(subject.safe_resource_translated_text).to eq(en_version) + expect(subject.safe_resource_translated_text).to be(en_version) end end end @@ -84,46 +84,46 @@ let(:organization) { create(:organization, enable_machine_translations: true, machine_translation_display_priority: "translation") } it "does perform translation" do - expect(subject.perform_translation?).to eq(translatable) + expect(subject.perform_translation?).to be(translatable) end it "does not have a missing translation" do - expect(subject.translation_missing?).to eq(false) + expect(subject.translation_missing?).to be(false) end it "does have content available in multiple languages" do - expect(subject.content_in_same_language?).to eq(false) + expect(subject.content_in_same_language?).to be(false) end it "does return the original language" do - expect(subject.safe_resource_text).to eq(en_version) + expect(subject.safe_resource_text).to be(en_version) end it "does not offer an alternate translation" do - expect(subject.safe_resource_translated_text).to eq(machine_translated) + expect(subject.safe_resource_translated_text).to be(machine_translated) end context "when translation is not available" do let(:body) { { en: en_body } } it "does perform translation" do - expect(subject.perform_translation?).to eq(translatable) + expect(subject.perform_translation?).to be(translatable) end it "does have a missing translation" do - expect(subject.translation_missing?).to eq(translatable) + expect(subject.translation_missing?).to be(translatable) end it "does have content available in multiple languages" do - expect(subject.content_in_same_language?).to eq(false) + expect(subject.content_in_same_language?).to be(false) end it "does return the original language" do - expect(subject.safe_resource_text).to eq(en_version) + expect(subject.safe_resource_text).to be(en_version) end it "does not offer an alternate translation" do - expect(subject.safe_resource_translated_text).to eq(en_version) + expect(subject.safe_resource_translated_text).to be(en_version) end end end From d9dbd3382ea3055da53a0c704ba947d7539898fc Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Tue, 12 Nov 2024 10:09:48 +0100 Subject: [PATCH 21/32] correct trad key link in method --- app/commands/decidim/proposals/publish_proposal.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/commands/decidim/proposals/publish_proposal.rb b/app/commands/decidim/proposals/publish_proposal.rb index a1dcb62fd1..a637eaed4f 100644 --- a/app/commands/decidim/proposals/publish_proposal.rb +++ b/app/commands/decidim/proposals/publish_proposal.rb @@ -82,7 +82,7 @@ def send_notification def send_publication_notification Decidim::EventsManager.publish( - event: "decidim.events.proposals.proposal_published_event", + event: "decidim.events.proposals.author_confirmation_proposal_event", event_class: Decidim::Proposals::AuthorConfirmationProposalEvent, resource: @proposal, affected_users: [@proposal.creator_identity], From b62d5250f147edfad078da287e9199f59b985fbb Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Tue, 12 Nov 2024 14:11:15 +0100 Subject: [PATCH 22/32] correst rspec --- .../proposals/author_confirmation_proposal_event_spec.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb index 6c6ccf17c0..72dab71b7f 100644 --- a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb +++ b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb @@ -9,7 +9,7 @@ module Proposals let(:participatory_process) { create :participatory_process, organization: organization } let(:proposal_component) { create(:extended_proposal_component, participatory_space: participatory_process) } let(:resource_title) { decidim_sanitize_translated(resource.title) } - let(:event_name) { "decidim.events.proposals.proposal_published" } + let(:event_name) { "decidim.events.proposals.author_confirmation_proposal_event" } include_context "when a simple event" From 58cdd95ab82e62117d1dd32cda39a79ff976f2a2 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Tue, 12 Nov 2024 14:39:59 +0100 Subject: [PATCH 23/32] fix rspec --- .../proposals/author_confirmation_proposal_event_spec.rb | 4 +++- spec/shared/simple_event.rb | 4 ++-- 2 files changed, 5 insertions(+), 3 deletions(-) diff --git a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb index 72dab71b7f..66b00f3112 100644 --- a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb +++ b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb @@ -5,10 +5,12 @@ module Decidim module Proposals describe AuthorConfirmationProposalEvent do + include Decidim::TranslatableAttributes + let(:resource) { create :extended_proposal } let(:participatory_process) { create :participatory_process, organization: organization } let(:proposal_component) { create(:extended_proposal_component, participatory_space: participatory_process) } - let(:resource_title) { decidim_sanitize_translated(resource.title) } + let(:resource_title) { decidim_sanitize(translated_attribute(resource.title)) } let(:event_name) { "decidim.events.proposals.author_confirmation_proposal_event" } include_context "when a simple event" diff --git a/spec/shared/simple_event.rb b/spec/shared/simple_event.rb index 4e8b980527..1305adae9e 100644 --- a/spec/shared/simple_event.rb +++ b/spec/shared/simple_event.rb @@ -28,10 +28,10 @@ let(:extra) { {} } let(:resource_path) { resource_locator(resource).path } let(:resource_url) { resource_locator(resource).url } - let(:resource_title) { decidim_sanitize_translated(resource.title) } + let(:resource_title) { decidim_sanitize(translated_attribute(resource.title)) } # to be used when resource is a component resource, not a participatory space, in which case should be overriden let(:participatory_space) { resource.participatory_space } - let(:participatory_space_title) { decidim_sanitize_translated(participatory_space.title) } + let(:participatory_space_title) { decidim_sanitize(translated_attribute(participatory_space.title)) } let(:participatory_space_path) { Decidim::ResourceLocatorPresenter.new(participatory_space).path } let(:participatory_space_url) { Decidim::ResourceLocatorPresenter.new(participatory_space).url } let(:author) do From faa18f9ed9270f2ff300afd0cede00801bf4e3c6 Mon Sep 17 00:00:00 2001 From: Quentin Champenois Date: Tue, 12 Nov 2024 15:31:13 +0100 Subject: [PATCH 24/32] fix: Notification small title --- .../decidim/proposals/author_confirmation_proposal_event.rb | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/app/events/decidim/proposals/author_confirmation_proposal_event.rb b/app/events/decidim/proposals/author_confirmation_proposal_event.rb index 6bd2162ac8..cb6c486e9f 100644 --- a/app/events/decidim/proposals/author_confirmation_proposal_event.rb +++ b/app/events/decidim/proposals/author_confirmation_proposal_event.rb @@ -3,6 +3,10 @@ module Decidim module Proposals class AuthorConfirmationProposalEvent < Decidim::Events::SimpleEvent + def self.model_name + ActiveModel::Name.new(self, nil, I18n.t('decidim.events.proposals.author_confirmation_proposal_event.email_subject')) + end + def resource_title translated_attribute(resource.title) end From d70fa7b4e294735c49a5a527c4f6f7a7427fe1fc Mon Sep 17 00:00:00 2001 From: Quentin Champenois Date: Tue, 12 Nov 2024 15:32:33 +0100 Subject: [PATCH 25/32] lint(rubocop): Fix offenses --- .../decidim/proposals/author_confirmation_proposal_event.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/events/decidim/proposals/author_confirmation_proposal_event.rb b/app/events/decidim/proposals/author_confirmation_proposal_event.rb index cb6c486e9f..22745650a4 100644 --- a/app/events/decidim/proposals/author_confirmation_proposal_event.rb +++ b/app/events/decidim/proposals/author_confirmation_proposal_event.rb @@ -4,7 +4,7 @@ module Decidim module Proposals class AuthorConfirmationProposalEvent < Decidim::Events::SimpleEvent def self.model_name - ActiveModel::Name.new(self, nil, I18n.t('decidim.events.proposals.author_confirmation_proposal_event.email_subject')) + ActiveModel::Name.new(self, nil, I18n.t("decidim.events.proposals.author_confirmation_proposal_event.email_subject")) end def resource_title From 5957def0217a2d8ad2f3af5835620f22cbe76ffe Mon Sep 17 00:00:00 2001 From: Quentin Champenois Date: Tue, 12 Nov 2024 15:34:30 +0100 Subject: [PATCH 26/32] ci: Exclude BeEq Rubocop rule --- .rubocop_rails.yml | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/.rubocop_rails.yml b/.rubocop_rails.yml index 68f4c40d64..4258353ee1 100644 --- a/.rubocop_rails.yml +++ b/.rubocop_rails.yml @@ -107,3 +107,7 @@ RSpec/MultipleMemoizedHelpers: RSpec/AnyInstance: Enabled: false + +RSpec/BeEq: + Exclude: + - spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb \ No newline at end of file From 1d89c03494c1a57e5e2e14cc36f48159d8c42e73 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Wed, 13 Nov 2024 12:23:36 +0100 Subject: [PATCH 27/32] clean --- .../proposals_controller_override.rb | 2 +- db/schema.rb | 1 - ...author_confirmation_proposal_event_spec.rb | 54 +---- spec/shared/simple_event.rb | 202 ------------------ spec/shared/translated_event_examples.rb | 131 ------------ 5 files changed, 11 insertions(+), 379 deletions(-) delete mode 100644 spec/shared/simple_event.rb delete mode 100644 spec/shared/translated_event_examples.rb diff --git a/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb b/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb index 717d58555b..b627ce8e14 100644 --- a/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb +++ b/app/controllers/concerns/decidim/simple_proposal/proposals_controller_override.rb @@ -103,9 +103,9 @@ def create end end end + # Overridden because of a core bug when the command posts the "invalid" # signal and when rendering the form. - def update_draft enforce_permission_to :edit, :proposal, proposal: @proposal @step = Decidim::Proposals::ProposalsController::STEP1 diff --git a/db/schema.rb b/db/schema.rb index af643bb11c..7b40047a8a 100644 --- a/db/schema.rb +++ b/db/schema.rb @@ -11,7 +11,6 @@ # It's strongly recommended that you check this file into your version control system. ActiveRecord::Schema.define(version: 2024_10_10_092645) do - # These are extensions that must be enabled in order to support this database enable_extension "ltree" enable_extension "pg_trgm" diff --git a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb index 66b00f3112..af040aa66a 100644 --- a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb +++ b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb @@ -5,67 +5,33 @@ module Decidim module Proposals describe AuthorConfirmationProposalEvent do - include Decidim::TranslatableAttributes - + let(:user) { create :user, organization: organization } let(:resource) { create :extended_proposal } let(:participatory_process) { create :participatory_process, organization: organization } let(:proposal_component) { create(:extended_proposal_component, participatory_space: participatory_process) } - let(:resource_title) { decidim_sanitize(translated_attribute(resource.title)) } let(:event_name) { "decidim.events.proposals.author_confirmation_proposal_event" } + let(:user_role) { :participant } + let(:extra) { {} } + let(:organization) { resource.organization || create(:organization) } - include_context "when a simple event" + subject { described_class.new(resource: resource, event_name: event_name, user: user, user_role: user_role, extra: extra) } - it_behaves_like "a simple event" - - describe "resource_text" do - it "returns the proposal body" do - expect(subject.resource_text).to be(resource.body) - end + before do + allow(subject).to receive(:participatory_space_title).and_return(participatory_process.title["en"]) end describe "email_subject" do it "matches the expected translation" do - expect(subject.email_subject).to be("Your proposal has been published!") - end - end - - describe "email_intro" do - it "matches the expected translation" do - expected_intro = "Your proposal \"#{resource_title}\" was successfully received and is now public. Thank you for participating! Here it is: #{resource_title}." - expect(subject.email_intro).to be(expected_intro) + expect(subject.email_subject).to eq("Your proposal has been published!") end end describe "email_outro" do it "matches the expected translation" do - expected_outro = "You have received this notification because you are the author of the proposal. You can unfollow it by going to the proposal page (\"#{resource_title}\") and clicking on \"Unfollow\"." - expect(subject.email_outro).to be(expected_outro) + expected_outro = "You received this notification because you are the author of the proposal" + expect(subject.email_outro).to include(expected_outro) end end - - describe "notification_title" do - it "matches the expected translation" do - expected_title = "Your proposal #{resource_title} is now live." - expect(subject.notification_title).to be(expected_title) - end - end - - describe "translated notifications" do - let(:en_body) { "A nice proposal" } - let(:body) { { en: en_body, machine_translations: { ca: "Une belle idee" } } } - let(:resource) do - create :extended_proposal, - component: proposal_component, - title: { en: "A nice proposal", machine_translations: { ca: "Une belle idee" } }, - body: body - end - - let(:en_version) { subject.resource_text["en"] } - let(:machine_translated) { subject.resource_text["machine_translations"]["ca"] } - let(:translatable) { true } - - it_behaves_like "a translated event" - end end end end diff --git a/spec/shared/simple_event.rb b/spec/shared/simple_event.rb deleted file mode 100644 index 1305adae9e..0000000000 --- a/spec/shared/simple_event.rb +++ /dev/null @@ -1,202 +0,0 @@ -# frozen_string_literal: true - -require "spec_helper" - -shared_context "when a simple event" do - include Decidim::SanitizeHelper - subject { event_instance } - - let(:event_instance) do - described_class.new( - resource: resource, - event_name: event_name, - user: user, - user_role: user_role, - extra: extra - ) - end - - let(:organization) do - if resource.respond_to?(:organization) - resource.organization - else - create :organization - end - end - let(:user) { create :user, organization: organization } - let(:user_role) { :follower } - let(:extra) { {} } - let(:resource_path) { resource_locator(resource).path } - let(:resource_url) { resource_locator(resource).url } - let(:resource_title) { decidim_sanitize(translated_attribute(resource.title)) } - # to be used when resource is a component resource, not a participatory space, in which case should be overriden - let(:participatory_space) { resource.participatory_space } - let(:participatory_space_title) { decidim_sanitize(translated_attribute(participatory_space.title)) } - let(:participatory_space_path) { Decidim::ResourceLocatorPresenter.new(participatory_space).path } - let(:participatory_space_url) { Decidim::ResourceLocatorPresenter.new(participatory_space).url } - let(:author) do - if resource.respond_to?(:creator_author) - resource.creator_author - else - resource.author - end - end - let(:author_presenter) { Decidim::UserPresenter.new(author) } - let(:author_name) { decidim_html_escape author.name } - let(:author_path) { author_presenter&.profile_path.to_s } - let(:author_nickname) { author_presenter&.nickname.to_s } - let(:i18n_scope) { event_name } -end - -shared_examples_for "a simple event" do |skip_space_checks| - describe "types" do - subject { described_class } - - it "supports notifications" do - expect(subject.types).to include :notification - end - - it "supports emails" do - expect(subject.types).to include :email - end - end - - describe "email_subject" do - it "is generated correctly" do - expect(subject.email_subject).to be_kind_of(String) - expect(subject.email_subject).not_to include("translation missing") - expect(subject.email_subject).not_to include("script") - end - end - - describe "email_intro" do - it "is generated correctly" do - expect(subject.email_intro).to be_kind_of(String) - expect(subject.email_intro).not_to include("translation missing") - end - end - - describe "email_outro" do - it "is generated correctly" do - expect(subject.email_outro).to be_kind_of(String) - expect(subject.email_outro).not_to include("translation missing") - end - end - - describe "email_greeting" do - it "is generated correctly" do - expect(subject.email_greeting).to be_kind_of(String) - expect(subject.email_greeting).not_to include("translation missing") - end - end - - describe "safe_resource_text" do - it "is generated correctly" do - expect(subject.safe_resource_text).to be_kind_of(String) - expect(subject.safe_resource_text).to be_html_safe - end - end - - describe "notification_title" do - it "is generated correctly" do - expect(subject.notification_title).to be_kind_of(String) - expect(subject.notification_title).not_to include("translation missing") - expect(subject.notification_title).not_to include("script") - end - end - - describe "resource_path" do - it "is generated correctly" do - expect(subject.resource_path).to be_kind_of(String) - end - end - - describe "resource_url" do - it "is generated correctly" do - expect(subject.resource_url).to be_kind_of(String) - expect(subject.resource_url).to start_with("http") - end - end - - describe "resource_title" do - it "responds to the method" do - expect(subject).to respond_to(:resource_title) - end - end - - unless skip_space_checks - describe "participatory_space_url" do - it "is generated correctly" do - expect(subject.participatory_space_url).to be_kind_of(String) - expect(subject.participatory_space_url).to start_with("http") - end - end - - describe "participatory_space_title" do - it "is generated correctly" do - expect(translated(participatory_space.title)).to include("script") - end - end - end - - describe "i18n_options" do - subject { super().i18n_options } - - it { is_expected.to include(resource_path: satisfy(&:present?)) } - it { is_expected.to include(resource_title: satisfy(&:present?)) } - it { is_expected.to include(resource_url: start_with("http")) } - - it "includes the i18n scope" do - if event_instance.event_has_roles? - expect(subject).to include(scope: "#{i18n_scope}.#{user_role}") - else - expect(subject).to include(scope: i18n_scope) - end - end - - unless skip_space_checks - it { is_expected.to include(participatory_space_title: satisfy(&:present?)) } - it { is_expected.to include(participatory_space_url: start_with("http")) } - end - end -end - -shared_examples_for "a simple event email" do - describe "email_subject" do - it "is generated correctly" do - expect(subject.email_subject).to eq(email_subject) - end - - # it "is html safe" do - # # pending "Enable after #12547 is merged" - # expect(subject.email_subject).not_to include("script") - # end - end - - describe "email_intro" do - it "is generated correctly" do - expect(subject.email_intro).to eq(email_intro) - end - end - - describe "email_outro" do - it "is generated correctly" do - expect(subject.email_outro).to eq(email_outro) - end - end -end - -shared_examples_for "a simple event notification" do - describe "notification_title" do - it "is generated correctly" do - expect(subject.notification_title) - .to eq(notification_title) - end - # - # it "is html safe" do - # pp subject.notification_title - # pending "Enable after #12547 is merged" - # expect(subject.notification_title).not_to include("script") - # end - end -end diff --git a/spec/shared/translated_event_examples.rb b/spec/shared/translated_event_examples.rb deleted file mode 100644 index 77938b01f4..0000000000 --- a/spec/shared/translated_event_examples.rb +++ /dev/null @@ -1,131 +0,0 @@ -# frozen_string_literal: true - -shared_examples_for "a translated event" do - context "when it is not machine machine translated" do - let(:organization) { create(:organization, enable_machine_translations: false, machine_translation_display_priority: "original") } - - it "does not perform translation" do - expect(subject.perform_translation?).to be(false) - end - - it "does not have a missing translation" do - expect(subject.translation_missing?).to be(false) - end - - it "does have content available in multiple languages" do - expect(subject.content_in_same_language?).to be(false) - end - - it "does return the original language" do - expect(subject.safe_resource_text).to be(en_version) - end - - it "does not offer an alternate translation" do - expect(subject.safe_resource_translated_text).to be(en_version) - end - end - - context "when is machine machine translated" do - let(:user) { create :user, organization: organization, locale: "ca" } - - around do |example| - I18n.with_locale(user.locale) { example.run } - end - - context "when priority is original" do - let(:organization) { create(:organization, enable_machine_translations: true, machine_translation_display_priority: "original") } - - it "does perform translation" do - expect(subject.perform_translation?).to be(translatable) - end - - it "does not have a missing translation" do - expect(subject.translation_missing?).to be(false) - end - - it "does have content available in multiple languages" do - expect(subject.content_in_same_language?).to be(false) - end - - it "does return the original language" do - expect(subject.safe_resource_text).to be(en_version) - end - - it "does not offer an alternate translation" do - expect(subject.safe_resource_translated_text).to be(machine_translated) - end - - context "when translation is not available" do - let(:body) { { en: en_body } } - - it "does perform translation" do - expect(subject.perform_translation?).to be(translatable) - end - - it "does have a missing translation" do - expect(subject.translation_missing?).to be(translatable) - end - - it "does have content available in multiple languages" do - expect(subject.content_in_same_language?).to be(false) - end - - it "does return the original language" do - expect(subject.safe_resource_text).to be(en_version) - end - - it "does not offer an alternate translation" do - expect(subject.safe_resource_translated_text).to be(en_version) - end - end - end - - context "when priority is translation" do - let(:organization) { create(:organization, enable_machine_translations: true, machine_translation_display_priority: "translation") } - - it "does perform translation" do - expect(subject.perform_translation?).to be(translatable) - end - - it "does not have a missing translation" do - expect(subject.translation_missing?).to be(false) - end - - it "does have content available in multiple languages" do - expect(subject.content_in_same_language?).to be(false) - end - - it "does return the original language" do - expect(subject.safe_resource_text).to be(en_version) - end - - it "does not offer an alternate translation" do - expect(subject.safe_resource_translated_text).to be(machine_translated) - end - - context "when translation is not available" do - let(:body) { { en: en_body } } - - it "does perform translation" do - expect(subject.perform_translation?).to be(translatable) - end - - it "does have a missing translation" do - expect(subject.translation_missing?).to be(translatable) - end - - it "does have content available in multiple languages" do - expect(subject.content_in_same_language?).to be(false) - end - - it "does return the original language" do - expect(subject.safe_resource_text).to be(en_version) - end - - it "does not offer an alternate translation" do - expect(subject.safe_resource_translated_text).to be(en_version) - end - end - end - end -end From 502452bb4492648c44b20e640aed53fb765cdd4d Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Wed, 13 Nov 2024 13:32:51 +0100 Subject: [PATCH 28/32] fix CI --- .../proposals/author_confirmation_proposal_event_spec.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb index af040aa66a..8a01f9d3a1 100644 --- a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb +++ b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb @@ -17,7 +17,7 @@ module Proposals subject { described_class.new(resource: resource, event_name: event_name, user: user, user_role: user_role, extra: extra) } before do - allow(subject).to receive(:participatory_space_title).and_return(participatory_process.title["en"]) + allow(subject).to receive(:participatory_space_title).and_return(participatory_process.title) end describe "email_subject" do From 76e36a656209a06242b44ae1a539aa54113b2663 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Wed, 13 Nov 2024 13:39:24 +0100 Subject: [PATCH 29/32] delete test file --- ...author_confirmation_proposal_event_spec.rb | 37 ------------------- 1 file changed, 37 deletions(-) delete mode 100644 spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb diff --git a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb b/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb deleted file mode 100644 index 8a01f9d3a1..0000000000 --- a/spec/events/decidim/proposals/author_confirmation_proposal_event_spec.rb +++ /dev/null @@ -1,37 +0,0 @@ -# frozen_string_literal: true - -require "spec_helper" - -module Decidim - module Proposals - describe AuthorConfirmationProposalEvent do - let(:user) { create :user, organization: organization } - let(:resource) { create :extended_proposal } - let(:participatory_process) { create :participatory_process, organization: organization } - let(:proposal_component) { create(:extended_proposal_component, participatory_space: participatory_process) } - let(:event_name) { "decidim.events.proposals.author_confirmation_proposal_event" } - let(:user_role) { :participant } - let(:extra) { {} } - let(:organization) { resource.organization || create(:organization) } - - subject { described_class.new(resource: resource, event_name: event_name, user: user, user_role: user_role, extra: extra) } - - before do - allow(subject).to receive(:participatory_space_title).and_return(participatory_process.title) - end - - describe "email_subject" do - it "matches the expected translation" do - expect(subject.email_subject).to eq("Your proposal has been published!") - end - end - - describe "email_outro" do - it "matches the expected translation" do - expected_outro = "You received this notification because you are the author of the proposal" - expect(subject.email_outro).to include(expected_outro) - end - end - end - end -end From d64a95b9cdd9c6cb2de1bfe591bf091ea09d0b7f Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Wed, 13 Nov 2024 15:04:47 +0100 Subject: [PATCH 30/32] add ignore trad key in i118n-tasks.yml to fix CI --- config/i18n-tasks.yml | 3 +++ 1 file changed, 3 insertions(+) diff --git a/config/i18n-tasks.yml b/config/i18n-tasks.yml index be5b5b9147..fef7c18f22 100644 --- a/config/i18n-tasks.yml +++ b/config/i18n-tasks.yml @@ -164,5 +164,8 @@ ignore_unused: - decidim.newsletters.unsubscribe.error - decidim.newsletters.unsubscribe.token_error - decidim.half_signup.quick_auth.sms_verification.text_message + - decidim.events.proposals.author_confirmation_proposal_event.email_intro + - decidim.events.proposals.author_confirmation_proposal_event.email_outro + - decidim.events.proposals.author_confirmation_proposal_event.notification_title From ae19d3cfb51ebabc548dacf17b06a9046ad069b6 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Thu, 14 Nov 2024 15:22:56 +0100 Subject: [PATCH 31/32] update text syntaxe via trad key --- config/locales/en.yml | 4 ++-- config/locales/fr.yml | 6 +++--- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/config/locales/en.yml b/config/locales/en.yml index e64c501a41..56be2f097d 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -79,8 +79,8 @@ en: notification_title: The initiative %{resource_title} has been answered. proposals: author_confirmation_proposal_event: - email_intro: 'Your proposal "%{resource_title}" was successfully received and is now public. Thank you for participating! You can view it here: %{resource_title}.' - email_outro: You received this notification because you are the author of the proposal. You can unfollow it by visiting the proposal page ("%{resource_title}") and clicking on "Unfollow". + email_intro: 'Your proposal " %{resource_title} " was successfully received and is now public. Thank you for participating ! You can view it here:' + email_outro: You received this notification because you are the author of the proposal. You can unfollow it by visiting the proposal page (" %{resource_title} ") and clicking on " Unfollow ". email_subject: Your proposal has been published! notification_title: Your proposal %{resource_title} is now live. users: diff --git a/config/locales/fr.yml b/config/locales/fr.yml index 5de02a32a2..300f0e15a4 100644 --- a/config/locales/fr.yml +++ b/config/locales/fr.yml @@ -81,10 +81,10 @@ fr: notification_title: La pétition %{resource_title} a reçu une réponse. proposals: author_confirmation_proposal_event: - email_intro: 'Votre proposition "%{resource_title}" a été reçue avec succès et est maintenant publique. Merci pour votre participation! Vous pouvez la consulter ici : %{resource_title}.' - email_outro: Vous recevez cette notification car vous êtes l’auteur de la proposition. Vous pouvez vous désabonner en visitant la page de la proposition ("%{resource_title}") et en cliquant sur "Ne plus suivre". + email_intro: 'Votre proposition « %{resource_title} » a été reçue avec succès et est maintenant publique. Merci pour votre participation ! Vous pouvez la consulter ici :' + email_outro: Vous recevez cette notification car vous êtes l’auteur de la proposition. Vous pouvez vous désabonner en visitant la page de la proposition (« %{resource_title} ») et en cliquant sur « Ne plus suivre ». email_subject: Votre proposition a été publiée ! - notification_title: Votre proposition %{resource_title} est maintenant en ligne. + notification_title: Votre proposition %{resource_title} est maintenant en ligne. users: user_officialized: email_intro: Le participant %{name} (%{nickname}) a été officialisé. From ad284ea07c83ffa6de5218f8fe18192f087860b5 Mon Sep 17 00:00:00 2001 From: barbara oliveira Date: Thu, 14 Nov 2024 15:44:05 +0100 Subject: [PATCH 32/32] add '' in fr trad key --- config/locales/fr.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/config/locales/fr.yml b/config/locales/fr.yml index 300f0e15a4..6e725c5912 100644 --- a/config/locales/fr.yml +++ b/config/locales/fr.yml @@ -84,7 +84,7 @@ fr: email_intro: 'Votre proposition « %{resource_title} » a été reçue avec succès et est maintenant publique. Merci pour votre participation ! Vous pouvez la consulter ici :' email_outro: Vous recevez cette notification car vous êtes l’auteur de la proposition. Vous pouvez vous désabonner en visitant la page de la proposition (« %{resource_title} ») et en cliquant sur « Ne plus suivre ». email_subject: Votre proposition a été publiée ! - notification_title: Votre proposition %{resource_title} est maintenant en ligne. + notification_title: Votre proposition %{resource_title} est maintenant en ligne. users: user_officialized: email_intro: Le participant %{name} (%{nickname}) a été officialisé.