From 9effa31f469781dfe8c051ad56cde505bed07e7b Mon Sep 17 00:00:00 2001 From: Ismael Bejarano Date: Wed, 4 Dec 2024 17:42:29 -0300 Subject: [PATCH 1/2] Wrap in a transaction operations to create, add and update a respondents list --- assets/js/actions/respondentGroups.js | 4 + lib/ask/runtime/respondent_group_action.ex | 123 ++++++++++++--------- 2 files changed, 76 insertions(+), 51 deletions(-) diff --git a/assets/js/actions/respondentGroups.js b/assets/js/actions/respondentGroups.js index 583144767..451f68232 100644 --- a/assets/js/actions/respondentGroups.js +++ b/assets/js/actions/respondentGroups.js @@ -102,6 +102,10 @@ const handleRespondentGroupUpload = (dispatch, promise, groupId = null) => { } else { dispatch(receiveInvalids(value)) } + }, + (err) => { + // Revert with original error + Promise.reject(e) }) } ) diff --git a/lib/ask/runtime/respondent_group_action.ex b/lib/ask/runtime/respondent_group_action.ex index ecf5bb247..8703b7618 100644 --- a/lib/ask/runtime/respondent_group_action.ex +++ b/lib/ask/runtime/respondent_group_action.ex @@ -21,27 +21,34 @@ defmodule Ask.Runtime.RespondentGroupAction do respondents_count = phone_numbers |> length - respondent_group = - %RespondentGroup{ - name: name, - survey_id: survey.id, - sample: sample, - respondents_count: respondents_count - } - |> Repo.insert!() - |> Repo.preload(:respondent_group_channels) - - insert_respondents(phone_numbers, respondent_group) - - survey - |> Repo.preload([:questionnaires]) - |> Repo.preload([:quota_buckets]) - |> Repo.preload(respondent_groups: [respondent_group_channels: :channel]) - |> Changeset.change() - |> Survey.update_state() - |> Repo.update!() + Repo.transaction(fn -> + respondent_group = + %RespondentGroup{ + name: name, + survey_id: survey.id, + sample: sample, + respondents_count: respondents_count + } + |> Repo.insert!() + |> Repo.preload(:respondent_group_channels) + + insert_respondents(phone_numbers, respondent_group) + + survey + |> Repo.preload([:questionnaires]) + |> Repo.preload([:quota_buckets]) + |> Repo.preload(respondent_groups: [respondent_group_channels: :channel]) + |> Changeset.change() + |> Survey.update_state() + |> Repo.update!() - respondent_group + respondent_group + end) + |> case do + {:ok, respondent_group} -> + respondent_group + err -> err + end end def add_respondents(respondent_group, loaded_entries, file_name, conn) do @@ -54,51 +61,65 @@ defmodule Ask.Runtime.RespondentGroupAction do loaded_entries = clean_entries(loaded_entries, phone_numbers) - insert_respondents(phone_numbers, respondent_group) + Repo.transaction(fn -> + insert_respondents(phone_numbers, respondent_group) - respondents_count = Enum.count(phone_numbers) + respondents_count = Enum.count(phone_numbers) - if Survey.launched?(survey) and respondents_count > 0 do - ActivityLog.add_respondents(survey.project, conn, survey, %{ - file_name: file_name, - respondents_count: respondents_count - }) - |> Repo.insert!() - end + if Survey.launched?(survey) and respondents_count > 0 do + ActivityLog.add_respondents(survey.project, conn, survey, %{ + file_name: file_name, + respondents_count: respondents_count + }) + |> Repo.insert!() + end - new_count = respondent_group.respondents_count + length(phone_numbers) + new_count = respondent_group.respondents_count + length(phone_numbers) - new_sample = merge_sample(respondent_group.sample, loaded_entries) + new_sample = merge_sample(respondent_group.sample, loaded_entries) - respondent_group - |> RespondentGroup.changeset(%{"respondents_count" => new_count, "sample" => new_sample}) - |> Repo.update!() + respondent_group + |> RespondentGroup.changeset(%{"respondents_count" => new_count, "sample" => new_sample}) + |> Repo.update!() + end) + |> case do + {:ok, respondent_group} -> + respondent_group + err -> err + end end def replace_respondents(respondent_group, loaded_entries) do respondent_group = Repo.preload(respondent_group, :survey) phone_numbers = map_phone_numbers_from_loaded_entries(loaded_entries) - # First delete existing respondents from that group - Repo.delete_all( - from(r in Respondent, - where: r.respondent_group_id == ^respondent_group.id + Repo.transaction(fn -> + # First delete existing respondents from that group + Repo.delete_all( + from(r in Respondent, + where: r.respondent_group_id == ^respondent_group.id + ) ) - ) - # Then create respondents from the CSV file - insert_respondents(phone_numbers, respondent_group) + # Then create respondents from the CSV file + insert_respondents(phone_numbers, respondent_group) - sample = take_sample(loaded_entries) - respondents_count = phone_numbers |> length + sample = take_sample(loaded_entries) + respondents_count = phone_numbers |> length - respondent_group - |> RespondentGroup.changeset(%{ - "sample" => sample, - "respondents_count" => respondents_count - }) - |> Repo.update!() - |> Repo.preload(:respondent_group_channels) + respondent_group + |> RespondentGroup.changeset(%{ + "sample" => sample, + "respondents_count" => respondents_count + }) + |> Repo.update!() + |> Repo.preload(:respondent_group_channels) + end) + |> case do + {:ok, respondent_group} -> + respondent_group + err -> err + end end defp clean_entries(loaded_entries, phone_numbers) do @@ -175,7 +196,7 @@ defmodule Ask.Runtime.RespondentGroupAction do end) end - def insert_respondents(phone_numbers, respondent_group) do + defp insert_respondents(phone_numbers, respondent_group) do respondent_group = Repo.preload(respondent_group, survey: :project) map_respondent = fn phone_number -> From 5339f9142c3e0c44c162c5b574ef3a8173b239ce Mon Sep 17 00:00:00 2001 From: Ismael Bejarano Date: Wed, 4 Dec 2024 17:46:38 -0300 Subject: [PATCH 2/2] Fix CI build --- assets/js/actions/respondentGroups.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/assets/js/actions/respondentGroups.js b/assets/js/actions/respondentGroups.js index 451f68232..503e4820d 100644 --- a/assets/js/actions/respondentGroups.js +++ b/assets/js/actions/respondentGroups.js @@ -103,8 +103,8 @@ const handleRespondentGroupUpload = (dispatch, promise, groupId = null) => { dispatch(receiveInvalids(value)) } }, - (err) => { - // Revert with original error + () => { + // Reject with original error Promise.reject(e) }) }