From 64b7c759101f7e1c81f6b2ee43579dec7dd37d4f Mon Sep 17 00:00:00 2001 From: Fredrik Jonsson <frjo@xdeb.org> Date: Tue, 4 Sep 2018 10:16:15 +0200 Subject: [PATCH] Migration of reviews. --- .../management/commands/migration_base.py | 2 +- opentech/apply/review/management/__init__.py | 0 .../review/management/commands/__init__.py | 0 .../commands/migrate_community_lab_reviews.py | 56 +++++ .../commands/migrate_concept_reviews.py | 182 +++++++++++++++ .../migrate_fellowship_application_reviews.py | 182 +++++++++++++++ .../migrate_fellowship_proposal_reviews.py | 178 +++++++++++++++ .../commands/migrate_proposal_reviews.py | 182 +++++++++++++++ .../management/commands/migrate_rr_reviews.py | 42 ++++ .../commands/migration_review_base.py | 213 ++++++++++++++++++ 10 files changed, 1036 insertions(+), 1 deletion(-) create mode 100644 opentech/apply/review/management/__init__.py create mode 100644 opentech/apply/review/management/commands/__init__.py create mode 100644 opentech/apply/review/management/commands/migrate_community_lab_reviews.py create mode 100644 opentech/apply/review/management/commands/migrate_concept_reviews.py create mode 100644 opentech/apply/review/management/commands/migrate_fellowship_application_reviews.py create mode 100644 opentech/apply/review/management/commands/migrate_fellowship_proposal_reviews.py create mode 100644 opentech/apply/review/management/commands/migrate_proposal_reviews.py create mode 100644 opentech/apply/review/management/commands/migrate_rr_reviews.py create mode 100644 opentech/apply/review/management/commands/migration_review_base.py diff --git a/opentech/apply/funds/management/commands/migration_base.py b/opentech/apply/funds/management/commands/migration_base.py index 77c2b73cc..c919238de 100644 --- a/opentech/apply/funds/management/commands/migration_base.py +++ b/opentech/apply/funds/management/commands/migration_base.py @@ -211,7 +211,7 @@ class MigrateCommand(BaseCommand): if option: value.append(option) elif mapping_type == 'file': - value = self.process_file(source_value) + value = [] # self.process_file(source_value) return value diff --git a/opentech/apply/review/management/__init__.py b/opentech/apply/review/management/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/opentech/apply/review/management/commands/__init__.py b/opentech/apply/review/management/commands/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/opentech/apply/review/management/commands/migrate_community_lab_reviews.py b/opentech/apply/review/management/commands/migrate_community_lab_reviews.py new file mode 100644 index 000000000..b74793094 --- /dev/null +++ b/opentech/apply/review/management/commands/migrate_community_lab_reviews.py @@ -0,0 +1,56 @@ +from opentech.apply.review.management.commands.migration_review_base import MigrateCommand + + +class Command(MigrateCommand): + LAB_NAME = "Community lab (archive fund)" + APPLICATION_TYPE = "request" + CONTENT_TYPE = "lab" + + STREAMFIELD_MAP = { + "field_pr_confidentiality": { + "id": "c1c6cedc-a084-4c55-87d5-7f6baf48441e", + "type": "boolean", + }, + "field_pr_conflicts": { + "id": "c29a7f43-009c-4341-bbe8-9582ba089d52", + "type": "map", + "map": { + "0": "No", + "1": "Yes", + }, + }, + "field_pr_disclosure": { + "id": "3aab69b1-6b60-4850-8f9f-7bc1b5871dcf", + "type": "value", + "key": "safe_value", + }, + "field_clr_remit_rate": { + "id": "732fc004-3086-44e1-8508-e0f17c3732a8", + "type": "score", + }, + "field_clr_like": { + "id": "f3c42cf1-e5ef-4674-bf6c-8e4640ee0d58", + "type": "value", + "key": "safe_value", + }, + "field_clr_not_like": { + "id": "e1e69628-c663-4cd2-a0ea-507ad01149de", + "type": "value", + "key": "safe_value", + }, + "field_clr_information": { + "id": "3033f228-58af-4944-b884-736fe6258bd6", + "type": "value", + "key": "safe_value", + }, + "field_clr_voices": { + "id": "20ec1ed7-4e3e-433c-944a-7c20cd6245c8", + "type": "value", + "key": "safe_value", + }, + "field_clr_reach_out": { + "id": "fd361c53-a263-4572-8403-74f6736d38fc", + "type": "value", + "key": "safe_value", + }, + } diff --git a/opentech/apply/review/management/commands/migrate_concept_reviews.py b/opentech/apply/review/management/commands/migrate_concept_reviews.py new file mode 100644 index 000000000..a1eae6870 --- /dev/null +++ b/opentech/apply/review/management/commands/migrate_concept_reviews.py @@ -0,0 +1,182 @@ +from opentech.apply.review.management.commands.migration_review_base import MigrateCommand + + +class Command(MigrateCommand): + CONTENT_TYPE = "fund" + FUND_NAME = "Internet Freedom Fund (archive fund)" + ROUND_NAME = "Internet Freedom Fund (archive round)" + APPLICATION_TYPE = "concept" + + STREAMFIELD_MAP = { + "title": { + "id": "title", + "type": "direct", + }, + "field_concept_name": { + "id": "full_name", + "type": "value", + "key": "safe_value", + }, + "field_concept_mail": { + "id": "email", + "type": "value", + "key": "email", + }, + "field_concept_preapplied": { + "id": "0305a465-8763-4c1f-9197-4ca4227d452a", + "type": "map", + "map": { + "0": "No", + "1": "Yes", + }, + }, + "field_concept_preapplied_how": { + "id": "c24691be-9861-4dbc-8be4-03b6e68c1973", + "type": "value", + "key": "safe_value", + }, + "field_concept_description": { + "id": "c21c58c3-cfbe-4409-b2f2-8f56398f1731", + "type": "value", + "key": "safe_value", + }, + "field_concept_outcome": { + "id": "27289c14-6926-4f61-bea2-8031a653f71c", + "type": "value", + "key": "safe_value", + }, + "field_type": { + "id": "404e2310-000b-4ccb-b772-3680946ff07d", + "type": "category", + "key": "tid", + }, + "field_proposal_status": { + "id": "145c364f-e0bb-4652-94e8-fe08c831da2b", + "type": "category", + "key": "tid", + }, + "field_technology_attribute": { + "id": "b4da2310-9654-4aa7-a04a-06335967ddc5", + "type": "category", + "key": "tid", + }, + "field_concept_how": { + "id": "418b8099-4525-437f-a55c-9b35745d0384", + "type": "value", + "key": "safe_value", + }, + "field_proposal_focus": { + "id": "390702bd-e4e1-4dc2-8c43-d51bf018b427", + "type": "category", + "key": "tid", + }, + "field_concept_time": { + "id": "duration", + "type": "value", + }, + "field_concept_amount": { + "id": "value", + "type": "value", + }, + "field_concept_how_long": { + "id": "dfb9c307-9328-4a99-9efc-321d474b2ba7", + "type": "value", + "key": "safe_value", + }, + "field_concept_who": { + "id": "11f94a22-0571-4491-a93e-87c050e02a4a", + "type": "value", + "key": "safe_value", + }, + "field_concept_community": { + "id": "c272969b-d89f-4b6e-859f-4606a15b3f28", + "type": "value", + "key": "safe_value", + }, + "field_proposal_beneficiaries": { + "id": "72002c3e-aaee-47da-9377-8bb493f14c21", + "type": "category", + "key": "tid", + }, + "field_term_region": { + "id": "369b248e-f669-4aeb-b771-7cba0eadb921", + "type": "category", + "key": "tid", + }, + "field_term_country": { + "id": "57bceb33-ebda-4708-9080-fd1a5923e008", + "type": "category", + "key": "tid", + }, + "field_concept_why": { + "id": "c14ee077-c0eb-48b1-9825-fbba9b91ede5", + "type": "value", + "key": "safe_value", + }, + "field_proposal_theme": { + "id": "a83a1884-f711-4196-8d15-ae2110466acb", + "type": "category", + "key": "tid", + }, + "field_proposal_similar_efforts": { + "id": "9ca35708-d611-4cd0-8d4a-3cc08349f45b", + "type": "value", + "key": "safe_value", + }, + "field_concept_contact": { + "id": "db7b1642-c03d-4af4-82c9-db67bf9713b0", + "type": "value", + "key": "safe_value", + }, + "field_concept_upload": { + "id": "8a91231c-5c3d-46fe-9de6-8d5c86817626", + "type": "file", + # TODO: finish mapping + }, + "field_application_otf_mission": { + "id": "4c661a64-2614-4169-b4d2-1fd39e2e831b", + "type": "boolean", + }, + "field_application_otf_tos": { + "id": "1bc4e113-1414-46ff-bba7-2dc02b2126df", + "type": "boolean", + }, + "field_application_otf_represent": { + "id": "42dd68a1-b699-4678-bea6-13e0f842e821", + "type": "boolean", + }, + "field_application_otf_license": { + "id": "72916731-ec97-4688-95f1-d3bf140b03c2", + "type": "boolean", + }, + "field_application_otf_complete": { + "id": "6856d26d-b169-4fdf-b598-63c3dd9278a2", + "type": "boolean", + }, + "field_application_otf_deadline": { + "id": "33838399-f292-4b63-83f0-e02d344f99d4", + "type": "boolean", + }, + "field_application_otf_list": { + "id": "fc571e12-d4a2-4d53-ab34-2c57321dc6ac", + "type": "boolean", + }, + "field_application_otf_newsletter": { + "id": "cd0d8a4b-e71a-4dff-964a-f547bd655e7d", + "type": "boolean", + }, + } + + REQUEST_QUESTION_MAP = { + "3618": { + 0: "Do the Goals and principles of the application align with the program?", + 1: "Does the application propose a unique contribution to the relevant field?", + 2: "Do you consider the application reasonable and realistic?", + 3: "General Comments", + }, + "3681": { + 0: "What are the positive aspects of this application?", + 1: "What are the negative aspects of this application?", + 2: "What items must the applicant address, if any?", + }, + } diff --git a/opentech/apply/review/management/commands/migrate_fellowship_application_reviews.py b/opentech/apply/review/management/commands/migrate_fellowship_application_reviews.py new file mode 100644 index 000000000..ca57de7e8 --- /dev/null +++ b/opentech/apply/review/management/commands/migrate_fellowship_application_reviews.py @@ -0,0 +1,182 @@ +from opentech.apply.review.management.commands.migration_review_base import MigrateCommand + + +class Command(MigrateCommand): + CONTENT_TYPE = "fund" + FUND_NAME = "Fellowship (archive fund)" + ROUND_NAME = "Fellowship (archive round)" + APPLICATION_TYPE = "concept" + + STREAMFIELD_MAP = { + "title": { + "id": "title", + "type": "direct", + }, + "field_concept_name": { + "id": "full_name", + "type": "value", + "key": "safe_value", + }, + "field_concept_mail": { + "id": "email", + "type": "value", + "key": "email", + }, + "field_concept_preapplied": { + "id": "0305a465-8763-4c1f-9197-4ca4227d452a", + "type": "map", + "map": { + "0": "No", + "1": "Yes", + }, + }, + "field_concept_preapplied_how": { + "id": "c24691be-9861-4dbc-8be4-03b6e68c1973", + "type": "value", + "key": "safe_value", + }, + "field_concept_description": { + "id": "c21c58c3-cfbe-4409-b2f2-8f56398f1731", + "type": "value", + "key": "safe_value", + }, + "field_concept_outcome": { + "id": "27289c14-6926-4f61-bea2-8031a653f71c", + "type": "value", + "key": "safe_value", + }, + "field_type": { + "id": "404e2310-000b-4ccb-b772-3680946ff07d", + "type": "category", + "key": "tid", + }, + "field_proposal_status": { + "id": "145c364f-e0bb-4652-94e8-fe08c831da2b", + "type": "category", + "key": "tid", + }, + "field_technology_attribute": { + "id": "b4da2310-9654-4aa7-a04a-06335967ddc5", + "type": "category", + "key": "tid", + }, + "field_concept_how": { + "id": "418b8099-4525-437f-a55c-9b35745d0384", + "type": "value", + "key": "safe_value", + }, + "field_proposal_focus": { + "id": "390702bd-e4e1-4dc2-8c43-d51bf018b427", + "type": "category", + "key": "tid", + }, + "field_concept_time": { + "id": "duration", + "type": "value", + }, + "field_concept_amount": { + "id": "value", + "type": "value", + }, + "field_concept_how_long": { + "id": "dfb9c307-9328-4a99-9efc-321d474b2ba7", + "type": "value", + "key": "safe_value", + }, + "field_concept_who": { + "id": "11f94a22-0571-4491-a93e-87c050e02a4a", + "type": "value", + "key": "safe_value", + }, + "field_concept_community": { + "id": "c272969b-d89f-4b6e-859f-4606a15b3f28", + "type": "value", + "key": "safe_value", + }, + "field_proposal_beneficiaries": { + "id": "72002c3e-aaee-47da-9377-8bb493f14c21", + "type": "category", + "key": "tid", + }, + "field_term_region": { + "id": "369b248e-f669-4aeb-b771-7cba0eadb921", + "type": "category", + "key": "tid", + }, + "field_term_country": { + "id": "57bceb33-ebda-4708-9080-fd1a5923e008", + "type": "category", + "key": "tid", + }, + "field_concept_why": { + "id": "c14ee077-c0eb-48b1-9825-fbba9b91ede5", + "type": "value", + "key": "safe_value", + }, + "field_proposal_theme": { + "id": "a83a1884-f711-4196-8d15-ae2110466acb", + "type": "category", + "key": "tid", + }, + "field_proposal_similar_efforts": { + "id": "9ca35708-d611-4cd0-8d4a-3cc08349f45b", + "type": "value", + "key": "safe_value", + }, + "field_concept_contact": { + "id": "db7b1642-c03d-4af4-82c9-db67bf9713b0", + "type": "value", + "key": "safe_value", + }, + "field_concept_upload": { + "id": "8a91231c-5c3d-46fe-9de6-8d5c86817626", + "type": "file", + # TODO: finish mapping + }, + "field_application_otf_mission": { + "id": "4c661a64-2614-4169-b4d2-1fd39e2e831b", + "type": "boolean", + }, + "field_application_otf_tos": { + "id": "1bc4e113-1414-46ff-bba7-2dc02b2126df", + "type": "boolean", + }, + "field_application_otf_represent": { + "id": "42dd68a1-b699-4678-bea6-13e0f842e821", + "type": "boolean", + }, + "field_application_otf_license": { + "id": "72916731-ec97-4688-95f1-d3bf140b03c2", + "type": "boolean", + }, + "field_application_otf_complete": { + "id": "6856d26d-b169-4fdf-b598-63c3dd9278a2", + "type": "boolean", + }, + "field_application_otf_deadline": { + "id": "33838399-f292-4b63-83f0-e02d344f99d4", + "type": "boolean", + }, + "field_application_otf_list": { + "id": "fc571e12-d4a2-4d53-ab34-2c57321dc6ac", + "type": "boolean", + }, + "field_application_otf_newsletter": { + "id": "cd0d8a4b-e71a-4dff-964a-f547bd655e7d", + "type": "boolean", + }, + } + + REQUEST_QUESTION_MAP = { + "3618": { + 0: "Do the Goals and principles of the application align with the program?", + 1: "Does the application propose a unique contribution to the relevant field?", + 2: "Do you consider the application reasonable and realistic?", + 3: "General Comments", + }, + "3681": { + 0: "What are the positive aspects of this application?", + 1: "What are the negative aspects of this application?", + 2: "What items must the applicant address, if any?", + }, + } diff --git a/opentech/apply/review/management/commands/migrate_fellowship_proposal_reviews.py b/opentech/apply/review/management/commands/migrate_fellowship_proposal_reviews.py new file mode 100644 index 000000000..75532af75 --- /dev/null +++ b/opentech/apply/review/management/commands/migrate_fellowship_proposal_reviews.py @@ -0,0 +1,178 @@ +from opentech.apply.review.management.commands.migration_review_base import MigrateCommand + + +class Command(MigrateCommand): + CONTENT_TYPE = "fund" + FUND_NAME = "Fellowship (archive fund)" + ROUND_NAME = "Fellowship (archive round)" + APPLICATION_TYPE = "proposal" + + STREAMFIELD_MAP = { + "title": { + "id": "title", + "type": "direct", + }, + "field_concept_name": { + "id": "full_name", + "type": "value", + "key": "safe_value", + }, + "field_concept_mail": { + "id": "email", + "type": "value", + "key": "email", + }, + "field_concept_preapplied": { + "id": "0305a465-8763-4c1f-9197-4ca4227d452a", + "type": "map", + "map": { + "0": "No", + "1": "Yes", + }, + }, + "field_concept_preapplied_how": { + "id": "c24691be-9861-4dbc-8be4-03b6e68c1973", + "type": "value", + "key": "safe_value", + }, + "field_concept_description": { + "id": "c21c58c3-cfbe-4409-b2f2-8f56398f1731", + "type": "value", + "key": "safe_value", + }, + "field_concept_outcome": { + "id": "27289c14-6926-4f61-bea2-8031a653f71c", + "type": "value", + "key": "safe_value", + }, + "field_type": { + "id": "404e2310-000b-4ccb-b772-3680946ff07d", + "type": "category", + "key": "tid", + }, + "field_proposal_status": { + "id": "145c364f-e0bb-4652-94e8-fe08c831da2b", + "type": "category", + "key": "tid", + }, + "field_technology_attribute": { + "id": "b4da2310-9654-4aa7-a04a-06335967ddc5", + "type": "category", + "key": "tid", + }, + "field_concept_how": { + "id": "418b8099-4525-437f-a55c-9b35745d0384", + "type": "value", + "key": "safe_value", + }, + "field_proposal_focus": { + "id": "390702bd-e4e1-4dc2-8c43-d51bf018b427", + "type": "category", + "key": "tid", + }, + "field_concept_time": { + "id": "duration", + "type": "value", + }, + "field_concept_amount": { + "id": "value", + "type": "value", + }, + "field_concept_how_long": { + "id": "dfb9c307-9328-4a99-9efc-321d474b2ba7", + "type": "value", + "key": "safe_value", + }, + "field_concept_who": { + "id": "11f94a22-0571-4491-a93e-87c050e02a4a", + "type": "value", + "key": "safe_value", + }, + "field_concept_community": { + "id": "c272969b-d89f-4b6e-859f-4606a15b3f28", + "type": "value", + "key": "safe_value", + }, + "field_proposal_beneficiaries": { + "id": "72002c3e-aaee-47da-9377-8bb493f14c21", + "type": "category", + "key": "tid", + }, + "field_term_region": { + "id": "369b248e-f669-4aeb-b771-7cba0eadb921", + "type": "category", + "key": "tid", + }, + "field_term_country": { + "id": "57bceb33-ebda-4708-9080-fd1a5923e008", + "type": "category", + "key": "tid", + }, + "field_concept_why": { + "id": "c14ee077-c0eb-48b1-9825-fbba9b91ede5", + "type": "value", + "key": "safe_value", + }, + "field_proposal_theme": { + "id": "a83a1884-f711-4196-8d15-ae2110466acb", + "type": "category", + "key": "tid", + }, + "field_proposal_similar_efforts": { + "id": "9ca35708-d611-4cd0-8d4a-3cc08349f45b", + "type": "value", + "key": "safe_value", + }, + "field_concept_contact": { + "id": "db7b1642-c03d-4af4-82c9-db67bf9713b0", + "type": "value", + "key": "safe_value", + }, + "field_concept_upload": { + "id": "8a91231c-5c3d-46fe-9de6-8d5c86817626", + "type": "file", + # TODO: finish mapping + }, + "field_application_otf_mission": { + "id": "4c661a64-2614-4169-b4d2-1fd39e2e831b", + "type": "boolean", + }, + "field_application_otf_tos": { + "id": "1bc4e113-1414-46ff-bba7-2dc02b2126df", + "type": "boolean", + }, + "field_application_otf_represent": { + "id": "42dd68a1-b699-4678-bea6-13e0f842e821", + "type": "boolean", + }, + "field_application_otf_license": { + "id": "72916731-ec97-4688-95f1-d3bf140b03c2", + "type": "boolean", + }, + "field_application_otf_complete": { + "id": "6856d26d-b169-4fdf-b598-63c3dd9278a2", + "type": "boolean", + }, + "field_application_otf_deadline": { + "id": "33838399-f292-4b63-83f0-e02d344f99d4", + "type": "boolean", + }, + "field_application_otf_list": { + "id": "fc571e12-d4a2-4d53-ab34-2c57321dc6ac", + "type": "boolean", + }, + "field_application_otf_newsletter": { + "id": "cd0d8a4b-e71a-4dff-964a-f547bd655e7d", + "type": "boolean", + }, + } + + REQUEST_QUESTION_MAP = { + "3618": { + 0: "Are the project objectives and timeline realistic?", + 1: "Should additional collaboration be included?", + 2: "Are the proposed outputs tailored to improve the likelihood of third party use such as utilizing more digestible and short form formats?", + 3: "Are there clear means of assessing the success of the project?", + 4: "General Comments", + }, + } diff --git a/opentech/apply/review/management/commands/migrate_proposal_reviews.py b/opentech/apply/review/management/commands/migrate_proposal_reviews.py new file mode 100644 index 000000000..38f1d3f85 --- /dev/null +++ b/opentech/apply/review/management/commands/migrate_proposal_reviews.py @@ -0,0 +1,182 @@ +from opentech.apply.review.management.commands.migration_review_base import MigrateCommand + + +class Command(MigrateCommand): + CONTENT_TYPE = "fund" + FUND_NAME = "Internet Freedom Fund (archive fund)" + ROUND_NAME = "Internet Freedom Fund (archive round)" + APPLICATION_TYPE = "proposal" + + STREAMFIELD_MAP = { + "title": { + "id": "title", + "type": "direct", + }, + "field_concept_name": { + "id": "full_name", + "type": "value", + "key": "safe_value", + }, + "field_concept_mail": { + "id": "email", + "type": "value", + "key": "email", + }, + "field_concept_preapplied": { + "id": "0305a465-8763-4c1f-9197-4ca4227d452a", + "type": "map", + "map": { + "0": "No", + "1": "Yes", + }, + }, + "field_concept_preapplied_how": { + "id": "c24691be-9861-4dbc-8be4-03b6e68c1973", + "type": "value", + "key": "safe_value", + }, + "field_concept_description": { + "id": "c21c58c3-cfbe-4409-b2f2-8f56398f1731", + "type": "value", + "key": "safe_value", + }, + "field_concept_outcome": { + "id": "27289c14-6926-4f61-bea2-8031a653f71c", + "type": "value", + "key": "safe_value", + }, + "field_type": { + "id": "404e2310-000b-4ccb-b772-3680946ff07d", + "type": "category", + "key": "tid", + }, + "field_proposal_status": { + "id": "145c364f-e0bb-4652-94e8-fe08c831da2b", + "type": "category", + "key": "tid", + }, + "field_technology_attribute": { + "id": "b4da2310-9654-4aa7-a04a-06335967ddc5", + "type": "category", + "key": "tid", + }, + "field_concept_how": { + "id": "418b8099-4525-437f-a55c-9b35745d0384", + "type": "value", + "key": "safe_value", + }, + "field_proposal_focus": { + "id": "390702bd-e4e1-4dc2-8c43-d51bf018b427", + "type": "category", + "key": "tid", + }, + "field_concept_time": { + "id": "duration", + "type": "value", + }, + "field_concept_amount": { + "id": "value", + "type": "value", + }, + "field_concept_how_long": { + "id": "dfb9c307-9328-4a99-9efc-321d474b2ba7", + "type": "value", + "key": "safe_value", + }, + "field_concept_who": { + "id": "11f94a22-0571-4491-a93e-87c050e02a4a", + "type": "value", + "key": "safe_value", + }, + "field_concept_community": { + "id": "c272969b-d89f-4b6e-859f-4606a15b3f28", + "type": "value", + "key": "safe_value", + }, + "field_proposal_beneficiaries": { + "id": "72002c3e-aaee-47da-9377-8bb493f14c21", + "type": "category", + "key": "tid", + }, + "field_term_region": { + "id": "369b248e-f669-4aeb-b771-7cba0eadb921", + "type": "category", + "key": "tid", + }, + "field_term_country": { + "id": "57bceb33-ebda-4708-9080-fd1a5923e008", + "type": "category", + "key": "tid", + }, + "field_concept_why": { + "id": "c14ee077-c0eb-48b1-9825-fbba9b91ede5", + "type": "value", + "key": "safe_value", + }, + "field_proposal_theme": { + "id": "a83a1884-f711-4196-8d15-ae2110466acb", + "type": "category", + "key": "tid", + }, + "field_proposal_similar_efforts": { + "id": "9ca35708-d611-4cd0-8d4a-3cc08349f45b", + "type": "value", + "key": "safe_value", + }, + "field_concept_contact": { + "id": "db7b1642-c03d-4af4-82c9-db67bf9713b0", + "type": "value", + "key": "safe_value", + }, + "field_concept_upload": { + "id": "8a91231c-5c3d-46fe-9de6-8d5c86817626", + "type": "file", + # TODO: finish mapping + }, + "field_application_otf_mission": { + "id": "4c661a64-2614-4169-b4d2-1fd39e2e831b", + "type": "boolean", + }, + "field_application_otf_tos": { + "id": "1bc4e113-1414-46ff-bba7-2dc02b2126df", + "type": "boolean", + }, + "field_application_otf_represent": { + "id": "42dd68a1-b699-4678-bea6-13e0f842e821", + "type": "boolean", + }, + "field_application_otf_license": { + "id": "72916731-ec97-4688-95f1-d3bf140b03c2", + "type": "boolean", + }, + "field_application_otf_complete": { + "id": "6856d26d-b169-4fdf-b598-63c3dd9278a2", + "type": "boolean", + }, + "field_application_otf_deadline": { + "id": "33838399-f292-4b63-83f0-e02d344f99d4", + "type": "boolean", + }, + "field_application_otf_list": { + "id": "fc571e12-d4a2-4d53-ab34-2c57321dc6ac", + "type": "boolean", + }, + "field_application_otf_newsletter": { + "id": "cd0d8a4b-e71a-4dff-964a-f547bd655e7d", + "type": "boolean", + }, + } + + REQUEST_QUESTION_MAP = { + "3618": { + 0: "Do the Goals and principles of the application align with the program?", + 1: "Does the application propose a unique contribution to the relevant field?", + 2: "Do you consider the application reasonable and realistic?", + 3: "General Comments", + }, + "3681": { + 0: "What are the positive aspects of this application?", + 1: "What are the negative aspects of this application?", + 2: "What items must the applicant address, if any?", + }, + } diff --git a/opentech/apply/review/management/commands/migrate_rr_reviews.py b/opentech/apply/review/management/commands/migrate_rr_reviews.py new file mode 100644 index 000000000..4d17c11fc --- /dev/null +++ b/opentech/apply/review/management/commands/migrate_rr_reviews.py @@ -0,0 +1,42 @@ +from opentech.apply.review.management.commands.migration_review_base import MigrateCommand + + +class Command(MigrateCommand): + CONTENT_TYPE = "fund" + FUND_NAME = "Rapid Response (archive fund)" + ROUND_NAME = "Rapid Response (archive round)" + APPLICATION_TYPE = "request" + + STREAMFIELD_MAP = { + "field_rrr_recommend": { + "id": "recommendation", + "type": "value", + }, + "field_rrr_overall_yes": { + "id": "cec815a0-fab1-4142-9fc6-71319b054b2a", + "type": "value", + "key": "safe_value", + }, + "field_rrr_overall_no": { + "id": "6915acf0-9a19-4e73-8d2b-d96e39e3b00e", + "type": "value", + "key": "safe_value", + }, + "field_rrr_objectives_rate": { + "id": "71bfe95d-89c5-401b-ae7a-778e91d5c8c5", + "type": "score", + }, + "field_rrr_capacity_rate": { + "id": "3aa164c1-4386-4046-997a-a2778e1d894e", + "type": "score", + }, + "field_rrr_justification_rate": { + "id": "7cc12bb6-4c12-48aa-a269-1fd6d725abfe", + "type": "score", + }, + "field_rrr_feedback": { + "id": "comments", + "type": "value", + "key": "safe_value", + }, + } diff --git a/opentech/apply/review/management/commands/migration_review_base.py b/opentech/apply/review/management/commands/migration_review_base.py new file mode 100644 index 000000000..3a14761c4 --- /dev/null +++ b/opentech/apply/review/management/commands/migration_review_base.py @@ -0,0 +1,213 @@ +import argparse +import json + +from datetime import datetime, timezone + +from django.contrib.auth import get_user_model +from django.core.management.base import BaseCommand +from django.db import transaction +from django.db.utils import IntegrityError + +from opentech.apply.funds.models import ApplicationSubmission, Round, LabType +from opentech.apply.funds.models.forms import RoundBaseReviewForm, LabBaseReviewForm +from opentech.apply.review.models import Review + + +class MigrateCommand(BaseCommand): + help = "Review migration script. Requires a source JSON file." + data = [] + + REVIEWFIELD_MAP = { + "concept_note_review": { + "submission": "field_review_concept_note", + "recommendation": "field_pr_recommendation", + "rec_map": { + "2": 1, + "1": 2, + "0": 0, + }, + }, + "proposal_review": { + "submission": "field_review_proposal", + "recommendation": "field_pr_recommendation", + "rec_map": { + "2": 1, + "1": 2, + "0": 0, + }, + }, + "fellowship_application_review": { + "submission": "field_review_fellowship_app", + "recommendation": "field_fr_overall_rate", + "rec_map": { + "5": 2, + "4": 2, + "3": 1, + "2": 0, + "1": 0, + }, + }, + "fellowship_proposal_review": { + "submission": "field_review_fellowship", + "recommendation": "field_fr_overall_rate", + "rec_map": { + "5": 2, + "4": 2, + "3": 1, + "2": 0, + "1": 0, + }, + }, + "rapid_response_review": { + "submission": "field_review_rapid_response", + "recommendation": "field_rrr_recommend", + "rec_map": { + "3": 2, + "2": 1, + "1": 0, + }, + }, + "community_lab_review": { + "submission": "field_review_community_lab", + "recommendation": "field_clr_recommendation", + "rec_map": { + "1": 2, + "0": 0, + }, + }, + } + + def add_arguments(self, parser): + parser.add_argument('source', type=argparse.FileType('r'), help='Migration source JSON file') + + @transaction.atomic + def handle(self, *args, **options): + with options['source'] as json_data: + self.data = json.load(json_data) + + counter = 0 + for id in self.data: + self.process(id) + counter += 1 + + self.stdout.write(f"Imported {counter} reviews.") + + def process(self, id): + node = self.data[id] + + try: + review = Review.objects.get(drupal_id=node['nid']) + except Review.DoesNotExist: + review = Review(drupal_id=node['nid']) + + # TODO timezone? + review.submit_time = datetime.fromtimestamp(int(node['created']), timezone.utc) + review.user = self.get_user(node['uid']) + review.recommendation = self.get_recommendation(node) + review.submission = self.get_submission(node) + + if self.CONTENT_TYPE == "fund": + ROUND = Round.objects.get(title=self.ROUND_NAME) + if self.APPLICATION_TYPE == "request": + FORM = RoundBaseReviewForm.objects.get(round=ROUND) + elif self.APPLICATION_TYPE == "concept": + FORM = RoundBaseReviewForm.objects.filter(round=ROUND)[0] + elif self.APPLICATION_TYPE == "proposal": + FORM = RoundBaseReviewForm.objects.filter(round=ROUND)[1] + review.form_fields = FORM.form.form_fields + elif self.CONTENT_TYPE == "lab": + LAB = LabType.objects.get(title=self.LAB_NAME) + FORM = LabBaseReviewForm.objects.get(lab=LAB) + review.form_fields = FORM.form.form_fields + + form_data = {} + + for field in node: + if field in self.STREAMFIELD_MAP: + try: + id = self.STREAMFIELD_MAP[field]['id'] + form_data[id] = self.get_field_value(field, node) + except TypeError: + pass + + review.form_data = form_data + + try: + review.save() + self.stdout.write(f"Processed \"{node['title']}\" ({node['nid']})") + except IntegrityError: + self.stdout.write(f"*** Skipped \"{node['title']}\" ({node['nid']}) due to IntegrityError") + + def get_user(self, uid): + try: + User = get_user_model() + return User.objects.get(drupal_id=uid) + except User.DoesNotExist: + return None + + def get_field_value(self, field, node): + """ + Handles the following formats: + field: {(safe_)value: VALUE} + field: {target_id: ID} -- Drupal ForeignKey. Reference to other node or user entities. + field: {tid: ID} -- or term ID. fk to Categories + field: [] + field: [{value|target_id|tid: VALUE},] + """ + mapping = self.STREAMFIELD_MAP[field] + mapping_type = mapping['type'] + key = mapping.get('key', 'value') + source_value = node[field] + value = None + + if mapping_type == "direct": + value = source_value + elif mapping_type == 'value': + if key in source_value: + value = self.nl2br(source_value[key]) if source_value else '' + else: + value = self.nl2br(source_value['value']) if source_value else '' + elif mapping_type == 'merge_value': + values = [] + i = 0 + for item in source_value: + question = self.REQUEST_QUESTION_MAP[node['field_application_request']['target_id']] + values.append(f"<strong>{question[i]}</strong>{item[key]}<br>\n") + i += 1 + merged_values = ''.join(values) + value = self.nl2br(merged_values) if source_value else '' + elif mapping_type == 'score': + value_rate = int(source_value[key]) if source_value else None + value_text = '' + if "_rate" in field: + text_field = field[:-5] + if text_field in self.STREAMFIELD_MAP: + value_text = self.nl2br(node[text_field]['safe_value']) + value = [value_text, value_rate] + elif mapping_type == 'map' and 'map' in 'mapping': + value = mapping['map'].get(source_value[key]) + elif mapping_type == 'boolean': + value = source_value[key] == '1' if source_value else False + + return value + + def get_recommendation(self, node): + mapping = self.REVIEWFIELD_MAP[node['type']] + field_name = mapping['recommendation'] + rec_map = mapping.get('rec_map') + try: + return rec_map[node[field_name]['value']] + except TypeError: + return 0 + + def get_submission(self, node): + mapping = self.REVIEWFIELD_MAP[node['type']] + field_name = mapping['submission'] + try: + nid = node[field_name]['target_id'] + return ApplicationSubmission.objects.get(drupal_id=nid) + except ApplicationSubmission.DoesNotExist: + return None + + def nl2br(self, value): + return value.replace('\r\n', '<br>\n') -- GitLab