cumulusci-plus 5.0.0__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of cumulusci-plus might be problematic. Click here for more details.

Files changed (744) hide show
  1. cumulusci/__about__.py +1 -0
  2. cumulusci/__init__.py +22 -0
  3. cumulusci/__main__.py +3 -0
  4. cumulusci/cli/__init__.py +0 -0
  5. cumulusci/cli/cci.py +244 -0
  6. cumulusci/cli/error.py +125 -0
  7. cumulusci/cli/flow.py +185 -0
  8. cumulusci/cli/logger.py +72 -0
  9. cumulusci/cli/org.py +692 -0
  10. cumulusci/cli/plan.py +181 -0
  11. cumulusci/cli/project.py +391 -0
  12. cumulusci/cli/robot.py +116 -0
  13. cumulusci/cli/runtime.py +190 -0
  14. cumulusci/cli/service.py +521 -0
  15. cumulusci/cli/task.py +295 -0
  16. cumulusci/cli/tests/__init__.py +0 -0
  17. cumulusci/cli/tests/test_cci.py +545 -0
  18. cumulusci/cli/tests/test_error.py +170 -0
  19. cumulusci/cli/tests/test_flow.py +276 -0
  20. cumulusci/cli/tests/test_logger.py +25 -0
  21. cumulusci/cli/tests/test_org.py +1438 -0
  22. cumulusci/cli/tests/test_plan.py +245 -0
  23. cumulusci/cli/tests/test_project.py +235 -0
  24. cumulusci/cli/tests/test_robot.py +177 -0
  25. cumulusci/cli/tests/test_runtime.py +197 -0
  26. cumulusci/cli/tests/test_service.py +853 -0
  27. cumulusci/cli/tests/test_task.py +266 -0
  28. cumulusci/cli/tests/test_ui.py +310 -0
  29. cumulusci/cli/tests/test_utils.py +122 -0
  30. cumulusci/cli/tests/utils.py +52 -0
  31. cumulusci/cli/ui.py +234 -0
  32. cumulusci/cli/utils.py +150 -0
  33. cumulusci/conftest.py +181 -0
  34. cumulusci/core/__init__.py +0 -0
  35. cumulusci/core/config/BaseConfig.py +5 -0
  36. cumulusci/core/config/BaseTaskFlowConfig.py +5 -0
  37. cumulusci/core/config/OrgConfig.py +5 -0
  38. cumulusci/core/config/ScratchOrgConfig.py +5 -0
  39. cumulusci/core/config/__init__.py +125 -0
  40. cumulusci/core/config/base_config.py +111 -0
  41. cumulusci/core/config/base_task_flow_config.py +82 -0
  42. cumulusci/core/config/marketing_cloud_service_config.py +83 -0
  43. cumulusci/core/config/oauth2_service_config.py +17 -0
  44. cumulusci/core/config/org_config.py +604 -0
  45. cumulusci/core/config/project_config.py +782 -0
  46. cumulusci/core/config/scratch_org_config.py +251 -0
  47. cumulusci/core/config/sfdx_org_config.py +220 -0
  48. cumulusci/core/config/tests/_test_config_backwards_compatibility.py +33 -0
  49. cumulusci/core/config/tests/test_config.py +1895 -0
  50. cumulusci/core/config/tests/test_config_expensive.py +839 -0
  51. cumulusci/core/config/tests/test_config_util.py +91 -0
  52. cumulusci/core/config/universal_config.py +88 -0
  53. cumulusci/core/config/util.py +18 -0
  54. cumulusci/core/datasets.py +303 -0
  55. cumulusci/core/debug.py +33 -0
  56. cumulusci/core/dependencies/__init__.py +55 -0
  57. cumulusci/core/dependencies/base.py +561 -0
  58. cumulusci/core/dependencies/dependencies.py +273 -0
  59. cumulusci/core/dependencies/github.py +177 -0
  60. cumulusci/core/dependencies/github_resolvers.py +244 -0
  61. cumulusci/core/dependencies/resolvers.py +580 -0
  62. cumulusci/core/dependencies/tests/__init__.py +0 -0
  63. cumulusci/core/dependencies/tests/conftest.py +385 -0
  64. cumulusci/core/dependencies/tests/test_dependencies.py +950 -0
  65. cumulusci/core/dependencies/tests/test_github.py +83 -0
  66. cumulusci/core/dependencies/tests/test_resolvers.py +1027 -0
  67. cumulusci/core/dependencies/utils.py +13 -0
  68. cumulusci/core/enums.py +11 -0
  69. cumulusci/core/exceptions.py +311 -0
  70. cumulusci/core/flowrunner.py +888 -0
  71. cumulusci/core/github.py +665 -0
  72. cumulusci/core/keychain/__init__.py +24 -0
  73. cumulusci/core/keychain/base_project_keychain.py +441 -0
  74. cumulusci/core/keychain/encrypted_file_project_keychain.py +945 -0
  75. cumulusci/core/keychain/environment_project_keychain.py +7 -0
  76. cumulusci/core/keychain/serialization.py +152 -0
  77. cumulusci/core/keychain/subprocess_keychain.py +24 -0
  78. cumulusci/core/keychain/tests/conftest.py +50 -0
  79. cumulusci/core/keychain/tests/test_base_project_keychain.py +299 -0
  80. cumulusci/core/keychain/tests/test_encrypted_file_project_keychain.py +1228 -0
  81. cumulusci/core/metadeploy/__init__.py +0 -0
  82. cumulusci/core/metadeploy/api.py +88 -0
  83. cumulusci/core/metadeploy/plans.py +25 -0
  84. cumulusci/core/metadeploy/tests/test_api.py +276 -0
  85. cumulusci/core/runtime.py +115 -0
  86. cumulusci/core/sfdx.py +162 -0
  87. cumulusci/core/source/__init__.py +16 -0
  88. cumulusci/core/source/github.py +50 -0
  89. cumulusci/core/source/local_folder.py +35 -0
  90. cumulusci/core/source_transforms/__init__.py +0 -0
  91. cumulusci/core/source_transforms/tests/test_transforms.py +1091 -0
  92. cumulusci/core/source_transforms/transforms.py +532 -0
  93. cumulusci/core/tasks.py +404 -0
  94. cumulusci/core/template_utils.py +59 -0
  95. cumulusci/core/tests/__init__.py +0 -0
  96. cumulusci/core/tests/cassettes/TestDatasetsE2E.test_datasets_e2e.yaml +215 -0
  97. cumulusci/core/tests/cassettes/TestDatasetsE2E.test_datasets_extract_standard_objects.yaml +199 -0
  98. cumulusci/core/tests/cassettes/TestDatasetsE2E.test_datasets_read_explicit_extract_declaration.yaml +3 -0
  99. cumulusci/core/tests/fake_remote_repo/cumulusci.yml +32 -0
  100. cumulusci/core/tests/fake_remote_repo/tasks/directory/example_2.py +6 -0
  101. cumulusci/core/tests/fake_remote_repo/tasks/example.py +43 -0
  102. cumulusci/core/tests/fake_remote_repo_2/cumulusci.yml +11 -0
  103. cumulusci/core/tests/fake_remote_repo_2/tasks/example_3.py +6 -0
  104. cumulusci/core/tests/test_datasets_e2e.py +386 -0
  105. cumulusci/core/tests/test_exceptions.py +11 -0
  106. cumulusci/core/tests/test_flowrunner.py +836 -0
  107. cumulusci/core/tests/test_github.py +942 -0
  108. cumulusci/core/tests/test_sfdx.py +138 -0
  109. cumulusci/core/tests/test_source.py +678 -0
  110. cumulusci/core/tests/test_tasks.py +262 -0
  111. cumulusci/core/tests/test_utils.py +141 -0
  112. cumulusci/core/tests/test_utils_merge_config.py +276 -0
  113. cumulusci/core/tests/test_versions.py +76 -0
  114. cumulusci/core/tests/untrusted_repo_child/cumulusci.yml +7 -0
  115. cumulusci/core/tests/untrusted_repo_child/tasks/untrusted_child.py +6 -0
  116. cumulusci/core/tests/untrusted_repo_parent/cumulusci.yml +26 -0
  117. cumulusci/core/tests/untrusted_repo_parent/tasks/untrusted_parent.py +6 -0
  118. cumulusci/core/tests/utils.py +116 -0
  119. cumulusci/core/tests/yaml/global.yaml +0 -0
  120. cumulusci/core/utils.py +402 -0
  121. cumulusci/core/versions.py +149 -0
  122. cumulusci/cumulusci.yml +1621 -0
  123. cumulusci/files/admin_profile.xml +20 -0
  124. cumulusci/files/delete_excludes.txt +424 -0
  125. cumulusci/files/templates/project/README.md +12 -0
  126. cumulusci/files/templates/project/cumulusci.yml +63 -0
  127. cumulusci/files/templates/project/dot-gitignore +60 -0
  128. cumulusci/files/templates/project/mapping.yml +45 -0
  129. cumulusci/files/templates/project/scratch_def.json +25 -0
  130. cumulusci/oauth/__init__.py +0 -0
  131. cumulusci/oauth/client.py +400 -0
  132. cumulusci/oauth/exceptions.py +9 -0
  133. cumulusci/oauth/salesforce.py +95 -0
  134. cumulusci/oauth/tests/__init__.py +0 -0
  135. cumulusci/oauth/tests/cassettes/test_get_device_code.yaml +22 -0
  136. cumulusci/oauth/tests/cassettes/test_get_device_oauth_token.yaml +74 -0
  137. cumulusci/oauth/tests/test_client.py +308 -0
  138. cumulusci/oauth/tests/test_salesforce.py +46 -0
  139. cumulusci/plugins/__init__.py +3 -0
  140. cumulusci/plugins/plugin_base.py +93 -0
  141. cumulusci/plugins/plugin_loader.py +59 -0
  142. cumulusci/robotframework/CumulusCI.py +340 -0
  143. cumulusci/robotframework/CumulusCI.robot +7 -0
  144. cumulusci/robotframework/Performance.py +165 -0
  145. cumulusci/robotframework/Salesforce.py +936 -0
  146. cumulusci/robotframework/Salesforce.robot +192 -0
  147. cumulusci/robotframework/SalesforceAPI.py +416 -0
  148. cumulusci/robotframework/SalesforcePlaywright.py +220 -0
  149. cumulusci/robotframework/SalesforcePlaywright.robot +40 -0
  150. cumulusci/robotframework/__init__.py +2 -0
  151. cumulusci/robotframework/base_library.py +39 -0
  152. cumulusci/robotframework/faker_mixin.py +89 -0
  153. cumulusci/robotframework/form_handlers.py +222 -0
  154. cumulusci/robotframework/javascript/cci_init.js +34 -0
  155. cumulusci/robotframework/javascript/cumulusci.js +4 -0
  156. cumulusci/robotframework/locator_manager.py +197 -0
  157. cumulusci/robotframework/locators_56.py +88 -0
  158. cumulusci/robotframework/locators_57.py +5 -0
  159. cumulusci/robotframework/pageobjects/BasePageObjects.py +433 -0
  160. cumulusci/robotframework/pageobjects/ObjectManagerPageObject.py +246 -0
  161. cumulusci/robotframework/pageobjects/PageObjectLibrary.py +45 -0
  162. cumulusci/robotframework/pageobjects/PageObjects.py +351 -0
  163. cumulusci/robotframework/pageobjects/__init__.py +12 -0
  164. cumulusci/robotframework/pageobjects/baseobjects.py +120 -0
  165. cumulusci/robotframework/perftests/short/collection_perf.robot +105 -0
  166. cumulusci/robotframework/tests/CustomObjectTestPage.py +10 -0
  167. cumulusci/robotframework/tests/FooTestPage.py +8 -0
  168. cumulusci/robotframework/tests/cumulusci/base.robot +40 -0
  169. cumulusci/robotframework/tests/cumulusci/bulkdata.robot +38 -0
  170. cumulusci/robotframework/tests/cumulusci/communities.robot +57 -0
  171. cumulusci/robotframework/tests/cumulusci/datagen.robot +84 -0
  172. cumulusci/robotframework/tests/salesforce/TestLibraryA.py +24 -0
  173. cumulusci/robotframework/tests/salesforce/TestLibraryB.py +20 -0
  174. cumulusci/robotframework/tests/salesforce/TestListener.py +93 -0
  175. cumulusci/robotframework/tests/salesforce/api.robot +178 -0
  176. cumulusci/robotframework/tests/salesforce/browsers.robot +143 -0
  177. cumulusci/robotframework/tests/salesforce/classic.robot +51 -0
  178. cumulusci/robotframework/tests/salesforce/create_contact.robot +59 -0
  179. cumulusci/robotframework/tests/salesforce/faker.robot +68 -0
  180. cumulusci/robotframework/tests/salesforce/forms.robot +172 -0
  181. cumulusci/robotframework/tests/salesforce/label_locator.robot +244 -0
  182. cumulusci/robotframework/tests/salesforce/labels.html +33 -0
  183. cumulusci/robotframework/tests/salesforce/locators.robot +149 -0
  184. cumulusci/robotframework/tests/salesforce/pageobjects/base_pageobjects.robot +100 -0
  185. cumulusci/robotframework/tests/salesforce/pageobjects/example_page_object.py +25 -0
  186. cumulusci/robotframework/tests/salesforce/pageobjects/listing_page.robot +115 -0
  187. cumulusci/robotframework/tests/salesforce/pageobjects/objectmanager.robot +74 -0
  188. cumulusci/robotframework/tests/salesforce/pageobjects/pageobjects.robot +171 -0
  189. cumulusci/robotframework/tests/salesforce/performance.robot +109 -0
  190. cumulusci/robotframework/tests/salesforce/playwright/javascript_keywords.robot +33 -0
  191. cumulusci/robotframework/tests/salesforce/playwright/open_test_browser.robot +48 -0
  192. cumulusci/robotframework/tests/salesforce/playwright/playwright.robot +24 -0
  193. cumulusci/robotframework/tests/salesforce/playwright/ui.robot +32 -0
  194. cumulusci/robotframework/tests/salesforce/populate.robot +89 -0
  195. cumulusci/robotframework/tests/salesforce/test_testlistener.py +37 -0
  196. cumulusci/robotframework/tests/salesforce/ui.robot +361 -0
  197. cumulusci/robotframework/tests/test_cumulusci_library.py +304 -0
  198. cumulusci/robotframework/tests/test_locator_manager.py +158 -0
  199. cumulusci/robotframework/tests/test_pageobjects.py +291 -0
  200. cumulusci/robotframework/tests/test_performance.py +38 -0
  201. cumulusci/robotframework/tests/test_salesforce.py +79 -0
  202. cumulusci/robotframework/tests/test_salesforce_locators.py +73 -0
  203. cumulusci/robotframework/tests/test_template_util.py +53 -0
  204. cumulusci/robotframework/tests/test_utils.py +106 -0
  205. cumulusci/robotframework/utils.py +283 -0
  206. cumulusci/salesforce_api/__init__.py +0 -0
  207. cumulusci/salesforce_api/exceptions.py +23 -0
  208. cumulusci/salesforce_api/filterable_objects.py +96 -0
  209. cumulusci/salesforce_api/mc_soap_envelopes.py +89 -0
  210. cumulusci/salesforce_api/metadata.py +721 -0
  211. cumulusci/salesforce_api/org_schema.py +571 -0
  212. cumulusci/salesforce_api/org_schema_models.py +226 -0
  213. cumulusci/salesforce_api/package_install.py +265 -0
  214. cumulusci/salesforce_api/package_zip.py +301 -0
  215. cumulusci/salesforce_api/rest_deploy.py +148 -0
  216. cumulusci/salesforce_api/retrieve_profile_api.py +301 -0
  217. cumulusci/salesforce_api/soap_envelopes.py +177 -0
  218. cumulusci/salesforce_api/tests/__init__.py +0 -0
  219. cumulusci/salesforce_api/tests/metadata_test_strings.py +24 -0
  220. cumulusci/salesforce_api/tests/test_metadata.py +1015 -0
  221. cumulusci/salesforce_api/tests/test_package_install.py +219 -0
  222. cumulusci/salesforce_api/tests/test_package_zip.py +380 -0
  223. cumulusci/salesforce_api/tests/test_rest_deploy.py +264 -0
  224. cumulusci/salesforce_api/tests/test_retrieve_profile_api.py +337 -0
  225. cumulusci/salesforce_api/tests/test_utils.py +124 -0
  226. cumulusci/salesforce_api/utils.py +51 -0
  227. cumulusci/schema/cumulusci.jsonschema.json +782 -0
  228. cumulusci/tasks/__init__.py +0 -0
  229. cumulusci/tasks/apex/__init__.py +0 -0
  230. cumulusci/tasks/apex/anon.py +157 -0
  231. cumulusci/tasks/apex/batch.py +180 -0
  232. cumulusci/tasks/apex/testrunner.py +835 -0
  233. cumulusci/tasks/apex/tests/cassettes/ManualEditTestApexIntegrationTests.test_run_tests__integration_test.yaml +703 -0
  234. cumulusci/tasks/apex/tests/test_apex_tasks.py +1558 -0
  235. cumulusci/tasks/base_source_control_task.py +17 -0
  236. cumulusci/tasks/bulkdata/__init__.py +15 -0
  237. cumulusci/tasks/bulkdata/base_generate_data_task.py +96 -0
  238. cumulusci/tasks/bulkdata/dates.py +97 -0
  239. cumulusci/tasks/bulkdata/delete.py +156 -0
  240. cumulusci/tasks/bulkdata/extract.py +441 -0
  241. cumulusci/tasks/bulkdata/extract_dataset_utils/calculate_dependencies.py +117 -0
  242. cumulusci/tasks/bulkdata/extract_dataset_utils/extract_yml.py +123 -0
  243. cumulusci/tasks/bulkdata/extract_dataset_utils/hardcoded_default_declarations.py +49 -0
  244. cumulusci/tasks/bulkdata/extract_dataset_utils/synthesize_extract_declarations.py +283 -0
  245. cumulusci/tasks/bulkdata/extract_dataset_utils/tests/test_extract_yml.py +142 -0
  246. cumulusci/tasks/bulkdata/extract_dataset_utils/tests/test_synthesize_extract_declarations.py +575 -0
  247. cumulusci/tasks/bulkdata/factory_utils.py +134 -0
  248. cumulusci/tasks/bulkdata/generate.py +4 -0
  249. cumulusci/tasks/bulkdata/generate_and_load_data.py +232 -0
  250. cumulusci/tasks/bulkdata/generate_and_load_data_from_yaml.py +19 -0
  251. cumulusci/tasks/bulkdata/generate_from_yaml.py +183 -0
  252. cumulusci/tasks/bulkdata/generate_mapping.py +434 -0
  253. cumulusci/tasks/bulkdata/generate_mapping_utils/dependency_map.py +169 -0
  254. cumulusci/tasks/bulkdata/generate_mapping_utils/extract_mapping_file_generator.py +45 -0
  255. cumulusci/tasks/bulkdata/generate_mapping_utils/generate_mapping_from_declarations.py +121 -0
  256. cumulusci/tasks/bulkdata/generate_mapping_utils/load_mapping_file_generator.py +127 -0
  257. cumulusci/tasks/bulkdata/generate_mapping_utils/mapping_generator_post_processes.py +53 -0
  258. cumulusci/tasks/bulkdata/generate_mapping_utils/mapping_transforms.py +139 -0
  259. cumulusci/tasks/bulkdata/generate_mapping_utils/tests/test_generate_extract_mapping_from_declarations.py +135 -0
  260. cumulusci/tasks/bulkdata/generate_mapping_utils/tests/test_generate_load_mapping_from_declarations.py +330 -0
  261. cumulusci/tasks/bulkdata/generate_mapping_utils/tests/test_mapping_generator_post_processes.py +60 -0
  262. cumulusci/tasks/bulkdata/generate_mapping_utils/tests/test_mapping_transforms.py +188 -0
  263. cumulusci/tasks/bulkdata/load.py +1196 -0
  264. cumulusci/tasks/bulkdata/mapping_parser.py +811 -0
  265. cumulusci/tasks/bulkdata/query_transformers.py +264 -0
  266. cumulusci/tasks/bulkdata/select_utils.py +792 -0
  267. cumulusci/tasks/bulkdata/snowfakery.py +753 -0
  268. cumulusci/tasks/bulkdata/snowfakery_utils/queue_manager.py +478 -0
  269. cumulusci/tasks/bulkdata/snowfakery_utils/snowfakery_run_until.py +141 -0
  270. cumulusci/tasks/bulkdata/snowfakery_utils/snowfakery_working_directory.py +53 -0
  271. cumulusci/tasks/bulkdata/snowfakery_utils/subtask_configurator.py +64 -0
  272. cumulusci/tasks/bulkdata/step.py +1242 -0
  273. cumulusci/tasks/bulkdata/tests/__init__.py +0 -0
  274. cumulusci/tasks/bulkdata/tests/cassettes/TestSelect.test_select_random_strategy.yaml +147 -0
  275. cumulusci/tasks/bulkdata/tests/cassettes/TestSelect.test_select_similarity_annoy_strategy.yaml +123 -0
  276. cumulusci/tasks/bulkdata/tests/cassettes/TestSelect.test_select_similarity_select_and_insert_strategy.yaml +313 -0
  277. cumulusci/tasks/bulkdata/tests/cassettes/TestSelect.test_select_similarity_select_and_insert_strategy_bulk.yaml +550 -0
  278. cumulusci/tasks/bulkdata/tests/cassettes/TestSelect.test_select_similarity_strategy.yaml +175 -0
  279. cumulusci/tasks/bulkdata/tests/cassettes/TestSelect.test_select_standard_strategy.yaml +147 -0
  280. cumulusci/tasks/bulkdata/tests/cassettes/TestSnowfakery.test_run_until_records_in_org__multiple_needed.yaml +69 -0
  281. cumulusci/tasks/bulkdata/tests/cassettes/TestSnowfakery.test_run_until_records_in_org__none_needed.yaml +22 -0
  282. cumulusci/tasks/bulkdata/tests/cassettes/TestSnowfakery.test_run_until_records_in_org__one_needed.yaml +24 -0
  283. cumulusci/tasks/bulkdata/tests/cassettes/TestSnowfakery.test_snowfakery_query_salesforce.yaml +25 -0
  284. cumulusci/tasks/bulkdata/tests/cassettes/TestUpdatesIntegrationTests.test_updates_task.yaml +80 -0
  285. cumulusci/tasks/bulkdata/tests/cassettes/TestUpsert.test_simple_upsert__rest.yaml +270 -0
  286. cumulusci/tasks/bulkdata/tests/cassettes/TestUpsert.test_upsert__rest.yaml +267 -0
  287. cumulusci/tasks/bulkdata/tests/cassettes/TestUpsert.test_upsert_complex_external_id_field__rest.yaml +369 -0
  288. cumulusci/tasks/bulkdata/tests/cassettes/TestUpsert.test_upsert_complex_external_id_field_rest__duplicate_error.yaml +204 -0
  289. cumulusci/tasks/bulkdata/tests/cassettes/TestUpsert.test_upsert_complex_fields__bulk.yaml +675 -0
  290. cumulusci/tasks/bulkdata/tests/dummy_data_factory.py +36 -0
  291. cumulusci/tasks/bulkdata/tests/integration_test_utils.py +49 -0
  292. cumulusci/tasks/bulkdata/tests/mapping-oid.yml +87 -0
  293. cumulusci/tasks/bulkdata/tests/mapping_after.yml +38 -0
  294. cumulusci/tasks/bulkdata/tests/mapping_poly.yml +34 -0
  295. cumulusci/tasks/bulkdata/tests/mapping_poly_incomplete.yml +20 -0
  296. cumulusci/tasks/bulkdata/tests/mapping_poly_wrong.yml +21 -0
  297. cumulusci/tasks/bulkdata/tests/mapping_select.yml +20 -0
  298. cumulusci/tasks/bulkdata/tests/mapping_select_invalid_strategy.yml +20 -0
  299. cumulusci/tasks/bulkdata/tests/mapping_select_invalid_threshold__invalid_number.yml +21 -0
  300. cumulusci/tasks/bulkdata/tests/mapping_select_invalid_threshold__invalid_strategy.yml +21 -0
  301. cumulusci/tasks/bulkdata/tests/mapping_select_invalid_threshold__non_float.yml +21 -0
  302. cumulusci/tasks/bulkdata/tests/mapping_select_missing_priority_fields.yml +22 -0
  303. cumulusci/tasks/bulkdata/tests/mapping_select_no_priority_fields.yml +18 -0
  304. cumulusci/tasks/bulkdata/tests/mapping_simple.yml +27 -0
  305. cumulusci/tasks/bulkdata/tests/mapping_v1.yml +28 -0
  306. cumulusci/tasks/bulkdata/tests/mapping_v2.yml +21 -0
  307. cumulusci/tasks/bulkdata/tests/mapping_v3.yml +32 -0
  308. cumulusci/tasks/bulkdata/tests/mapping_vanilla_sf.yml +69 -0
  309. cumulusci/tasks/bulkdata/tests/mock_data_factory_without_mapping.py +12 -0
  310. cumulusci/tasks/bulkdata/tests/person_accounts.yml +23 -0
  311. cumulusci/tasks/bulkdata/tests/person_accounts_minimal.yml +15 -0
  312. cumulusci/tasks/bulkdata/tests/recordtypes.yml +8 -0
  313. cumulusci/tasks/bulkdata/tests/recordtypes_2.yml +6 -0
  314. cumulusci/tasks/bulkdata/tests/recordtypes_with_ispersontype.yml +8 -0
  315. cumulusci/tasks/bulkdata/tests/snowfakery/child/child2.yml +3 -0
  316. cumulusci/tasks/bulkdata/tests/snowfakery/child.yml +4 -0
  317. cumulusci/tasks/bulkdata/tests/snowfakery/gen_npsp_standard_objects.recipe.yml +89 -0
  318. cumulusci/tasks/bulkdata/tests/snowfakery/include_parent.yml +3 -0
  319. cumulusci/tasks/bulkdata/tests/snowfakery/npsp_standard_objects_macros.yml +34 -0
  320. cumulusci/tasks/bulkdata/tests/snowfakery/options.recipe.yml +6 -0
  321. cumulusci/tasks/bulkdata/tests/snowfakery/query_snowfakery.recipe.yml +16 -0
  322. cumulusci/tasks/bulkdata/tests/snowfakery/sf_standard_object_macros.yml +83 -0
  323. cumulusci/tasks/bulkdata/tests/snowfakery/simple_snowfakery.load.yml +2 -0
  324. cumulusci/tasks/bulkdata/tests/snowfakery/simple_snowfakery.recipe.yml +13 -0
  325. cumulusci/tasks/bulkdata/tests/snowfakery/simple_snowfakery_2.load.yml +5 -0
  326. cumulusci/tasks/bulkdata/tests/snowfakery/simple_snowfakery_channels.load.yml +13 -0
  327. cumulusci/tasks/bulkdata/tests/snowfakery/simple_snowfakery_channels.recipe.yml +12 -0
  328. cumulusci/tasks/bulkdata/tests/snowfakery/simple_snowfakery_channels_2.load.yml +13 -0
  329. cumulusci/tasks/bulkdata/tests/snowfakery/unique_values.recipe.yml +4 -0
  330. cumulusci/tasks/bulkdata/tests/snowfakery/upsert.recipe.yml +23 -0
  331. cumulusci/tasks/bulkdata/tests/snowfakery/upsert_2.recipe.yml +29 -0
  332. cumulusci/tasks/bulkdata/tests/snowfakery/upsert_before.yml +10 -0
  333. cumulusci/tasks/bulkdata/tests/test_base_generate_data_tasks.py +61 -0
  334. cumulusci/tasks/bulkdata/tests/test_dates.py +99 -0
  335. cumulusci/tasks/bulkdata/tests/test_delete.py +404 -0
  336. cumulusci/tasks/bulkdata/tests/test_extract.py +1311 -0
  337. cumulusci/tasks/bulkdata/tests/test_factory_utils.py +55 -0
  338. cumulusci/tasks/bulkdata/tests/test_generate_and_load.py +252 -0
  339. cumulusci/tasks/bulkdata/tests/test_generate_from_snowfakery_task.py +343 -0
  340. cumulusci/tasks/bulkdata/tests/test_generatemapping.py +1039 -0
  341. cumulusci/tasks/bulkdata/tests/test_load.py +3175 -0
  342. cumulusci/tasks/bulkdata/tests/test_mapping_parser.py +1658 -0
  343. cumulusci/tasks/bulkdata/tests/test_query_db__joins_self_lookups.yml +12 -0
  344. cumulusci/tasks/bulkdata/tests/test_query_db_joins_lookups.yml +26 -0
  345. cumulusci/tasks/bulkdata/tests/test_query_db_joins_lookups_select.yml +48 -0
  346. cumulusci/tasks/bulkdata/tests/test_select.py +171 -0
  347. cumulusci/tasks/bulkdata/tests/test_select_utils.py +1057 -0
  348. cumulusci/tasks/bulkdata/tests/test_snowfakery.py +1153 -0
  349. cumulusci/tasks/bulkdata/tests/test_step.py +3957 -0
  350. cumulusci/tasks/bulkdata/tests/test_updates.py +513 -0
  351. cumulusci/tasks/bulkdata/tests/test_upsert.py +1015 -0
  352. cumulusci/tasks/bulkdata/tests/test_utils.py +158 -0
  353. cumulusci/tasks/bulkdata/tests/testdata.db +0 -0
  354. cumulusci/tasks/bulkdata/tests/update_describe.py +50 -0
  355. cumulusci/tasks/bulkdata/tests/update_person_accounts.yml +23 -0
  356. cumulusci/tasks/bulkdata/tests/utils.py +114 -0
  357. cumulusci/tasks/bulkdata/update_data.py +260 -0
  358. cumulusci/tasks/bulkdata/upsert_utils.py +130 -0
  359. cumulusci/tasks/bulkdata/utils.py +249 -0
  360. cumulusci/tasks/command.py +178 -0
  361. cumulusci/tasks/connectedapp.py +186 -0
  362. cumulusci/tasks/create_package_version.py +778 -0
  363. cumulusci/tasks/datadictionary.py +745 -0
  364. cumulusci/tasks/dx_convert_from.py +26 -0
  365. cumulusci/tasks/github/__init__.py +17 -0
  366. cumulusci/tasks/github/base.py +16 -0
  367. cumulusci/tasks/github/commit_status.py +13 -0
  368. cumulusci/tasks/github/merge.py +11 -0
  369. cumulusci/tasks/github/publish.py +11 -0
  370. cumulusci/tasks/github/pull_request.py +11 -0
  371. cumulusci/tasks/github/release.py +11 -0
  372. cumulusci/tasks/github/release_report.py +11 -0
  373. cumulusci/tasks/github/tag.py +11 -0
  374. cumulusci/tasks/github/tests/__init__.py +0 -0
  375. cumulusci/tasks/github/tests/test_util.py +202 -0
  376. cumulusci/tasks/github/tests/test_vcs_migration.py +44 -0
  377. cumulusci/tasks/github/tests/util_github_api.py +666 -0
  378. cumulusci/tasks/github/util.py +252 -0
  379. cumulusci/tasks/marketing_cloud/__init__.py +0 -0
  380. cumulusci/tasks/marketing_cloud/api.py +188 -0
  381. cumulusci/tasks/marketing_cloud/base.py +38 -0
  382. cumulusci/tasks/marketing_cloud/deploy.py +345 -0
  383. cumulusci/tasks/marketing_cloud/get_user_info.py +40 -0
  384. cumulusci/tasks/marketing_cloud/mc_constants.py +1 -0
  385. cumulusci/tasks/marketing_cloud/tests/__init__.py +0 -0
  386. cumulusci/tasks/marketing_cloud/tests/conftest.py +46 -0
  387. cumulusci/tasks/marketing_cloud/tests/expected-payload.json +110 -0
  388. cumulusci/tasks/marketing_cloud/tests/test_api.py +97 -0
  389. cumulusci/tasks/marketing_cloud/tests/test_api_soap_envelopes.py +145 -0
  390. cumulusci/tasks/marketing_cloud/tests/test_base.py +14 -0
  391. cumulusci/tasks/marketing_cloud/tests/test_deploy.py +400 -0
  392. cumulusci/tasks/marketing_cloud/tests/test_get_user_info.py +141 -0
  393. cumulusci/tasks/marketing_cloud/tests/validation-response.json +39 -0
  394. cumulusci/tasks/metadata/__init__.py +0 -0
  395. cumulusci/tasks/metadata/ee_src.py +94 -0
  396. cumulusci/tasks/metadata/managed_src.py +100 -0
  397. cumulusci/tasks/metadata/metadata_map.yml +868 -0
  398. cumulusci/tasks/metadata/modify.py +99 -0
  399. cumulusci/tasks/metadata/package.py +684 -0
  400. cumulusci/tasks/metadata/tests/__init__.py +0 -0
  401. cumulusci/tasks/metadata/tests/package_metadata/namespaced_report_folder/.hidden/.keep +0 -0
  402. cumulusci/tasks/metadata/tests/package_metadata/namespaced_report_folder/destructiveChanges.xml +9 -0
  403. cumulusci/tasks/metadata/tests/package_metadata/namespaced_report_folder/package.xml +9 -0
  404. cumulusci/tasks/metadata/tests/package_metadata/namespaced_report_folder/package_install_uninstall.xml +11 -0
  405. cumulusci/tasks/metadata/tests/package_metadata/namespaced_report_folder/reports/namespace__TestFolder/TestReport.report +3 -0
  406. cumulusci/tasks/metadata/tests/sample_package.xml +9 -0
  407. cumulusci/tasks/metadata/tests/test_ee_src.py +112 -0
  408. cumulusci/tasks/metadata/tests/test_managed_src.py +111 -0
  409. cumulusci/tasks/metadata/tests/test_modify.py +123 -0
  410. cumulusci/tasks/metadata/tests/test_package.py +476 -0
  411. cumulusci/tasks/metadata_etl/__init__.py +29 -0
  412. cumulusci/tasks/metadata_etl/base.py +436 -0
  413. cumulusci/tasks/metadata_etl/duplicate_rules.py +24 -0
  414. cumulusci/tasks/metadata_etl/field_sets.py +70 -0
  415. cumulusci/tasks/metadata_etl/help_text.py +92 -0
  416. cumulusci/tasks/metadata_etl/layouts.py +550 -0
  417. cumulusci/tasks/metadata_etl/objects.py +68 -0
  418. cumulusci/tasks/metadata_etl/permissions.py +167 -0
  419. cumulusci/tasks/metadata_etl/picklists.py +221 -0
  420. cumulusci/tasks/metadata_etl/remote_site_settings.py +99 -0
  421. cumulusci/tasks/metadata_etl/sharing.py +138 -0
  422. cumulusci/tasks/metadata_etl/tests/test_base.py +512 -0
  423. cumulusci/tasks/metadata_etl/tests/test_duplicate_rules.py +22 -0
  424. cumulusci/tasks/metadata_etl/tests/test_field_sets.py +156 -0
  425. cumulusci/tasks/metadata_etl/tests/test_help_text.py +387 -0
  426. cumulusci/tasks/metadata_etl/tests/test_ip_ranges.py +85 -0
  427. cumulusci/tasks/metadata_etl/tests/test_layouts.py +858 -0
  428. cumulusci/tasks/metadata_etl/tests/test_objects.py +236 -0
  429. cumulusci/tasks/metadata_etl/tests/test_permissions.py +223 -0
  430. cumulusci/tasks/metadata_etl/tests/test_picklists.py +547 -0
  431. cumulusci/tasks/metadata_etl/tests/test_remote_site_settings.py +46 -0
  432. cumulusci/tasks/metadata_etl/tests/test_sharing.py +333 -0
  433. cumulusci/tasks/metadata_etl/tests/test_value_sets.py +298 -0
  434. cumulusci/tasks/metadata_etl/value_sets.py +106 -0
  435. cumulusci/tasks/metadeploy.py +393 -0
  436. cumulusci/tasks/metaxml.py +88 -0
  437. cumulusci/tasks/preflight/__init__.py +0 -0
  438. cumulusci/tasks/preflight/dataset_load.py +49 -0
  439. cumulusci/tasks/preflight/licenses.py +86 -0
  440. cumulusci/tasks/preflight/packages.py +14 -0
  441. cumulusci/tasks/preflight/permsets.py +23 -0
  442. cumulusci/tasks/preflight/recordtypes.py +16 -0
  443. cumulusci/tasks/preflight/retrieve_tasks.py +30 -0
  444. cumulusci/tasks/preflight/settings.py +77 -0
  445. cumulusci/tasks/preflight/sobjects.py +202 -0
  446. cumulusci/tasks/preflight/tests/test_dataset_load.py +85 -0
  447. cumulusci/tasks/preflight/tests/test_licenses.py +174 -0
  448. cumulusci/tasks/preflight/tests/test_packages.py +14 -0
  449. cumulusci/tasks/preflight/tests/test_permset_preflights.py +51 -0
  450. cumulusci/tasks/preflight/tests/test_recordtypes.py +30 -0
  451. cumulusci/tasks/preflight/tests/test_retrieve_tasks.py +62 -0
  452. cumulusci/tasks/preflight/tests/test_settings.py +130 -0
  453. cumulusci/tasks/preflight/tests/test_sobjects.py +231 -0
  454. cumulusci/tasks/push/README.md +59 -0
  455. cumulusci/tasks/push/__init__.py +0 -0
  456. cumulusci/tasks/push/push_api.py +659 -0
  457. cumulusci/tasks/push/pushfails.py +136 -0
  458. cumulusci/tasks/push/tasks.py +476 -0
  459. cumulusci/tasks/push/tests/conftest.py +263 -0
  460. cumulusci/tasks/push/tests/test_push_api.py +951 -0
  461. cumulusci/tasks/push/tests/test_push_tasks.py +659 -0
  462. cumulusci/tasks/release_notes/README.md +63 -0
  463. cumulusci/tasks/release_notes/__init__.py +0 -0
  464. cumulusci/tasks/release_notes/exceptions.py +5 -0
  465. cumulusci/tasks/release_notes/generator.py +137 -0
  466. cumulusci/tasks/release_notes/parser.py +232 -0
  467. cumulusci/tasks/release_notes/provider.py +44 -0
  468. cumulusci/tasks/release_notes/task.py +300 -0
  469. cumulusci/tasks/release_notes/tests/__init__.py +0 -0
  470. cumulusci/tasks/release_notes/tests/change_notes/full/example1.md +17 -0
  471. cumulusci/tasks/release_notes/tests/change_notes/multi/1.txt +1 -0
  472. cumulusci/tasks/release_notes/tests/change_notes/multi/2.txt +1 -0
  473. cumulusci/tasks/release_notes/tests/change_notes/multi/3.txt +1 -0
  474. cumulusci/tasks/release_notes/tests/change_notes/single/1.txt +1 -0
  475. cumulusci/tasks/release_notes/tests/test_generator.py +582 -0
  476. cumulusci/tasks/release_notes/tests/test_parser.py +867 -0
  477. cumulusci/tasks/release_notes/tests/test_provider.py +512 -0
  478. cumulusci/tasks/release_notes/tests/test_task.py +461 -0
  479. cumulusci/tasks/release_notes/tests/utils.py +153 -0
  480. cumulusci/tasks/robotframework/__init__.py +3 -0
  481. cumulusci/tasks/robotframework/debugger/DebugListener.py +100 -0
  482. cumulusci/tasks/robotframework/debugger/__init__.py +10 -0
  483. cumulusci/tasks/robotframework/debugger/model.py +87 -0
  484. cumulusci/tasks/robotframework/debugger/ui.py +259 -0
  485. cumulusci/tasks/robotframework/libdoc.py +269 -0
  486. cumulusci/tasks/robotframework/robotframework.py +392 -0
  487. cumulusci/tasks/robotframework/stylesheet.css +130 -0
  488. cumulusci/tasks/robotframework/template.html +109 -0
  489. cumulusci/tasks/robotframework/tests/TestLibrary.py +18 -0
  490. cumulusci/tasks/robotframework/tests/TestPageObjects.py +31 -0
  491. cumulusci/tasks/robotframework/tests/TestResource.robot +8 -0
  492. cumulusci/tasks/robotframework/tests/failing_tests.robot +16 -0
  493. cumulusci/tasks/robotframework/tests/performance.robot +23 -0
  494. cumulusci/tasks/robotframework/tests/test_browser_proxies.py +137 -0
  495. cumulusci/tasks/robotframework/tests/test_debugger.py +360 -0
  496. cumulusci/tasks/robotframework/tests/test_robot_parallel.py +141 -0
  497. cumulusci/tasks/robotframework/tests/test_robotframework.py +860 -0
  498. cumulusci/tasks/salesforce/BaseRetrieveMetadata.py +58 -0
  499. cumulusci/tasks/salesforce/BaseSalesforceApiTask.py +45 -0
  500. cumulusci/tasks/salesforce/BaseSalesforceMetadataApiTask.py +18 -0
  501. cumulusci/tasks/salesforce/BaseSalesforceTask.py +4 -0
  502. cumulusci/tasks/salesforce/BaseUninstallMetadata.py +41 -0
  503. cumulusci/tasks/salesforce/CreateCommunity.py +124 -0
  504. cumulusci/tasks/salesforce/CreatePackage.py +29 -0
  505. cumulusci/tasks/salesforce/Deploy.py +240 -0
  506. cumulusci/tasks/salesforce/DeployBundles.py +88 -0
  507. cumulusci/tasks/salesforce/DescribeMetadataTypes.py +26 -0
  508. cumulusci/tasks/salesforce/EnsureRecordTypes.py +202 -0
  509. cumulusci/tasks/salesforce/GetInstalledPackages.py +8 -0
  510. cumulusci/tasks/salesforce/ListCommunities.py +40 -0
  511. cumulusci/tasks/salesforce/ListCommunityTemplates.py +19 -0
  512. cumulusci/tasks/salesforce/PublishCommunity.py +62 -0
  513. cumulusci/tasks/salesforce/RetrievePackaged.py +41 -0
  514. cumulusci/tasks/salesforce/RetrieveReportsAndDashboards.py +82 -0
  515. cumulusci/tasks/salesforce/RetrieveUnpackaged.py +36 -0
  516. cumulusci/tasks/salesforce/SOQLQuery.py +39 -0
  517. cumulusci/tasks/salesforce/UninstallLocal.py +15 -0
  518. cumulusci/tasks/salesforce/UninstallLocalBundles.py +28 -0
  519. cumulusci/tasks/salesforce/UninstallLocalNamespacedBundles.py +58 -0
  520. cumulusci/tasks/salesforce/UninstallPackage.py +32 -0
  521. cumulusci/tasks/salesforce/UninstallPackaged.py +56 -0
  522. cumulusci/tasks/salesforce/UpdateAdminProfile.py +8 -0
  523. cumulusci/tasks/salesforce/__init__.py +79 -0
  524. cumulusci/tasks/salesforce/activate_flow.py +74 -0
  525. cumulusci/tasks/salesforce/check_components.py +324 -0
  526. cumulusci/tasks/salesforce/composite.py +142 -0
  527. cumulusci/tasks/salesforce/create_permission_sets.py +35 -0
  528. cumulusci/tasks/salesforce/custom_settings.py +134 -0
  529. cumulusci/tasks/salesforce/custom_settings_wait.py +132 -0
  530. cumulusci/tasks/salesforce/enable_prediction.py +107 -0
  531. cumulusci/tasks/salesforce/insert_record.py +40 -0
  532. cumulusci/tasks/salesforce/install_package_version.py +242 -0
  533. cumulusci/tasks/salesforce/license_preflights.py +8 -0
  534. cumulusci/tasks/salesforce/network_member_group.py +178 -0
  535. cumulusci/tasks/salesforce/nonsourcetracking.py +228 -0
  536. cumulusci/tasks/salesforce/org_settings.py +193 -0
  537. cumulusci/tasks/salesforce/package_upload.py +328 -0
  538. cumulusci/tasks/salesforce/profiles.py +74 -0
  539. cumulusci/tasks/salesforce/promote_package_version.py +376 -0
  540. cumulusci/tasks/salesforce/retrieve_profile.py +195 -0
  541. cumulusci/tasks/salesforce/salesforce_files.py +244 -0
  542. cumulusci/tasks/salesforce/sourcetracking.py +507 -0
  543. cumulusci/tasks/salesforce/tests/__init__.py +3 -0
  544. cumulusci/tasks/salesforce/tests/test_CreateCommunity.py +278 -0
  545. cumulusci/tasks/salesforce/tests/test_CreatePackage.py +22 -0
  546. cumulusci/tasks/salesforce/tests/test_Deploy.py +470 -0
  547. cumulusci/tasks/salesforce/tests/test_DeployBundles.py +76 -0
  548. cumulusci/tasks/salesforce/tests/test_EnsureRecordTypes.py +345 -0
  549. cumulusci/tasks/salesforce/tests/test_ListCommunities.py +84 -0
  550. cumulusci/tasks/salesforce/tests/test_ListCommunityTemplates.py +49 -0
  551. cumulusci/tasks/salesforce/tests/test_PackageUpload.py +547 -0
  552. cumulusci/tasks/salesforce/tests/test_ProfileGrantAllAccess.py +699 -0
  553. cumulusci/tasks/salesforce/tests/test_PublishCommunity.py +181 -0
  554. cumulusci/tasks/salesforce/tests/test_RetrievePackaged.py +24 -0
  555. cumulusci/tasks/salesforce/tests/test_RetrieveReportsAndDashboards.py +56 -0
  556. cumulusci/tasks/salesforce/tests/test_RetrieveUnpackaged.py +21 -0
  557. cumulusci/tasks/salesforce/tests/test_SOQLQuery.py +30 -0
  558. cumulusci/tasks/salesforce/tests/test_UninstallLocal.py +15 -0
  559. cumulusci/tasks/salesforce/tests/test_UninstallLocalBundles.py +19 -0
  560. cumulusci/tasks/salesforce/tests/test_UninstallLocalNamespacedBundles.py +22 -0
  561. cumulusci/tasks/salesforce/tests/test_UninstallPackage.py +19 -0
  562. cumulusci/tasks/salesforce/tests/test_UninstallPackaged.py +66 -0
  563. cumulusci/tasks/salesforce/tests/test_UninstallPackagedIncremental.py +127 -0
  564. cumulusci/tasks/salesforce/tests/test_activate_flow.py +132 -0
  565. cumulusci/tasks/salesforce/tests/test_base_tasks.py +110 -0
  566. cumulusci/tasks/salesforce/tests/test_check_components.py +445 -0
  567. cumulusci/tasks/salesforce/tests/test_composite.py +250 -0
  568. cumulusci/tasks/salesforce/tests/test_create_permission_sets.py +41 -0
  569. cumulusci/tasks/salesforce/tests/test_custom_settings.py +227 -0
  570. cumulusci/tasks/salesforce/tests/test_custom_settings_wait.py +174 -0
  571. cumulusci/tasks/salesforce/tests/test_describemetadatatypes.py +18 -0
  572. cumulusci/tasks/salesforce/tests/test_enable_prediction.py +240 -0
  573. cumulusci/tasks/salesforce/tests/test_insert_record.py +110 -0
  574. cumulusci/tasks/salesforce/tests/test_install_package_version.py +464 -0
  575. cumulusci/tasks/salesforce/tests/test_network_member_group.py +444 -0
  576. cumulusci/tasks/salesforce/tests/test_nonsourcetracking.py +235 -0
  577. cumulusci/tasks/salesforce/tests/test_org_settings.py +407 -0
  578. cumulusci/tasks/salesforce/tests/test_profiles.py +202 -0
  579. cumulusci/tasks/salesforce/tests/test_retrieve_profile.py +287 -0
  580. cumulusci/tasks/salesforce/tests/test_salesforce_files.py +228 -0
  581. cumulusci/tasks/salesforce/tests/test_sourcetracking.py +350 -0
  582. cumulusci/tasks/salesforce/tests/test_trigger_handlers.py +300 -0
  583. cumulusci/tasks/salesforce/tests/test_update_dependencies.py +509 -0
  584. cumulusci/tasks/salesforce/tests/util.py +79 -0
  585. cumulusci/tasks/salesforce/trigger_handlers.py +119 -0
  586. cumulusci/tasks/salesforce/uninstall_packaged_incremental.py +136 -0
  587. cumulusci/tasks/salesforce/update_dependencies.py +290 -0
  588. cumulusci/tasks/salesforce/update_profile.py +339 -0
  589. cumulusci/tasks/salesforce/users/permsets.py +227 -0
  590. cumulusci/tasks/salesforce/users/photos.py +162 -0
  591. cumulusci/tasks/salesforce/users/tests/photo.mock.txt +1 -0
  592. cumulusci/tasks/salesforce/users/tests/test_permsets.py +950 -0
  593. cumulusci/tasks/salesforce/users/tests/test_photos.py +373 -0
  594. cumulusci/tasks/sample_data/capture_sample_data.py +77 -0
  595. cumulusci/tasks/sample_data/load_sample_data.py +85 -0
  596. cumulusci/tasks/sample_data/test_capture_sample_data.py +117 -0
  597. cumulusci/tasks/sample_data/test_load_sample_data.py +121 -0
  598. cumulusci/tasks/sfdx.py +83 -0
  599. cumulusci/tasks/tests/__init__.py +1 -0
  600. cumulusci/tasks/tests/conftest.py +30 -0
  601. cumulusci/tasks/tests/test_command.py +129 -0
  602. cumulusci/tasks/tests/test_connectedapp.py +236 -0
  603. cumulusci/tasks/tests/test_create_package_version.py +847 -0
  604. cumulusci/tasks/tests/test_datadictionary.py +1575 -0
  605. cumulusci/tasks/tests/test_dx_convert_from.py +60 -0
  606. cumulusci/tasks/tests/test_metadeploy.py +624 -0
  607. cumulusci/tasks/tests/test_metaxml.py +99 -0
  608. cumulusci/tasks/tests/test_promote_package_version.py +488 -0
  609. cumulusci/tasks/tests/test_pushfails.py +96 -0
  610. cumulusci/tasks/tests/test_salesforce.py +72 -0
  611. cumulusci/tasks/tests/test_sfdx.py +105 -0
  612. cumulusci/tasks/tests/test_util.py +207 -0
  613. cumulusci/tasks/util.py +261 -0
  614. cumulusci/tasks/vcs/__init__.py +19 -0
  615. cumulusci/tasks/vcs/commit_status.py +58 -0
  616. cumulusci/tasks/vcs/create_commit_status.py +37 -0
  617. cumulusci/tasks/vcs/download_extract.py +199 -0
  618. cumulusci/tasks/vcs/merge.py +298 -0
  619. cumulusci/tasks/vcs/publish.py +207 -0
  620. cumulusci/tasks/vcs/pull_request.py +9 -0
  621. cumulusci/tasks/vcs/release.py +134 -0
  622. cumulusci/tasks/vcs/release_report.py +105 -0
  623. cumulusci/tasks/vcs/tag.py +31 -0
  624. cumulusci/tasks/vcs/tests/github/test_commit_status.py +196 -0
  625. cumulusci/tasks/vcs/tests/github/test_download_extract.py +896 -0
  626. cumulusci/tasks/vcs/tests/github/test_merge.py +1118 -0
  627. cumulusci/tasks/vcs/tests/github/test_publish.py +823 -0
  628. cumulusci/tasks/vcs/tests/github/test_pull_request.py +29 -0
  629. cumulusci/tasks/vcs/tests/github/test_release.py +390 -0
  630. cumulusci/tasks/vcs/tests/github/test_release_report.py +109 -0
  631. cumulusci/tasks/vcs/tests/github/test_tag.py +90 -0
  632. cumulusci/tasks/vlocity/exceptions.py +2 -0
  633. cumulusci/tasks/vlocity/tests/test_vlocity.py +283 -0
  634. cumulusci/tasks/vlocity/vlocity.py +342 -0
  635. cumulusci/tests/__init__.py +1 -0
  636. cumulusci/tests/cassettes/GET_sobjects_Account_PersonAccount_describe.yaml +18 -0
  637. cumulusci/tests/cassettes/TestIntegrationInfrastructure.test_integration_tests.yaml +19 -0
  638. cumulusci/tests/pytest_plugins/pytest_sf_orgconnect.py +307 -0
  639. cumulusci/tests/pytest_plugins/pytest_sf_vcr.py +275 -0
  640. cumulusci/tests/pytest_plugins/pytest_sf_vcr_serializer.py +160 -0
  641. cumulusci/tests/pytest_plugins/pytest_typeguard.py +5 -0
  642. cumulusci/tests/pytest_plugins/test_vcr_string_compressor.py +49 -0
  643. cumulusci/tests/pytest_plugins/vcr_string_compressor.py +97 -0
  644. cumulusci/tests/shared_cassettes/GET_sobjects_Account_describe.yaml +18 -0
  645. cumulusci/tests/shared_cassettes/GET_sobjects_Case_describe.yaml +18 -0
  646. cumulusci/tests/shared_cassettes/GET_sobjects_Contact_describe.yaml +4838 -0
  647. cumulusci/tests/shared_cassettes/GET_sobjects_Custom__c_describe.yaml +242 -0
  648. cumulusci/tests/shared_cassettes/GET_sobjects_Event_describe.yaml +19 -0
  649. cumulusci/tests/shared_cassettes/GET_sobjects_Global_describe.yaml +1338 -0
  650. cumulusci/tests/shared_cassettes/GET_sobjects_Lead_describe.yaml +18 -0
  651. cumulusci/tests/shared_cassettes/GET_sobjects_OpportunityContactRole_describe.yaml +34 -0
  652. cumulusci/tests/shared_cassettes/GET_sobjects_Opportunity_describe.yaml +1261 -0
  653. cumulusci/tests/shared_cassettes/GET_sobjects_Organization.yaml +49 -0
  654. cumulusci/tests/shared_cassettes/vcr_string_templates/batchInfoList_xml.tpl +15 -0
  655. cumulusci/tests/shared_cassettes/vcr_string_templates/batchInfo_xml.tpl +13 -0
  656. cumulusci/tests/shared_cassettes/vcr_string_templates/jobInfo_insert_xml.tpl +24 -0
  657. cumulusci/tests/shared_cassettes/vcr_string_templates/jobInfo_upsert_xml.tpl +25 -0
  658. cumulusci/tests/test_entry_points.py +20 -0
  659. cumulusci/tests/test_integration_infrastructure.py +131 -0
  660. cumulusci/tests/test_main.py +9 -0
  661. cumulusci/tests/test_schema.py +32 -0
  662. cumulusci/tests/test_utils.py +657 -0
  663. cumulusci/tests/test_vcr_serializer.py +134 -0
  664. cumulusci/tests/uncompressed_cassette.yaml +83 -0
  665. cumulusci/tests/util.py +344 -0
  666. cumulusci/utils/__init__.py +731 -0
  667. cumulusci/utils/classutils.py +9 -0
  668. cumulusci/utils/collections.py +32 -0
  669. cumulusci/utils/deprecation.py +11 -0
  670. cumulusci/utils/encryption.py +31 -0
  671. cumulusci/utils/fileutils.py +295 -0
  672. cumulusci/utils/git.py +142 -0
  673. cumulusci/utils/http/multi_request.py +214 -0
  674. cumulusci/utils/http/requests_utils.py +103 -0
  675. cumulusci/utils/http/tests/cassettes/ManualEditTestCompositeParallelSalesforce.test_http_headers.yaml +32 -0
  676. cumulusci/utils/http/tests/cassettes/TestCompositeParallelSalesforce.test_composite_parallel_salesforce.yaml +65 -0
  677. cumulusci/utils/http/tests/cassettes/TestCompositeParallelSalesforce.test_errors.yaml +24 -0
  678. cumulusci/utils/http/tests/cassettes/TestCompositeParallelSalesforce.test_reference_ids.yaml +49 -0
  679. cumulusci/utils/http/tests/test_multi_request.py +255 -0
  680. cumulusci/utils/iterators.py +21 -0
  681. cumulusci/utils/logging.py +128 -0
  682. cumulusci/utils/metaprogramming.py +10 -0
  683. cumulusci/utils/options.py +138 -0
  684. cumulusci/utils/parallel/queries_in_parallel/run_queries_in_parallel.py +29 -0
  685. cumulusci/utils/parallel/queries_in_parallel/tests/test_run_queries_in_parallel.py +50 -0
  686. cumulusci/utils/parallel/task_worker_queues/parallel_worker.py +238 -0
  687. cumulusci/utils/parallel/task_worker_queues/parallel_worker_queue.py +243 -0
  688. cumulusci/utils/parallel/task_worker_queues/tests/test_parallel_worker.py +353 -0
  689. cumulusci/utils/salesforce/count_sobjects.py +46 -0
  690. cumulusci/utils/salesforce/soql.py +17 -0
  691. cumulusci/utils/salesforce/tests/cassettes/ManualEdit_TestCountSObjects.test_count_sobjects__network_errors.yaml +23 -0
  692. cumulusci/utils/salesforce/tests/cassettes/TestCountSObjects.test_count_sobjects__errors.yaml +33 -0
  693. cumulusci/utils/salesforce/tests/cassettes/TestCountSObjects.test_count_sobjects_simple.yaml +29 -0
  694. cumulusci/utils/salesforce/tests/test_count_sobjects.py +29 -0
  695. cumulusci/utils/salesforce/tests/test_soql.py +30 -0
  696. cumulusci/utils/tests/cassettes/ManualEditTestDescribeOrg.test_minimal_schema.yaml +36 -0
  697. cumulusci/utils/tests/cassettes/ManualEdit_test_describe_to_sql.yaml +191 -0
  698. cumulusci/utils/tests/test_fileutils.py +284 -0
  699. cumulusci/utils/tests/test_git.py +85 -0
  700. cumulusci/utils/tests/test_logging.py +70 -0
  701. cumulusci/utils/tests/test_option_parsing.py +188 -0
  702. cumulusci/utils/tests/test_org_schema.py +691 -0
  703. cumulusci/utils/tests/test_org_schema_models.py +79 -0
  704. cumulusci/utils/tests/test_waiting.py +25 -0
  705. cumulusci/utils/version_strings.py +391 -0
  706. cumulusci/utils/waiting.py +42 -0
  707. cumulusci/utils/xml/__init__.py +91 -0
  708. cumulusci/utils/xml/metadata_tree.py +299 -0
  709. cumulusci/utils/xml/robot_xml.py +114 -0
  710. cumulusci/utils/xml/salesforce_encoding.py +100 -0
  711. cumulusci/utils/xml/test/test_metadata_tree.py +251 -0
  712. cumulusci/utils/xml/test/test_salesforce_encoding.py +173 -0
  713. cumulusci/utils/yaml/cumulusci_yml.py +401 -0
  714. cumulusci/utils/yaml/model_parser.py +156 -0
  715. cumulusci/utils/yaml/safer_loader.py +74 -0
  716. cumulusci/utils/yaml/tests/bad_cci.yml +5 -0
  717. cumulusci/utils/yaml/tests/cassettes/TestCumulusciYml.test_validate_url__with_errors.yaml +20 -0
  718. cumulusci/utils/yaml/tests/test_cumulusci_yml.py +286 -0
  719. cumulusci/utils/yaml/tests/test_model_parser.py +175 -0
  720. cumulusci/utils/yaml/tests/test_safer_loader.py +88 -0
  721. cumulusci/utils/ziputils.py +61 -0
  722. cumulusci/vcs/base.py +143 -0
  723. cumulusci/vcs/bootstrap.py +272 -0
  724. cumulusci/vcs/github/__init__.py +24 -0
  725. cumulusci/vcs/github/adapter.py +689 -0
  726. cumulusci/vcs/github/release_notes/generator.py +219 -0
  727. cumulusci/vcs/github/release_notes/parser.py +151 -0
  728. cumulusci/vcs/github/release_notes/provider.py +143 -0
  729. cumulusci/vcs/github/service.py +569 -0
  730. cumulusci/vcs/github/tests/test_adapter.py +138 -0
  731. cumulusci/vcs/github/tests/test_service.py +408 -0
  732. cumulusci/vcs/models.py +586 -0
  733. cumulusci/vcs/tests/conftest.py +41 -0
  734. cumulusci/vcs/tests/dummy_service.py +241 -0
  735. cumulusci/vcs/tests/test_vcs_base.py +687 -0
  736. cumulusci/vcs/tests/test_vcs_bootstrap.py +727 -0
  737. cumulusci/vcs/utils/__init__.py +31 -0
  738. cumulusci/vcs/vcs_source.py +287 -0
  739. cumulusci_plus-5.0.0.dist-info/METADATA +145 -0
  740. cumulusci_plus-5.0.0.dist-info/RECORD +744 -0
  741. cumulusci_plus-5.0.0.dist-info/WHEEL +4 -0
  742. cumulusci_plus-5.0.0.dist-info/entry_points.txt +3 -0
  743. cumulusci_plus-5.0.0.dist-info/licenses/AUTHORS.rst +41 -0
  744. cumulusci_plus-5.0.0.dist-info/licenses/LICENSE +30 -0
@@ -0,0 +1,1311 @@
1
+ import os
2
+ from contextlib import contextmanager
3
+ from datetime import date, timedelta
4
+ from tempfile import TemporaryDirectory
5
+ from unittest import mock
6
+
7
+ import pytest
8
+ import responses
9
+ from sqlalchemy import create_engine
10
+
11
+ from cumulusci.core.exceptions import (
12
+ BulkDataException,
13
+ ConfigError,
14
+ CumulusCIException,
15
+ TaskOptionsError,
16
+ )
17
+ from cumulusci.tasks.bulkdata import ExtractData
18
+ from cumulusci.tasks.bulkdata.mapping_parser import MappingLookup, MappingStep
19
+ from cumulusci.tasks.bulkdata.step import (
20
+ BaseQueryOperation,
21
+ DataApi,
22
+ DataOperationJobResult,
23
+ DataOperationStatus,
24
+ DataOperationType,
25
+ )
26
+ from cumulusci.tasks.bulkdata.tests.utils import _make_task
27
+ from cumulusci.tests.util import (
28
+ assert_max_memory_usage,
29
+ mock_describe_calls,
30
+ mock_salesforce_client,
31
+ )
32
+ from cumulusci.utils import temporary_dir
33
+
34
+
35
+ @contextmanager
36
+ def mock_extract_jobs(task, extracted_records):
37
+ def _job_state_from_batches(self, job_id):
38
+ return DataOperationJobResult(
39
+ DataOperationStatus.SUCCESS,
40
+ [],
41
+ 10,
42
+ 0,
43
+ )
44
+
45
+ def get_results(self):
46
+ return extracted_records[self.sobject]
47
+
48
+ with mock.patch(
49
+ "cumulusci.tasks.bulkdata.step.BulkApiQueryOperation.get_results",
50
+ get_results,
51
+ ), mock.patch(
52
+ "cumulusci.tasks.bulkdata.step.BulkJobMixin._job_state_from_batches",
53
+ _job_state_from_batches,
54
+ ):
55
+ yield
56
+
57
+
58
+ class MockBulkQueryOperation(BaseQueryOperation):
59
+ def __init__(self, *, sobject, api_options, context, query):
60
+ super().__init__(
61
+ sobject=sobject, api_options=api_options, context=context, query=query
62
+ )
63
+ self.results = []
64
+
65
+ def query(self):
66
+ self.job_id = "JOB"
67
+ self.job_result = DataOperationJobResult(
68
+ DataOperationStatus.SUCCESS, [], len(self.results), 0
69
+ )
70
+
71
+ def get_results(self):
72
+ return iter(self.results)
73
+
74
+
75
+ class MockScalableBulkQueryOperation(MockBulkQueryOperation):
76
+ def query(self):
77
+ self.job_id = "JOB"
78
+ self.job_result = DataOperationJobResult(
79
+ DataOperationStatus.SUCCESS, [], self.result_len, 0
80
+ )
81
+
82
+
83
+ class TestExtractData:
84
+
85
+ mapping_file_v1 = "mapping_v1.yml"
86
+ mapping_file_v2 = "mapping_v2.yml"
87
+ mapping_file_poly = "mapping_poly.yml"
88
+ mapping_file_poly_wrong = "mapping_poly_wrong.yml"
89
+ mapping_file_poly_incomplete = "mapping_poly_incomplete.yml"
90
+ mapping_file_vanilla = "mapping_vanilla_sf.yml"
91
+
92
+ @responses.activate
93
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
94
+ def test_run__person_accounts_disabled(self, query_op_mock):
95
+ base_path = os.path.dirname(__file__)
96
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
97
+ mock_describe_calls()
98
+ with temporary_dir() as d:
99
+ tmp_db_path = os.path.join(d, "testdata.db")
100
+
101
+ task = _make_task(
102
+ ExtractData,
103
+ {
104
+ "options": {
105
+ "database_url": f"sqlite:///{tmp_db_path}",
106
+ "mapping": mapping_path,
107
+ }
108
+ },
109
+ )
110
+ task.bulk = mock.Mock()
111
+ task.sf = mock.Mock()
112
+ task.org_config._is_person_accounts_enabled = False
113
+
114
+ mock_query_households = MockBulkQueryOperation(
115
+ sobject="Account",
116
+ api_options={},
117
+ context=task,
118
+ query="SELECT Id, Name FROM Account",
119
+ )
120
+ mock_query_contacts = MockBulkQueryOperation(
121
+ sobject="Contact",
122
+ api_options={},
123
+ context=task,
124
+ query="SELECT Id, FirstName, LastName, Email, AccountId FROM Contact",
125
+ )
126
+ mock_query_households.results = [["1", "None"]]
127
+ mock_query_contacts.results = [
128
+ ["2", "First", "Last", "test@example.com", "1"]
129
+ ]
130
+
131
+ query_op_mock.side_effect = [mock_query_households, mock_query_contacts]
132
+
133
+ task()
134
+
135
+ with create_engine(task.options["database_url"]).connect() as conn:
136
+ household = next(conn.execute("select * from households"))
137
+ assert household.sf_id == "1"
138
+ assert not hasattr(household, "IsPersonAccount")
139
+ assert household.record_type == "HH_Account"
140
+
141
+ contact = next(conn.execute("select * from contacts"))
142
+ assert contact.sf_id == "2"
143
+ assert not hasattr(contact, "IsPersonAccount")
144
+ assert contact.household_id == "1"
145
+
146
+ @responses.activate
147
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
148
+ def test_run__person_accounts_enabled(self, query_op_mock):
149
+ base_path = os.path.dirname(__file__)
150
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
151
+ mock_describe_calls()
152
+
153
+ with temporary_dir() as d:
154
+ tmp_db_path = os.path.join(d, "testdata.db")
155
+
156
+ task = _make_task(
157
+ ExtractData,
158
+ {
159
+ "options": {
160
+ "database_url": f"sqlite:///{tmp_db_path}",
161
+ "mapping": mapping_path,
162
+ }
163
+ },
164
+ )
165
+ task.bulk = mock.Mock()
166
+ task.sf = mock.Mock()
167
+ task.org_config._is_person_accounts_enabled = True
168
+
169
+ mock_query_households = MockBulkQueryOperation(
170
+ sobject="Account",
171
+ api_options={},
172
+ context=task,
173
+ query="SELECT Id, Name IsPersonAccount FROM Account",
174
+ )
175
+ mock_query_contacts = MockBulkQueryOperation(
176
+ sobject="Contact",
177
+ api_options={},
178
+ context=task,
179
+ query="SELECT Id, FirstName, LastName, Email, IsPersonAccount, AccountId FROM Contact",
180
+ )
181
+ mock_query_households.results = [["1", "None", "false"]]
182
+ mock_query_contacts.results = [
183
+ ["2", "First", "Last", "test@example.com", "true", "1"]
184
+ ]
185
+
186
+ query_op_mock.side_effect = [mock_query_households, mock_query_contacts]
187
+
188
+ task()
189
+ with create_engine(task.options["database_url"]).connect() as conn:
190
+
191
+ household = next(conn.execute("select * from households"))
192
+ assert household.sf_id == "1"
193
+ assert household.IsPersonAccount == "false"
194
+ assert household.record_type == "HH_Account"
195
+
196
+ contact = next(conn.execute("select * from contacts"))
197
+ assert contact.sf_id == "2"
198
+ assert contact.IsPersonAccount == "true"
199
+ assert contact.household_id == "1"
200
+
201
+ @responses.activate
202
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
203
+ def test_run__sql(self, query_op_mock):
204
+ base_path = os.path.dirname(__file__)
205
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
206
+ mock_describe_calls()
207
+
208
+ with temporary_dir():
209
+ task = _make_task(
210
+ ExtractData,
211
+ {"options": {"sql_path": "testdata.sql", "mapping": mapping_path}},
212
+ )
213
+ task.bulk = mock.Mock()
214
+ task.sf = mock.Mock()
215
+ task.org_config._is_person_accounts_enabled = False
216
+
217
+ mock_query_households = MockBulkQueryOperation(
218
+ sobject="Account",
219
+ api_options={},
220
+ context=task,
221
+ query="SELECT Id FROM Account",
222
+ )
223
+ mock_query_contacts = MockBulkQueryOperation(
224
+ sobject="Contact",
225
+ api_options={},
226
+ context=task,
227
+ query="SELECT Id, FirstName, LastName, Email, AccountId FROM Contact",
228
+ )
229
+ mock_query_households.results = [["1"]]
230
+ mock_query_contacts.results = [
231
+ ["2", "First☃", "Last", "test@example.com", "1"]
232
+ ]
233
+ query_op_mock.side_effect = [mock_query_households, mock_query_contacts]
234
+
235
+ with mock.patch(
236
+ "cumulusci.tasks.bulkdata.extract.create_engine", wraps=create_engine
237
+ ) as ce_mock:
238
+ task()
239
+
240
+ assert os.path.exists("testdata.sql")
241
+ assert ce_mock.mock_calls[0][1][0].endswith(
242
+ "temp_db.db"
243
+ ), ce_mock.mock_calls[0][1][0]
244
+ assert ce_mock.mock_calls[0][1][0].startswith(
245
+ "sqlite:///"
246
+ ), ce_mock.mock_calls[0][1][0]
247
+
248
+ @responses.activate
249
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
250
+ def test_run__v2__person_accounts_disabled(self, query_op_mock):
251
+ base_path = os.path.dirname(__file__)
252
+ mapping_path = os.path.join(base_path, self.mapping_file_v2)
253
+ mock_describe_calls()
254
+
255
+ with TemporaryDirectory() as t:
256
+ task = _make_task(
257
+ ExtractData,
258
+ {
259
+ "options": {
260
+ "database_url": f"sqlite:///{t}/temp_db", # in memory
261
+ "mapping": mapping_path,
262
+ }
263
+ },
264
+ )
265
+ task.bulk = mock.Mock()
266
+ task.sf = mock.Mock()
267
+ task.org_config._is_person_accounts_enabled = False
268
+
269
+ mock_query_households = MockBulkQueryOperation(
270
+ sobject="Account",
271
+ api_options={},
272
+ context=task,
273
+ query="SELECT Id, Name FROM Account",
274
+ )
275
+ mock_query_contacts = MockBulkQueryOperation(
276
+ sobject="Contact",
277
+ api_options={},
278
+ context=task,
279
+ query="SELECT Id, FirstName, LastName, Email, AccountId FROM Contact",
280
+ )
281
+ mock_query_households.results = [["1", "TestHousehold"]]
282
+ mock_query_contacts.results = [
283
+ ["2", "First", "Last", "test@example.com", "1"]
284
+ ]
285
+
286
+ query_op_mock.side_effect = [mock_query_households, mock_query_contacts]
287
+
288
+ task()
289
+ with create_engine(task.options["database_url"]).connect() as conn:
290
+ household = next(conn.execute("select * from households"))
291
+ assert household.name == "TestHousehold"
292
+ assert not hasattr(household, "IsPersonAccount")
293
+ assert household.record_type == "HH_Account"
294
+
295
+ contact = next(conn.execute("select * from contacts"))
296
+ assert contact.household_id == "Account-1"
297
+ assert not hasattr(contact, "IsPersonAccount")
298
+
299
+ @responses.activate
300
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
301
+ def test_run__v2__person_accounts_enabled(self, query_op_mock):
302
+ base_path = os.path.dirname(__file__)
303
+ mapping_path = os.path.join(base_path, self.mapping_file_v2)
304
+ mock_describe_calls()
305
+ with temporary_dir() as d:
306
+ tmp_db_path = os.path.join(d, "testdata.db")
307
+ task = _make_task(
308
+ ExtractData,
309
+ {
310
+ "options": {
311
+ "database_url": f"sqlite:///{tmp_db_path}",
312
+ "mapping": mapping_path,
313
+ }
314
+ },
315
+ )
316
+ task.bulk = mock.Mock()
317
+ task.sf = mock.Mock()
318
+ task.org_config._is_person_accounts_enabled = True
319
+
320
+ mock_query_households = MockBulkQueryOperation(
321
+ sobject="Account",
322
+ api_options={},
323
+ context=task,
324
+ query="SELECT Id, Name, IsPersonAccount FROM Account",
325
+ )
326
+ mock_query_contacts = MockBulkQueryOperation(
327
+ sobject="Contact",
328
+ api_options={},
329
+ context=task,
330
+ query="SELECT Id, FirstName, LastName, Email, IsPersonAccount, AccountId FROM Contact",
331
+ )
332
+ mock_query_households.results = [["1", "TestHousehold", "false"]]
333
+ mock_query_contacts.results = [
334
+ ["2", "First", "Last", "test@example.com", "true", "1"]
335
+ ]
336
+
337
+ query_op_mock.side_effect = [mock_query_households, mock_query_contacts]
338
+
339
+ task()
340
+ with create_engine(task.options["database_url"]).connect() as conn:
341
+ household = next(conn.execute("select * from households"))
342
+ assert household.name == "TestHousehold"
343
+ assert household.IsPersonAccount == "false"
344
+ assert household.record_type == "HH_Account"
345
+
346
+ contact = next(conn.execute("select * from contacts"))
347
+ assert contact.household_id == "Account-1"
348
+ assert contact.IsPersonAccount == "true"
349
+
350
+ @responses.activate
351
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
352
+ def test_run__poly__polymorphic_lookups(self, query_op_mock):
353
+ """Test for polymorphic lookups"""
354
+ base_path = os.path.dirname(__file__)
355
+ mapping_path = os.path.join(base_path, self.mapping_file_poly)
356
+ mock_describe_calls()
357
+
358
+ with temporary_dir() as t:
359
+ task = _make_task(
360
+ ExtractData,
361
+ {
362
+ "options": {
363
+ "database_url": f"sqlite:///{t}/temp_poly.db", # in memory
364
+ "mapping": mapping_path,
365
+ }
366
+ },
367
+ )
368
+ task.bulk = mock.Mock()
369
+ task.sf = mock.Mock()
370
+ task.org_config._is_person_accounts_enabled = False
371
+
372
+ mock_query_households = MockBulkQueryOperation(
373
+ sobject="Account",
374
+ api_options={},
375
+ context=task,
376
+ query="SELECT Id, Name FROM Account",
377
+ )
378
+ mock_query_contacts = MockBulkQueryOperation(
379
+ sobject="Contact",
380
+ api_options={},
381
+ context=task,
382
+ query="SELECT Id, FirstName, LastName, Email, AccountId FROM Contact",
383
+ )
384
+ mock_query_events = MockBulkQueryOperation(
385
+ sobject="Event",
386
+ api_options={},
387
+ context=task,
388
+ query="SELECT Id, LastName, WhoId FROM Event",
389
+ )
390
+ mock_query_households.results = [["abc123", "TestHousehold"]]
391
+ mock_query_contacts.results = [
392
+ ["def456", "First", "Last", "test@example.com", "abc123"]
393
+ ]
394
+ mock_query_events.results = [
395
+ ["ijk789", "Last1", "abc123"],
396
+ ["lmn010", "Last2", "def456"],
397
+ ]
398
+
399
+ query_op_mock.side_effect = [
400
+ mock_query_households,
401
+ mock_query_contacts,
402
+ mock_query_events,
403
+ ]
404
+ task()
405
+ with create_engine(task.options["database_url"]).connect() as conn:
406
+ household = next(conn.execute("select * from households"))
407
+ assert household.name == "TestHousehold"
408
+ assert not hasattr(household, "IsPersonAccount")
409
+ assert household.record_type == "HH_Account"
410
+
411
+ contact = next(conn.execute("select * from contacts"))
412
+ assert contact.household_id == "Account-1"
413
+ assert not hasattr(contact, "IsPersonAccount")
414
+
415
+ events = conn.execute("select * from events").fetchall()
416
+ assert events[0].who_id == "Account-1"
417
+ assert events[1].who_id == "Contact-1"
418
+
419
+ @responses.activate
420
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
421
+ def test_run__poly__wrong_mapping(self, query_op_mock):
422
+ """Test for polymorphic lookups with wrong mapping file
423
+ (missing table)"""
424
+ base_path = os.path.dirname(__file__)
425
+ mapping_path = os.path.join(base_path, self.mapping_file_poly_wrong)
426
+ mock_describe_calls()
427
+
428
+ with temporary_dir() as t:
429
+ task = _make_task(
430
+ ExtractData,
431
+ {
432
+ "options": {
433
+ "database_url": f"sqlite:///{t}/temp_poly.db", # in memory
434
+ "mapping": mapping_path,
435
+ }
436
+ },
437
+ )
438
+ task.bulk = mock.Mock()
439
+ task.sf = mock.Mock()
440
+ task.org_config._is_person_accounts_enabled = False
441
+
442
+ mock_query_households = MockBulkQueryOperation(
443
+ sobject="Account",
444
+ api_options={},
445
+ context=task,
446
+ query="SELECT Id, Name FROM Account",
447
+ )
448
+ mock_query_contacts = MockBulkQueryOperation(
449
+ sobject="Contact",
450
+ api_options={},
451
+ context=task,
452
+ query="SELECT Id, FirstName, LastName, Email, AccountId FROM Contact",
453
+ )
454
+ mock_query_events = MockBulkQueryOperation(
455
+ sobject="Event",
456
+ api_options={},
457
+ context=task,
458
+ query="SELECT Id, LastName, WhoId FROM Event",
459
+ )
460
+ mock_query_households.results = [["abc123", "TestHousehold"]]
461
+ mock_query_contacts.results = [
462
+ ["def456", "First", "Last", "test@example.com", "abc123"]
463
+ ]
464
+ mock_query_events.results = [
465
+ ["ijk789", "Last1", "abc123"],
466
+ ["lmn010", "Last2", "def456"],
467
+ ]
468
+
469
+ query_op_mock.side_effect = [
470
+ mock_query_households,
471
+ mock_query_contacts,
472
+ mock_query_events,
473
+ ]
474
+ with pytest.raises(CumulusCIException) as e:
475
+ task()
476
+
477
+ assert "The following tables are missing in the mapping file:" in str(
478
+ e.value
479
+ )
480
+
481
+ @responses.activate
482
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
483
+ def test_run__poly__incomplete_mapping(self, query_op_mock):
484
+ """Test for polymorphic lookups with incomplete mapping file"""
485
+ base_path = os.path.dirname(__file__)
486
+ mapping_path = os.path.join(base_path, self.mapping_file_poly_incomplete)
487
+ mock_describe_calls()
488
+
489
+ with temporary_dir() as t:
490
+ task = _make_task(
491
+ ExtractData,
492
+ {
493
+ "options": {
494
+ "database_url": f"sqlite:///{t}/temp_poly.db", # in memory
495
+ "mapping": mapping_path,
496
+ }
497
+ },
498
+ )
499
+ task.bulk = mock.Mock()
500
+ task.sf = mock.Mock()
501
+ task.org_config._is_person_accounts_enabled = False
502
+
503
+ mock_query_households = MockBulkQueryOperation(
504
+ sobject="Account",
505
+ api_options={},
506
+ context=task,
507
+ query="SELECT Id, Name FROM Account",
508
+ )
509
+ mock_query_events = MockBulkQueryOperation(
510
+ sobject="Event",
511
+ api_options={},
512
+ context=task,
513
+ query="SELECT Id, LastName, WhoId FROM Event",
514
+ )
515
+ mock_query_households.results = [["abc123", "TestHousehold"]]
516
+ mock_query_events.results = [
517
+ ["ijk789", "Last1", "abc123"],
518
+ ["lmn010", "Last2", "def456"],
519
+ ]
520
+
521
+ query_op_mock.side_effect = [mock_query_households, mock_query_events]
522
+ with pytest.raises(ConfigError) as e:
523
+ task()
524
+
525
+ assert "Total mapping operations" in str(e.value)
526
+ assert "do not match total non-empty rows" in str(e.value)
527
+
528
+ @mock.patch("cumulusci.tasks.bulkdata.extract.log_progress")
529
+ def test_import_results__oid_as_pk(self, log_mock):
530
+ task = _make_task(
531
+ ExtractData,
532
+ {"options": {"database_url": "sqlite://", "mapping": "mapping.yml"}},
533
+ )
534
+
535
+ mapping = MappingStep(
536
+ sf_object="Opportunity",
537
+ fields={"Id": "sf_id", "Name": "Name"},
538
+ lookups={"AccountId": MappingLookup(table="Account", name="AccountId")},
539
+ )
540
+ step = mock.Mock()
541
+ task.session = mock.Mock()
542
+ task.metadata = mock.MagicMock()
543
+
544
+ with mock.patch(
545
+ "cumulusci.tasks.bulkdata.extract.sql_bulk_insert_from_records"
546
+ ) as sql_bulk_insert_from_records:
547
+ task._import_results(mapping, step)
548
+
549
+ task.session.connection.assert_called_once_with()
550
+ step.get_results.assert_called_once_with()
551
+ sql_bulk_insert_from_records.assert_called_once_with(
552
+ connection=task.session.connection.return_value,
553
+ table=task.metadata.tables[mapping.table],
554
+ columns=["sf_id", "Name", "AccountId"],
555
+ record_iterable=log_mock.return_value,
556
+ )
557
+
558
+ @responses.activate
559
+ def test_import_results__no_columns(self): # , query_op_mock):
560
+ base_path = os.path.dirname(__file__)
561
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
562
+ mock_describe_calls()
563
+
564
+ task = _make_task(
565
+ ExtractData,
566
+ {"options": {"database_url": "sqlite://", "mapping": mapping_path}},
567
+ )
568
+
569
+ mapping = MappingStep(
570
+ sf_object="Opportunity",
571
+ table="Opportunity",
572
+ fields={},
573
+ lookups={},
574
+ )
575
+ step = mock.Mock()
576
+ step.get_results.return_value = [[1], [2]]
577
+ task.session = mock.Mock()
578
+ task._init_task()
579
+ task._init_mapping()
580
+ task.mapping["Opportunity"] = mapping
581
+ with task._init_db():
582
+ task._import_results(mapping, step)
583
+ output_Opportunties = list(
584
+ task.session.execute("select * from Opportunity")
585
+ )
586
+ assert output_Opportunties == [("Opportunity-1",), ("Opportunity-2",)]
587
+
588
+ @responses.activate
589
+ def test_import_results__relative_dates(self):
590
+ mock_describe_calls()
591
+ task = _make_task(
592
+ ExtractData,
593
+ {"options": {"database_url": "sqlite://", "mapping": "mapping.yml"}},
594
+ )
595
+
596
+ mapping = MappingStep(
597
+ sf_object="Opportunity",
598
+ fields={"Id": "sf_id", "CloseDate": "CloseDate"},
599
+ anchor_date="2020-07-01",
600
+ )
601
+ step = mock.Mock()
602
+ step.get_results.return_value = iter(
603
+ [["006000000000001", (date.today() + timedelta(days=9)).isoformat()]]
604
+ )
605
+ task.session = mock.Mock()
606
+ task.metadata = mock.MagicMock()
607
+ task._init_task()
608
+ with mock.patch(
609
+ "cumulusci.tasks.bulkdata.extract.sql_bulk_insert_from_records"
610
+ ) as sql_bulk_insert_from_records:
611
+ task._import_results(mapping, step)
612
+
613
+ task.session.connection.assert_called_once_with()
614
+ step.get_results.assert_called_once_with()
615
+ sql_bulk_insert_from_records.assert_called_once_with(
616
+ connection=task.session.connection.return_value,
617
+ table=task.metadata.tables[mapping.table],
618
+ columns=["sf_id", "CloseDate"],
619
+ record_iterable=mock.ANY,
620
+ )
621
+
622
+ result = list(
623
+ sql_bulk_insert_from_records.call_args_list[0][1]["record_iterable"]
624
+ )
625
+ assert result == [["006000000000001", "2020-07-10"]]
626
+
627
+ def test_import_results__record_type_mapping(self):
628
+ base_path = os.path.dirname(__file__)
629
+ mapping_path = os.path.join(base_path, "recordtypes.yml")
630
+ task = _make_task(
631
+ ExtractData,
632
+ {"options": {"database_url": "sqlite://", "mapping": mapping_path}},
633
+ )
634
+ task._extract_record_types = mock.Mock()
635
+ task.metadata = mock.MagicMock()
636
+ task.session = mock.Mock()
637
+ task.org_config._is_person_accounts_enabled = False
638
+
639
+ step = mock.Mock()
640
+ step.get_results.return_value = [["000000000000001", "Test", "012000000000000"]]
641
+
642
+ mapping = MappingStep(
643
+ sf_object="Account",
644
+ fields={"Name": "Name", "RecordTypeId": "RecordTypeId"},
645
+ lookups={},
646
+ table="accounts",
647
+ )
648
+ with mock.patch(
649
+ "cumulusci.tasks.bulkdata.extract.sql_bulk_insert_from_records_incremental",
650
+ return_value=[None, None],
651
+ ):
652
+ task._import_results(
653
+ mapping,
654
+ step,
655
+ )
656
+ task._extract_record_types.assert_called_once_with(
657
+ "Account",
658
+ mapping.get_source_record_type_table(),
659
+ task.session.connection.return_value,
660
+ task.org_config._is_person_accounts_enabled,
661
+ )
662
+
663
+ def test_import_results__person_account_name_stripped(self):
664
+ base_path = os.path.dirname(__file__)
665
+ mapping_path = os.path.join(base_path, "recordtypes.yml")
666
+ task = _make_task(
667
+ ExtractData,
668
+ {"options": {"database_url": "sqlite://", "mapping": mapping_path}},
669
+ )
670
+ task._extract_record_types = mock.Mock()
671
+ task.session = mock.Mock()
672
+ task.metadata = mock.MagicMock()
673
+ task.org_config._is_person_accounts_enabled = True
674
+
675
+ step = mock.Mock()
676
+ step.get_results.return_value = [
677
+ ["000000000000001", "Person Account", "012000000000001", "true"],
678
+ ["000000000000002", "Business Account", "012000000000002", "false"],
679
+ ]
680
+
681
+ with mock.patch(
682
+ "cumulusci.tasks.bulkdata.extract.sql_bulk_insert_from_records"
683
+ ) as sql_bulk_insert_from_records:
684
+ task._import_results(
685
+ MappingStep(
686
+ sf_object="Account",
687
+ fields={
688
+ "Id": "sf_id",
689
+ "Name": "Name",
690
+ "RecordTypeId": "RecordTypeId",
691
+ "IsPersonAccount": "IsPersonAccount",
692
+ },
693
+ lookups={},
694
+ ),
695
+ step,
696
+ )
697
+
698
+ sql_bulk_insert_from_records.assert_called()
699
+ args, kwargs = sql_bulk_insert_from_records.call_args_list[0]
700
+
701
+ records = [record for record in kwargs["record_iterable"]]
702
+
703
+ assert [
704
+ ["000000000000001", "", "012000000000001", "true"],
705
+ ["000000000000002", "Business Account", "012000000000002", "false"],
706
+ ] == records
707
+
708
+ @responses.activate
709
+ def test_map_autopks(self):
710
+ mock_describe_calls()
711
+ base_path = os.path.dirname(__file__)
712
+ mapping_path = os.path.join(base_path, self.mapping_file_v2)
713
+ mock_describe_calls()
714
+
715
+ task = _make_task(
716
+ ExtractData,
717
+ {
718
+ "options": {
719
+ "database_url": "sqlite://", # in memory
720
+ "mapping": mapping_path,
721
+ }
722
+ },
723
+ )
724
+ task._convert_lookups_to_id = mock.Mock()
725
+ task.metadata = mock.MagicMock()
726
+
727
+ task._init_task()
728
+ task._init_mapping()
729
+ task._map_autopks()
730
+
731
+ task._convert_lookups_to_id.assert_called_once_with(
732
+ task.mapping["Insert Contacts"], ["AccountId"]
733
+ )
734
+ task.metadata.tables.__getitem__.return_value.drop.assert_has_calls(
735
+ [mock.call(), mock.call()]
736
+ )
737
+ task.metadata.tables.__getitem__.assert_has_calls(
738
+ [mock.call("contacts_sf_ids"), mock.call("households_sf_ids")],
739
+ any_order=True,
740
+ )
741
+
742
+ def test_convert_lookups_to_id(self):
743
+ task = _make_task(
744
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
745
+ )
746
+
747
+ task.session = mock.Mock()
748
+ task.models = {
749
+ "Account": mock.Mock(),
750
+ "Account_sf_ids": mock.Mock(),
751
+ "Opportunity": mock.Mock(),
752
+ "Opportunity_sf_ids": mock.Mock(),
753
+ }
754
+ task.mapping = {
755
+ "Account": MappingStep(sf_object="Account"),
756
+ "Opportunity": MappingStep(sf_object="Opportunity"),
757
+ }
758
+
759
+ task.session.query.return_value.filter.return_value.count.return_value = 0
760
+ task.session.query.return_value.filter.return_value.update.return_value.rowcount = (
761
+ 0
762
+ )
763
+ task._convert_lookups_to_id(
764
+ MappingStep(
765
+ sf_object="Opportunity",
766
+ lookups={"AccountId": MappingLookup(table="Account", name="AccountId")},
767
+ ),
768
+ ["AccountId"],
769
+ )
770
+
771
+ task.session.query.return_value.filter.return_value.update.assert_called_once_with(
772
+ {task.models["Opportunity"].AccountId: task.models["Account_sf_ids"].id},
773
+ synchronize_session=False,
774
+ )
775
+ task.session.commit.assert_called_once_with()
776
+
777
+ def test_convert_lookups_to_id__sqlite(self):
778
+ task = _make_task(
779
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
780
+ )
781
+
782
+ task.session = mock.Mock()
783
+ task.models = {
784
+ "Account": mock.Mock(),
785
+ "Account_sf_ids": mock.Mock(),
786
+ "Opportunity": mock.Mock(),
787
+ "Opportunity_sf_ids": mock.Mock(),
788
+ }
789
+ task.mapping = {
790
+ "Account": MappingStep(sf_object="Account"),
791
+ "Opportunity": MappingStep(sf_object="Opportunity"),
792
+ }
793
+ task.session.query.return_value.filter.return_value.update.side_effect = (
794
+ NotImplementedError
795
+ )
796
+
797
+ item = mock.Mock()
798
+
799
+ task.session.query.return_value.join.return_value = [(item, "1")]
800
+ task.session.query.return_value.filter.return_value.count.return_value = 1
801
+
802
+ task._convert_lookups_to_id(
803
+ MappingStep(
804
+ sf_object="Opportunity",
805
+ lookups={"AccountId": MappingLookup(table="Account", name="AccountId")},
806
+ ),
807
+ ["AccountId"],
808
+ )
809
+
810
+ task.session.bulk_update_mappings.assert_called_once_with(
811
+ task.models["Opportunity"], [{"id": item.id, "AccountId": "1"}]
812
+ )
813
+ task.session.commit.assert_called_once_with()
814
+
815
+ @mock.patch("cumulusci.tasks.bulkdata.extract.create_table")
816
+ @mock.patch("cumulusci.tasks.bulkdata.extract.mapper")
817
+ def test_create_table(self, mapper_mock, create_mock):
818
+ task = _make_task(
819
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
820
+ )
821
+ mapping = MappingStep(
822
+ sf_object="Account",
823
+ fields={"Name": "Name", "Id": "sf_id"},
824
+ lookups={},
825
+ table="accounts",
826
+ )
827
+ task.models = {}
828
+ task.metadata = mock.Mock()
829
+ task._create_table(mapping)
830
+ create_mock.assert_called_once_with(mapping, task.metadata)
831
+
832
+ assert "accounts" in task.models
833
+
834
+ @responses.activate
835
+ def test_create_table__already_exists(self):
836
+ base_path = os.path.dirname(__file__)
837
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
838
+ db_path = os.path.join(base_path, "testdata.db")
839
+ mock_describe_calls()
840
+ task = _make_task(
841
+ ExtractData,
842
+ {
843
+ "options": {
844
+ "database_url": f"sqlite:///{db_path}",
845
+ "mapping": mapping_path,
846
+ }
847
+ },
848
+ )
849
+ task.org_config._is_person_accounts_enabled = False
850
+
851
+ with pytest.raises(BulkDataException):
852
+ task()
853
+
854
+ def test_create_table__record_type_mapping(self):
855
+ task = _make_task(
856
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
857
+ )
858
+ task.mapping = {
859
+ "Insert Accounts": MappingStep(
860
+ sf_object="Account",
861
+ table="accounts",
862
+ fields={"Name": "Name", "RecordTypeId": "RecordTypeId"},
863
+ lookups={},
864
+ ),
865
+ "Insert Other Accounts": MappingStep(
866
+ sf_object="Account",
867
+ fields={"Name": "Name", "RecordTypeId": "RecordTypeId"},
868
+ lookups={},
869
+ table="accounts_2",
870
+ ),
871
+ }
872
+ task.org_config._is_person_accounts_enabled = False
873
+
874
+ def create_table_mock(table_name):
875
+ task.models[table_name] = mock.Mock()
876
+
877
+ task._create_record_type_table = mock.Mock(side_effect=create_table_mock)
878
+ with task._init_db():
879
+ task._create_record_type_table.assert_called_once_with("Account_rt_mapping")
880
+
881
+ @mock.patch("cumulusci.tasks.bulkdata.extract.create_table")
882
+ @mock.patch("cumulusci.tasks.bulkdata.extract.Table")
883
+ @mock.patch("cumulusci.tasks.bulkdata.extract.mapper")
884
+ def test_create_table__autopk(self, mapper_mock, table_mock, create_mock):
885
+ task = _make_task(
886
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
887
+ )
888
+ mapping = MappingStep(
889
+ sf_object="Account",
890
+ fields={"Name": "Name"},
891
+ table="accounts",
892
+ )
893
+ task.models = {}
894
+ task.metadata = mock.Mock()
895
+ task.org_config._is_person_accounts_enabled = False
896
+
897
+ task._create_table(mapping)
898
+
899
+ create_mock.assert_called_once_with(mapping, task.metadata)
900
+ assert len(table_mock.mock_calls) == 1
901
+
902
+ assert "accounts" in task.models
903
+ assert mapping.get_sf_id_table() in task.models
904
+
905
+ def test_create_tables(self):
906
+ task = _make_task(
907
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
908
+ )
909
+ task.mapping = {1: "test", 2: "foo", 3: "bar"}
910
+ task.metadata = mock.Mock()
911
+ task._create_table = mock.Mock()
912
+
913
+ task._create_tables()
914
+
915
+ task._create_table.assert_has_calls(
916
+ [mock.call("test"), mock.call("foo"), mock.call("bar")]
917
+ )
918
+ task.metadata.create_all.assert_called_once_with()
919
+
920
+ def test_init_db(self):
921
+ task = _make_task(
922
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
923
+ )
924
+ task._create_tables = mock.Mock()
925
+ with task._init_db():
926
+ assert task.models == {}
927
+ assert task.session.query
928
+
929
+ def assert_person_accounts_in_mapping(
930
+ self, mapping, org_has_person_accounts_enabled
931
+ ):
932
+ for step in mapping.values():
933
+ if step["sf_object"] in ("Account", "Contact"):
934
+ assert org_has_person_accounts_enabled == (
935
+ "IsPersonAccount" in step["fields"]
936
+ )
937
+
938
+ @responses.activate
939
+ def test_init_mapping(self):
940
+ base_path = os.path.dirname(__file__)
941
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
942
+ mock_describe_calls()
943
+ task = _make_task(
944
+ ExtractData,
945
+ {"options": {"database_url": "sqlite:///", "mapping": mapping_path}},
946
+ )
947
+ task.org_config._is_person_accounts_enabled = False
948
+
949
+ task._init_task()
950
+ task._init_mapping()
951
+ assert "Insert Households" in task.mapping
952
+
953
+ # Person Accounts should not be added to mapping
954
+ self.assert_person_accounts_in_mapping(task.mapping, False)
955
+
956
+ @responses.activate
957
+ def test_init_mapping_org_has_person_accounts_enabled(self):
958
+ base_path = os.path.dirname(__file__)
959
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
960
+ mock_describe_calls()
961
+ task = _make_task(
962
+ ExtractData,
963
+ {"options": {"database_url": "sqlite:///", "mapping": mapping_path}},
964
+ )
965
+ task.org_config._is_person_accounts_enabled = True
966
+
967
+ task._init_task()
968
+ task._init_mapping()
969
+ assert "Insert Households" in task.mapping
970
+
971
+ # Person Accounts should not be added to mapping
972
+ self.assert_person_accounts_in_mapping(task.mapping, True)
973
+
974
+ @mock.patch("cumulusci.tasks.bulkdata.extract.validate_and_inject_mapping")
975
+ def test_init_mapping_passes_options_to_validate(self, validate_and_inject_mapping):
976
+ base_path = os.path.dirname(__file__)
977
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
978
+ t = _make_task(
979
+ ExtractData,
980
+ {
981
+ "options": {
982
+ "database_url": "sqlite:///",
983
+ "mapping": mapping_path,
984
+ "inject_namespaces": True,
985
+ "drop_missing_schema": True,
986
+ }
987
+ },
988
+ )
989
+ t.org_config._is_person_accounts_enabled = True
990
+
991
+ t._init_task()
992
+ t._init_mapping()
993
+
994
+ validate_and_inject_mapping.assert_called_once_with(
995
+ mapping=t.mapping,
996
+ sf=t.sf,
997
+ namespace=t.project_config.project__package__namespace,
998
+ data_operation=DataOperationType.QUERY,
999
+ inject_namespaces=True,
1000
+ drop_missing=True,
1001
+ org_has_person_accounts_enabled=t.org_config._is_person_accounts_enabled,
1002
+ )
1003
+
1004
+ def test_soql_for_mapping(self):
1005
+ task = _make_task(
1006
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
1007
+ )
1008
+ mapping = MappingStep(
1009
+ sf_object="Contact",
1010
+ fields={"Id": "sf_id", "Test__c": "Test"},
1011
+ )
1012
+ assert task._soql_for_mapping(mapping) == "SELECT Id, Test__c FROM Contact"
1013
+
1014
+ mapping = MappingStep(
1015
+ sf_object="Contact",
1016
+ record_type="Devel",
1017
+ fields={"Id": "sf_id", "Test__c": "Test"},
1018
+ )
1019
+ assert (
1020
+ task._soql_for_mapping(mapping)
1021
+ == "SELECT Id, Test__c FROM Contact WHERE RecordType.DeveloperName = 'Devel'"
1022
+ )
1023
+
1024
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
1025
+ def test_run_query(self, query_op_mock):
1026
+ task = _make_task(
1027
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
1028
+ )
1029
+ task._import_results = mock.Mock()
1030
+ query_op_mock.return_value.job_result = DataOperationJobResult(
1031
+ DataOperationStatus.SUCCESS, [], 1, 0
1032
+ )
1033
+
1034
+ task._run_query("SELECT Id FROM Contact", MappingStep(sf_object="Contact"))
1035
+
1036
+ query_op_mock.assert_called_once_with(
1037
+ sobject="Contact",
1038
+ fields=["Id"],
1039
+ api=DataApi.SMART,
1040
+ api_options={},
1041
+ context=task,
1042
+ query="SELECT Id FROM Contact",
1043
+ )
1044
+ query_op_mock.return_value.query.assert_called_once_with()
1045
+ task._import_results.assert_called_once_with(
1046
+ MappingStep(sf_object="Contact"), query_op_mock.return_value
1047
+ )
1048
+
1049
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
1050
+ def test_run_query__no_results(self, query_op_mock):
1051
+ task = _make_task(
1052
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
1053
+ )
1054
+ task._import_results = mock.Mock()
1055
+ query_op_mock.return_value.job_result = DataOperationJobResult(
1056
+ DataOperationStatus.SUCCESS, [], 0, 0
1057
+ )
1058
+
1059
+ task._run_query("SELECT Id FROM Contact", MappingStep(sf_object="Contact"))
1060
+
1061
+ query_op_mock.assert_called_once_with(
1062
+ sobject="Contact",
1063
+ fields=["Id"],
1064
+ api=DataApi.SMART,
1065
+ api_options={},
1066
+ context=task,
1067
+ query="SELECT Id FROM Contact",
1068
+ )
1069
+ query_op_mock.return_value.query.assert_called_once_with()
1070
+ task._import_results.assert_not_called()
1071
+
1072
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
1073
+ def test_run_query__failure(self, query_op_mock):
1074
+ task = _make_task(
1075
+ ExtractData, {"options": {"database_url": "sqlite:///", "mapping": ""}}
1076
+ )
1077
+ query_op_mock.return_value.job_result = DataOperationJobResult(
1078
+ DataOperationStatus.JOB_FAILURE, [], 1, 0
1079
+ )
1080
+
1081
+ with pytest.raises(BulkDataException):
1082
+ task._run_query("SELECT Id FROM Contact", MappingStep(sf_object="Contact"))
1083
+
1084
+ def test_init_options__missing_output(self):
1085
+ with pytest.raises(TaskOptionsError):
1086
+ _make_task(ExtractData, {"options": {}})
1087
+
1088
+ @mock.patch("cumulusci.tasks.bulkdata.extract.log_progress")
1089
+ def test_extract_respects_key_field(self, log_mock):
1090
+ task = _make_task(
1091
+ ExtractData,
1092
+ {"options": {"database_url": "sqlite://", "mapping": "mapping.yml"}},
1093
+ )
1094
+
1095
+ mapping = MappingStep(
1096
+ sf_object="Opportunity",
1097
+ table="Opportunity",
1098
+ fields={"Id": "sf_id", "Name": "Name"},
1099
+ lookups={
1100
+ "AccountId": MappingLookup(
1101
+ table="Account", key_field="account_id", name="AccountId"
1102
+ )
1103
+ },
1104
+ )
1105
+ step = mock.Mock()
1106
+ task.session = mock.Mock()
1107
+ task.metadata = mock.MagicMock()
1108
+
1109
+ with mock.patch(
1110
+ "cumulusci.tasks.bulkdata.extract.sql_bulk_insert_from_records"
1111
+ ) as sql_bulk_insert_from_records:
1112
+ task._import_results(mapping, step)
1113
+
1114
+ task.session.connection.assert_called_once_with()
1115
+ step.get_results.assert_called_once_with()
1116
+ sql_bulk_insert_from_records.assert_called_once_with(
1117
+ connection=task.session.connection.return_value,
1118
+ table=task.metadata.tables["Opportunity"],
1119
+ columns=["sf_id", "Name", "account_id"],
1120
+ record_iterable=log_mock.return_value,
1121
+ )
1122
+
1123
+ @responses.activate
1124
+ @mock.patch("cumulusci.tasks.bulkdata.extract.get_query_operation")
1125
+ def test_extract_memory_usage(self, step_mock):
1126
+ with TemporaryDirectory() as t:
1127
+ base_path = os.path.dirname(__file__)
1128
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
1129
+ mock_describe_calls()
1130
+
1131
+ task = _make_task(
1132
+ ExtractData,
1133
+ {
1134
+ "options": {
1135
+ "database_url": f"sqlite:///{t}/foo.db", # tempdir database
1136
+ "mapping": mapping_path,
1137
+ }
1138
+ },
1139
+ )
1140
+ task.bulk = mock.Mock()
1141
+ task.sf = mock.Mock()
1142
+ task.org_config._is_person_accounts_enabled = False
1143
+
1144
+ mock_query_households = MockScalableBulkQueryOperation(
1145
+ sobject="Account",
1146
+ api_options={},
1147
+ context=task,
1148
+ query="SELECT Id FROM Account",
1149
+ )
1150
+ mock_query_contacts = MockScalableBulkQueryOperation(
1151
+ sobject="Contact",
1152
+ api_options={},
1153
+ context=task,
1154
+ query="SELECT Id, FirstName, LastName, Email, AccountId FROM Contact",
1155
+ )
1156
+ mock_query_households.results = ([str(num)] for num in range(1, 20000))
1157
+ mock_query_households.result_len = 20000
1158
+ mock_query_contacts.results = [
1159
+ ["2", "First", "Last", "test@example.com", "1"]
1160
+ ]
1161
+ mock_query_contacts.result_len = 1
1162
+
1163
+ step_mock.side_effect = [mock_query_households, mock_query_contacts]
1164
+
1165
+ with assert_max_memory_usage(15 * 10**6):
1166
+ task()
1167
+
1168
+ @responses.activate
1169
+ def test_import_results__autopk(self, create_task_fixture):
1170
+ mock_describe_calls()
1171
+ base_path = os.path.dirname(__file__)
1172
+ mapping_path = os.path.join(base_path, self.mapping_file_vanilla)
1173
+ with temporary_dir() as d:
1174
+ task = create_task_fixture(
1175
+ ExtractData,
1176
+ {"database_url": f"sqlite:///{d}/temp.db", "mapping": mapping_path},
1177
+ )
1178
+
1179
+ extracted_records = {
1180
+ # ['Name', 'Description', 'BillingStreet', 'BillingCity', 'BillingState', 'BillingPostalCode', 'BillingCountry', 'ParentId', 'ShippingStreet', 'ShippingCity', 'ShippingState', 'ShippingPostalCode', 'ShippingCountry', 'Phone', 'Fax', 'Website', 'NumberOfEmployees', 'AccountNumber', 'Site', 'Type', 'IsPersonAccount']
1181
+ "Account": [
1182
+ ["001002", "Account1"] + [""] * 20 + [False],
1183
+ ["001003", "Account2"] + [""] * 20 + [False],
1184
+ ],
1185
+ # Salutation|FirstName|LastName|Email|Phone|MobilePhone|OtherPhone|HomePhone|Title|Birthdate|MailingStreet|MailingCity|MailingState|MailingPostalCode|MailingCountry|AccountId
1186
+ "Contact": [
1187
+ ["002001", "", "Bob 1", "Barker 2"] + [""] * 4,
1188
+ ["002002", "", "Sam 2", "Smith 2"] + [""] * 4,
1189
+ ], # id|Name|StageName|CloseDate|Amount|AccountId|ContactId|record_type
1190
+ "Opportunity": [
1191
+ [
1192
+ "0003001",
1193
+ "Dickenson Mobile Generators",
1194
+ "Qualification",
1195
+ "2020-07-18",
1196
+ "15000.0",
1197
+ "001003",
1198
+ "002001",
1199
+ ]
1200
+ ],
1201
+ }
1202
+ with mock_extract_jobs(task, extracted_records), mock_salesforce_client(
1203
+ task
1204
+ ):
1205
+ task()
1206
+ with create_engine(task.options["database_url"]).connect() as conn:
1207
+ output_accounts = list(conn.execute("select * from Account"))
1208
+ assert output_accounts[0][0:2] == ("Account-1", "Account1")
1209
+ assert output_accounts[1][0:2] == ("Account-2", "Account2")
1210
+ assert len(output_accounts) == 2
1211
+ output_opportunities = list(conn.execute("select * from Opportunity"))
1212
+
1213
+ assert output_opportunities[0].AccountId == "Account-2"
1214
+ assert output_opportunities[0].ContactId == "Contact-1"
1215
+
1216
+ def test_run_soql_filter(self):
1217
+ """This test case is to verify when soql_filter is specified with valid filter in the mapping yml"""
1218
+
1219
+ base_path = os.path.dirname(__file__)
1220
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
1221
+ mock_describe_calls()
1222
+ with temporary_dir() as d:
1223
+ tmp_db_path = os.path.join(d, "testdata.db")
1224
+
1225
+ task = _make_task(
1226
+ ExtractData,
1227
+ {
1228
+ "options": {
1229
+ "database_url": f"sqlite:///{tmp_db_path}",
1230
+ "mapping": mapping_path,
1231
+ }
1232
+ },
1233
+ )
1234
+
1235
+ mapping = MappingStep(
1236
+ sf_object="Contact",
1237
+ fields={"Id": "Id", "Name": "Name"},
1238
+ record_type="Business",
1239
+ soql_filter="Name = 'John Doe'",
1240
+ )
1241
+
1242
+ soql = task._soql_for_mapping(mapping)
1243
+ assert (
1244
+ "WHERE RecordType.DeveloperName = 'Business' AND Name = 'John Doe'"
1245
+ in soql
1246
+ ), "Filter should be applied on Name and DeveloperName"
1247
+
1248
+ def test_run_soql_filter_where_specified(self):
1249
+ """This test case is to verify when soql_filter is specified with mapping yml with WHERE keyword in it"""
1250
+
1251
+ base_path = os.path.dirname(__file__)
1252
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
1253
+ mock_describe_calls()
1254
+ with temporary_dir() as d:
1255
+ tmp_db_path = os.path.join(d, "testdata.db")
1256
+
1257
+ task = _make_task(
1258
+ ExtractData,
1259
+ {
1260
+ "options": {
1261
+ "database_url": f"sqlite:///{tmp_db_path}",
1262
+ "mapping": mapping_path,
1263
+ }
1264
+ },
1265
+ )
1266
+
1267
+ mapping = MappingStep(
1268
+ sf_object="Contact",
1269
+ fields={"Id": "Id", "Name": "Name"},
1270
+ record_type="Business",
1271
+ soql_filter=" wHeRe Name = 'John Doe'",
1272
+ )
1273
+
1274
+ soql = task._soql_for_mapping(mapping)
1275
+ assert (
1276
+ "WHERE RecordType.DeveloperName = 'Business' AND Name = 'John Doe'"
1277
+ in soql
1278
+ ), "Filter should be applied on Name and RecordType"
1279
+
1280
+ def test_run_soql_filter_no_record_type(self):
1281
+ """This test case is to verify when soql_filter is specified in mapping yml file but no record_type"""
1282
+
1283
+ base_path = os.path.dirname(__file__)
1284
+ mapping_path = os.path.join(base_path, self.mapping_file_v1)
1285
+ mock_describe_calls()
1286
+ with temporary_dir() as d:
1287
+ tmp_db_path = os.path.join(d, "testdata.db")
1288
+
1289
+ task = _make_task(
1290
+ ExtractData,
1291
+ {
1292
+ "options": {
1293
+ "database_url": f"sqlite:///{tmp_db_path}",
1294
+ "mapping": mapping_path,
1295
+ }
1296
+ },
1297
+ )
1298
+
1299
+ mapping = MappingStep(
1300
+ sf_object="Contact",
1301
+ fields={"Id": "Id", "Name": "Name"},
1302
+ soql_filter=" wHeRe Name = 'John Doe'",
1303
+ )
1304
+
1305
+ soql = task._soql_for_mapping(mapping)
1306
+ assert (
1307
+ "WHERE Name = 'John Doe'" in soql
1308
+ ), "filter should be applied just on name"
1309
+ assert (
1310
+ "DeveloperName" not in soql
1311
+ ), "DeveloperName should not appear in the soql query as it is missing in mapping"