decidim-core 0.29.1 → 0.30.0.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (517) hide show
  1. checksums.yaml +4 -4
  2. data/app/cells/decidim/activity_cell.rb +0 -3
  3. data/app/cells/decidim/amendable/amend_button_card/show.erb +6 -4
  4. data/app/cells/decidim/amendable/amend_button_card_cell.rb +8 -0
  5. data/app/cells/decidim/amendable/amendments/show.erb +1 -1
  6. data/app/cells/decidim/amendable/promote_button_card/show.erb +2 -0
  7. data/app/cells/decidim/author/show.erb +5 -4
  8. data/app/cells/decidim/author_cell.rb +27 -1
  9. data/app/cells/decidim/card_l/extra_data.erb +1 -0
  10. data/app/cells/decidim/card_metadata_cell.rb +9 -16
  11. data/app/cells/decidim/card_s/show.erb +5 -3
  12. data/app/cells/decidim/comments_button_cell.rb +14 -2
  13. data/app/cells/decidim/content_blocks/highlighted_elements_cell.rb +2 -0
  14. data/app/cells/decidim/content_blocks/highlighted_elements_with_cell_for_list_cell.rb +1 -1
  15. data/app/cells/decidim/diff_cell.rb +4 -0
  16. data/app/cells/decidim/endorsement_block/show.erb +0 -1
  17. data/app/cells/decidim/endorsement_block_cell.rb +6 -0
  18. data/app/cells/decidim/endorsement_buttons/button_content.erb +1 -1
  19. data/app/cells/decidim/endorsement_buttons/select_identity_button.erb +1 -1
  20. data/app/cells/decidim/endorsement_buttons/show.erb +7 -7
  21. data/app/cells/decidim/endorsement_buttons/verification_modal.erb +4 -2
  22. data/app/cells/decidim/endorsers_list/empty.erb +3 -0
  23. data/app/cells/decidim/endorsers_list/full.erb +17 -5
  24. data/app/cells/decidim/endorsers_list/show.erb +19 -10
  25. data/app/cells/decidim/endorsers_list_cell.rb +19 -6
  26. data/app/cells/decidim/follow_button/show.erb +1 -2
  27. data/app/cells/decidim/follow_button_cell.rb +5 -0
  28. data/app/cells/decidim/map/show.erb +3 -0
  29. data/app/cells/decidim/map/template.erb +14 -0
  30. data/app/cells/decidim/map_cell.rb +39 -0
  31. data/app/cells/decidim/newsletter_templates/image_text_cta_cell.rb +1 -1
  32. data/app/cells/decidim/notification_actions/buttons_cell.rb +1 -1
  33. data/app/cells/decidim/onboarding_action_message/show.erb +15 -0
  34. data/app/cells/decidim/onboarding_action_message_cell.rb +81 -0
  35. data/app/cells/decidim/participatory_space_private_user/show.erb +23 -0
  36. data/app/cells/decidim/participatory_space_private_user_cell.rb +21 -0
  37. data/app/cells/decidim/progress_bar/show.erb +2 -2
  38. data/app/cells/decidim/report_button_cell.rb +1 -0
  39. data/app/cells/decidim/resource_history/show.erb +20 -0
  40. data/app/cells/decidim/resource_history_cell.rb +66 -0
  41. data/app/cells/decidim/share_button_cell.rb +0 -4
  42. data/app/cells/decidim/share_widget/modal.erb +26 -0
  43. data/app/cells/decidim/share_widget/show.erb +2 -0
  44. data/app/cells/decidim/share_widget_cell.rb +20 -0
  45. data/app/cells/decidim/tags/show.erb +7 -4
  46. data/app/cells/decidim/tags_cell.rb +11 -72
  47. data/app/cells/decidim/translation_bar/show.erb +2 -2
  48. data/app/cells/decidim/translation_bar_cell.rb +1 -1
  49. data/app/commands/decidim/amendable/create_draft.rb +2 -2
  50. data/app/commands/decidim/create_ephemeral_user.rb +52 -0
  51. data/app/commands/decidim/create_omniauth_registration.rb +13 -3
  52. data/app/commands/decidim/create_report.rb +1 -5
  53. data/app/commands/decidim/create_user_group.rb +6 -2
  54. data/app/commands/decidim/destroy_account.rb +3 -0
  55. data/app/commands/decidim/destroy_ephemeral_user.rb +47 -0
  56. data/app/commands/decidim/update_account.rb +11 -3
  57. data/app/commands/decidim/update_notifications_settings.rb +1 -0
  58. data/app/commands/decidim/update_resources_taxonomies.rb +67 -0
  59. data/app/commands/decidim/update_user_group.rb +7 -1
  60. data/app/constraints/decidim/current_component.rb +1 -1
  61. data/app/controllers/concerns/decidim/ajax_permission_handler.rb +21 -0
  62. data/app/controllers/concerns/decidim/devise_authentication_methods.rb +3 -6
  63. data/app/controllers/concerns/decidim/devise_controllers.rb +2 -0
  64. data/app/controllers/concerns/decidim/disable_redirection_to_external_host.rb +1 -1
  65. data/app/controllers/concerns/decidim/ephemeral_session_checker.rb +87 -0
  66. data/app/controllers/concerns/decidim/filter_resource.rb +4 -2
  67. data/app/controllers/concerns/decidim/has_members_page.rb +25 -0
  68. data/app/controllers/concerns/decidim/headers/browser_feature_permissions.rb +50 -0
  69. data/app/controllers/concerns/decidim/locale_switcher.rb +2 -2
  70. data/app/controllers/concerns/decidim/needs_password_change.rb +0 -1
  71. data/app/controllers/concerns/decidim/needs_permission.rb +2 -1
  72. data/app/controllers/concerns/decidim/needs_tos_accepted.rb +4 -5
  73. data/app/controllers/concerns/decidim/onboarding_action_methods.rb +52 -0
  74. data/app/controllers/decidim/application_controller.rb +10 -0
  75. data/app/controllers/decidim/authorization_modals_controller.rb +8 -2
  76. data/app/controllers/decidim/components/base_controller.rb +3 -7
  77. data/app/controllers/decidim/devise/confirmations_controller.rb +4 -0
  78. data/app/controllers/decidim/devise/omniauth_registrations_controller.rb +8 -1
  79. data/app/controllers/decidim/devise/registrations_controller.rb +1 -1
  80. data/app/controllers/decidim/devise/sessions_controller.rb +2 -0
  81. data/app/controllers/decidim/doorkeeper/credentials_controller.rb +1 -1
  82. data/app/controllers/decidim/download_your_data_controller.rb +33 -2
  83. data/app/controllers/decidim/editor_images_controller.rb +1 -11
  84. data/app/controllers/decidim/follows_controller.rb +1 -1
  85. data/app/controllers/decidim/geolocation_controller.rb +19 -0
  86. data/app/controllers/decidim/homepage_controller.rb +0 -1
  87. data/app/controllers/decidim/links_controller.rb +1 -1
  88. data/app/controllers/decidim/open_data_controller.rb +33 -8
  89. data/app/controllers/decidim/profiles_controller.rb +4 -0
  90. data/app/controllers/decidim/user_conversations_controller.rb +1 -11
  91. data/app/events/decidim/soft_delete_resource_event.rb +15 -0
  92. data/app/forms/decidim/amendable/form.rb +4 -2
  93. data/app/forms/decidim/ephemeral_user_form.rb +22 -0
  94. data/app/forms/decidim/notifications_settings_form.rb +1 -0
  95. data/app/forms/decidim/omniauth_registration_form.rb +14 -1
  96. data/app/helpers/concerns/decidim/flash_helper_extensions.rb +17 -2
  97. data/app/helpers/concerns/decidim/user_role_checker.rb +46 -0
  98. data/app/helpers/decidim/action_authorization_helper.rb +75 -14
  99. data/app/helpers/decidim/amendments_helper.rb +0 -9
  100. data/app/helpers/decidim/application_helper.rb +6 -2
  101. data/app/helpers/decidim/breadcrumb_helper.rb +6 -0
  102. data/app/helpers/decidim/cache_helper.rb +1 -1
  103. data/app/helpers/decidim/check_boxes_tree_helper.rb +25 -53
  104. data/app/helpers/decidim/cta_button_helper.rb +1 -1
  105. data/app/helpers/decidim/decidim_form_helper.rb +0 -44
  106. data/app/helpers/decidim/endorsable_helper.rb +0 -10
  107. data/app/helpers/decidim/layout_helper.rb +6 -0
  108. data/app/helpers/decidim/map_helper.rb +6 -1
  109. data/app/helpers/decidim/messaging/conversation_helper.rb +2 -3
  110. data/app/helpers/decidim/meta_tags_helper.rb +32 -23
  111. data/app/helpers/decidim/modal_helper.rb +23 -0
  112. data/app/helpers/decidim/omniauth_helper.rb +5 -5
  113. data/app/helpers/decidim/onboarding_action_helper.rb +13 -0
  114. data/app/helpers/decidim/passwords_helper.rb +2 -2
  115. data/app/helpers/decidim/sanitize_helper.rb +11 -2
  116. data/app/helpers/decidim/taxonomies_helper.rb +40 -0
  117. data/app/jobs/decidim/download_your_data_export_job.rb +2 -19
  118. data/app/jobs/decidim/event_publisher_job.rb +18 -9
  119. data/app/jobs/decidim/export_job.rb +5 -1
  120. data/app/jobs/decidim/export_participatory_space_job.rb +4 -1
  121. data/app/jobs/decidim/open_data_job.rb +4 -4
  122. data/app/mailers/decidim/export_mailer.rb +7 -12
  123. data/app/models/decidim/action_log.rb +11 -0
  124. data/app/models/decidim/attachment.rb +1 -1
  125. data/app/models/decidim/category.rb +4 -0
  126. data/app/models/decidim/component.rb +25 -3
  127. data/app/models/decidim/metric.rb +2 -1
  128. data/app/models/decidim/moderation.rb +16 -0
  129. data/app/models/decidim/newsletter.rb +12 -8
  130. data/app/models/decidim/organization.rb +7 -3
  131. data/app/models/decidim/participatory_space_private_user.rb +20 -0
  132. data/app/models/decidim/private_export.rb +24 -0
  133. data/app/models/decidim/push_notification_message.rb +1 -1
  134. data/app/models/decidim/scope.rb +4 -0
  135. data/app/models/decidim/share_token.rb +36 -10
  136. data/app/models/decidim/taxonomization.rb +23 -0
  137. data/app/models/decidim/taxonomy.rb +139 -0
  138. data/app/models/decidim/taxonomy_filter.rb +130 -0
  139. data/app/models/decidim/taxonomy_filter_item.rb +33 -0
  140. data/app/models/decidim/user.rb +9 -23
  141. data/app/models/decidim/user_base_entity.rb +12 -0
  142. data/app/models/decidim/user_moderation.rb +9 -0
  143. data/app/models/decidim/user_report.rb +4 -0
  144. data/app/packs/entrypoints/decidim_core.js +1 -0
  145. data/app/packs/entrypoints/decidim_geocoding.js +2 -0
  146. data/app/packs/images/decidim/brands/facebook.svg +10 -0
  147. data/app/packs/images/decidim/brands/google.svg +7 -1
  148. data/app/packs/images/decidim/brands/twitter-x.svg +3 -0
  149. data/app/packs/src/decidim/append_redirect_url_to_modals.js +14 -6
  150. data/app/packs/src/decidim/check_boxes_tree.js +0 -1
  151. data/app/packs/src/decidim/clipboard.js +18 -9
  152. data/app/packs/src/decidim/confirm.js +79 -59
  153. data/app/packs/src/decidim/decidim_application.js +1 -0
  154. data/app/packs/src/decidim/direct_uploads/upload_field.js +21 -8
  155. data/app/packs/src/decidim/form_remote.js +1 -1
  156. data/app/packs/src/decidim/geocoding/reverse_geocoding.js +60 -0
  157. data/app/packs/src/decidim/impersonation.js +1 -1
  158. data/app/packs/src/decidim/index.js +14 -1
  159. data/app/packs/src/decidim/onboarding_pending_action.js +24 -0
  160. data/app/packs/src/decidim/remote_tooltips.js +38 -0
  161. data/app/packs/src/decidim/session_timeouter.js +1 -1
  162. data/app/packs/src/decidim/sticky_footer.js +29 -0
  163. data/app/packs/src/decidim/sticky_header.js +6 -31
  164. data/app/packs/src/decidim/toggle.js +1 -1
  165. data/app/packs/src/decidim/tooltips.js +42 -22
  166. data/app/packs/src/decidim/user_registrations.js +13 -0
  167. data/app/packs/src/decidim/utilities/dom.js +148 -0
  168. data/app/packs/stylesheets/decidim/_accordion.scss +30 -0
  169. data/app/packs/stylesheets/decidim/_buttons.scss +19 -0
  170. data/app/packs/stylesheets/decidim/_dropdown.scss +1 -1
  171. data/app/packs/stylesheets/decidim/_endorsers_list.scss +26 -20
  172. data/app/packs/stylesheets/decidim/_forms.scss +2 -2
  173. data/app/packs/stylesheets/decidim/_header.scss +5 -3
  174. data/app/packs/stylesheets/decidim/_labels.scss +1 -1
  175. data/app/packs/stylesheets/decidim/_layout.scss +32 -2
  176. data/app/packs/stylesheets/decidim/_login.scss +53 -6
  177. data/app/packs/stylesheets/decidim/_modal.scss +42 -0
  178. data/app/packs/stylesheets/decidim/_modal_tos_refuse.scss +4 -0
  179. data/app/packs/stylesheets/decidim/_participatory_spaces.scss +46 -0
  180. data/app/packs/stylesheets/decidim/_profile.scss +2 -2
  181. data/app/packs/stylesheets/decidim/_progress-bar.scss +1 -1
  182. data/app/packs/stylesheets/decidim/_success_image.scss +64 -0
  183. data/app/packs/stylesheets/decidim/_tribute.scss +36 -0
  184. data/app/packs/stylesheets/decidim/application.scss +1 -0
  185. data/app/packs/stylesheets/decidim/geocoding_addons.scss +5 -0
  186. data/app/packs/stylesheets/decidim/legacy/conference-diploma.scss +2 -1
  187. data/app/packs/stylesheets/decidim/map.scss +7 -10
  188. data/app/packs/stylesheets/decidim/resource_history.scss +31 -0
  189. data/app/permissions/decidim/permissions.rb +8 -2
  190. data/app/presenters/decidim/admin_log/base_user_presenter.rb +67 -0
  191. data/app/presenters/decidim/admin_log/component_presenter.rb +32 -3
  192. data/app/presenters/decidim/admin_log/moderation_presenter.rb +30 -3
  193. data/app/presenters/decidim/admin_log/share_token_presenter.rb +39 -0
  194. data/app/presenters/decidim/admin_log/taxonomy_filter_presenter.rb +57 -0
  195. data/app/presenters/decidim/admin_log/taxonomy_presenter.rb +48 -0
  196. data/app/presenters/decidim/admin_log/user_group_presenter.rb +6 -6
  197. data/app/presenters/decidim/admin_log/user_presenter.rb +4 -18
  198. data/app/presenters/decidim/attachment_presenter.rb +1 -1
  199. data/app/presenters/decidim/log/value_types/date_presenter.rb +1 -0
  200. data/app/presenters/decidim/log/value_types/taxonomy_presenter.rb +29 -0
  201. data/app/presenters/decidim/participatory_space_private_user_presenter.rb +50 -0
  202. data/app/presenters/decidim/taxonomy_filter_presenter.rb +19 -0
  203. data/app/presenters/decidim/taxonomy_presenter.rb +14 -0
  204. data/app/queries/decidim/metrics/blocked_users_metric_manage.rb +2 -2
  205. data/app/queries/decidim/metrics/users_metric_manage.rb +2 -2
  206. data/app/queries/decidim/public_activities.rb +1 -12
  207. data/app/queries/decidim/stats_users_count.rb +2 -2
  208. data/app/resolvers/decidim/meta_image_url_resolver.rb +128 -0
  209. data/app/serializers/decidim/exporters/open_data_blocked_user_serializer.rb +25 -0
  210. data/app/serializers/decidim/exporters/open_data_metric_serializer.rb +22 -0
  211. data/app/serializers/decidim/exporters/open_data_moderation_serializer.rb +30 -0
  212. data/app/serializers/decidim/exporters/open_data_taxonomy_serializer.rb +30 -0
  213. data/app/serializers/decidim/exporters/open_data_user_group_serializer.rb +38 -0
  214. data/app/serializers/decidim/exporters/open_data_user_serializer.rb +39 -0
  215. data/app/serializers/decidim/exporters/participatory_space_serializer.rb +104 -0
  216. data/app/serializers/decidim/exporters/serializer.rb +25 -0
  217. data/app/serializers/decidim/schema_org_breadcrumb_list_serializer.rb +52 -0
  218. data/app/services/decidim/action_authorizer.rb +32 -3
  219. data/app/services/decidim/base_diff_renderer.rb +27 -3
  220. data/app/services/decidim/download_your_data_exporter.rb +58 -24
  221. data/app/services/decidim/email_notification_generator.rb +14 -5
  222. data/app/services/decidim/onboarding_manager.rb +272 -0
  223. data/app/services/decidim/open_data_exporter.rb +139 -13
  224. data/app/services/decidim/traceability.rb +13 -0
  225. data/app/uploaders/decidim/application_uploader.rb +1 -1
  226. data/app/validators/etiquette_validator.rb +9 -2
  227. data/app/validators/passthru_validator.rb +1 -1
  228. data/app/validators/translated_etiquette_validator.rb +32 -0
  229. data/app/views/decidim/account/_password_fields.html.erb +2 -2
  230. data/app/views/decidim/application/_accordion_section.html.erb +14 -0
  231. data/app/views/decidim/application/_collection.html.erb +30 -18
  232. data/app/views/decidim/devise/omniauth_registrations/new.html.erb +5 -1
  233. data/app/views/decidim/devise/omniauth_registrations/new_tos_fields.html.erb +29 -0
  234. data/app/views/decidim/devise/registrations/new.html.erb +10 -22
  235. data/app/views/decidim/devise/sessions/new.html.erb +24 -29
  236. data/app/views/decidim/devise/shared/_omniauth_buttons.html.erb +2 -4
  237. data/app/views/decidim/devise/shared/_tos_fields.html.erb +16 -0
  238. data/app/views/decidim/download_your_data/_export.html.erb +15 -0
  239. data/app/views/decidim/download_your_data/show.html.erb +51 -5
  240. data/app/views/decidim/endorsements/update_buttons_and_counters.js.erb +16 -19
  241. data/app/views/decidim/export_mailer/download_your_data_export.html.erb +2 -2
  242. data/app/views/decidim/export_mailer/export.html.erb +5 -1
  243. data/app/views/decidim/notifications_settings/show.html.erb +22 -0
  244. data/app/views/decidim/open_data/_how_to_open_accordion_section.html.erb +15 -0
  245. data/app/views/decidim/open_data/index.html.erb +108 -0
  246. data/app/views/decidim/pages/_tabbed.html.erb +2 -2
  247. data/app/views/decidim/pages/show.html.erb +2 -1
  248. data/app/views/decidim/participatory_space_private_users/_participatory_space_private_user.html.erb +1 -0
  249. data/app/views/decidim/shared/_login_modal.html.erb +26 -20
  250. data/app/views/decidim/shared/_resource_actions.html.erb +21 -0
  251. data/app/views/decidim/shared/filters/_check_boxes_tree.html.erb +1 -2
  252. data/app/views/decidim/shared/filters/_collection.html.erb +1 -1
  253. data/app/views/decidim/shared/filters/_dropdown_label.html.erb +7 -4
  254. data/app/views/layouts/decidim/_application.html.erb +0 -1
  255. data/app/views/layouts/decidim/_js_configuration.html.erb +1 -0
  256. data/app/views/layouts/decidim/_logo.html.erb +1 -1
  257. data/app/views/layouts/decidim/_logo_mobile.html.erb +1 -1
  258. data/app/views/layouts/decidim/_meta_tags_config.html.erb +6 -11
  259. data/app/views/layouts/decidim/_schema_org_breadcrumb_list.html.erb +3 -0
  260. data/app/views/layouts/decidim/_wrapper.html.erb +15 -9
  261. data/app/views/layouts/decidim/footer/_main_links.html.erb +1 -1
  262. data/app/views/layouts/decidim/header/_close_ephemeral_session.html.erb +25 -0
  263. data/app/views/layouts/decidim/header/_main.html.erb +14 -10
  264. data/app/views/layouts/decidim/header/_menu_breadcrumb_items.html.erb +2 -0
  265. data/app/views/layouts/decidim/header/_menu_breadcrumb_mobile_tablet.html.erb +1 -1
  266. data/app/views/layouts/decidim/shared/_layout_item.html.erb +1 -1
  267. data/config/assets.rb +2 -1
  268. data/config/locales/ar.yml +38 -54
  269. data/config/locales/bg.yml +3 -64
  270. data/config/locales/bn-BD.yml +1 -0
  271. data/config/locales/bs-BA.yml +98 -0
  272. data/config/locales/ca.yml +380 -69
  273. data/config/locales/cs.yml +369 -55
  274. data/config/locales/de.yml +371 -67
  275. data/config/locales/el.yml +5 -45
  276. data/config/locales/en.yml +365 -54
  277. data/config/locales/eo.yml +0 -2
  278. data/config/locales/es-MX.yml +366 -58
  279. data/config/locales/es-PY.yml +366 -58
  280. data/config/locales/es.yml +375 -64
  281. data/config/locales/eu.yml +547 -223
  282. data/config/locales/fi-plain.yml +361 -57
  283. data/config/locales/fi.yml +400 -93
  284. data/config/locales/fr-CA.yml +181 -55
  285. data/config/locales/fr.yml +182 -53
  286. data/config/locales/ga-IE.yml +4 -16
  287. data/config/locales/gl.yml +5 -30
  288. data/config/locales/hu.yml +3 -55
  289. data/config/locales/id-ID.yml +5 -30
  290. data/config/locales/is-IS.yml +4 -13
  291. data/config/locales/it.yml +10 -46
  292. data/config/locales/ja.yml +230 -69
  293. data/config/locales/lb.yml +5 -47
  294. data/config/locales/lt.yml +5 -59
  295. data/config/locales/lv.yml +5 -36
  296. data/config/locales/nl.yml +5 -48
  297. data/config/locales/no.yml +6 -47
  298. data/config/locales/pl.yml +4 -66
  299. data/config/locales/pt-BR.yml +12 -52
  300. data/config/locales/pt.yml +11 -47
  301. data/config/locales/ro-RO.yml +448 -247
  302. data/config/locales/ru.yml +4 -16
  303. data/config/locales/sk.yml +5 -37
  304. data/config/locales/sv.yml +108 -64
  305. data/config/locales/tr-TR.yml +5 -45
  306. data/config/locales/uk.yml +4 -15
  307. data/config/locales/zh-CN.yml +5 -45
  308. data/config/locales/zh-TW.yml +5 -50
  309. data/config/routes.rb +5 -10
  310. data/db/migrate/20181025082245_add_timestamps_to_components.rb +2 -0
  311. data/db/migrate/20240628111905_add_visible_to_components.rb +7 -0
  312. data/db/migrate/20240704115429_create_decidim_taxonomies.rb +23 -0
  313. data/db/migrate/20240717093514_add_registered_only_to_decidim_share_tokens.rb +7 -0
  314. data/db/migrate/20240722215500_change_object_changes_on_versions.rb +39 -0
  315. data/db/migrate/20240806065417_create_decidim_taxonomy_filters.rb +23 -0
  316. data/db/migrate/20240822161348_add_deleted_at_to_decidim_components.rb +8 -0
  317. data/db/migrate/20240906162524_add_part_of_to_taxonomies.rb +26 -0
  318. data/db/migrate/20241001135825_add_role_to_participatory_space_private_users.rb +7 -0
  319. data/db/migrate/20241001140408_add_published_to_participatory_space_private_users.rb +7 -0
  320. data/db/migrate/20241022002600_create_private_exports.rb +18 -0
  321. data/db/migrate/20241028114430_add_component_counter_to_taxonomy_filters.rb +7 -0
  322. data/db/migrate/20241111104357_add_names_and_space_boolean_to_taxonomy_filters.rb +9 -0
  323. data/db/migrate/20241127093708_add_taxonomy_to_metrics.rb +7 -0
  324. data/db/migrate/20241204121445_add_participatory_space_manifests_from_taxonomy_filters.rb +9 -0
  325. data/decidim-core.gemspec +18 -18
  326. data/lib/decidim/action_authorization.rb +3 -2
  327. data/lib/decidim/amendable.rb +1 -1
  328. data/lib/decidim/api/functions/component_list.rb +1 -1
  329. data/lib/decidim/api/functions/participatory_space_finder_base.rb +11 -1
  330. data/lib/decidim/api/input_filters/category_input_filter.rb +5 -3
  331. data/lib/decidim/api/input_filters/component_input_filter.rb +33 -20
  332. data/lib/decidim/api/input_filters/user_entity_input_filter.rb +53 -44
  333. data/lib/decidim/api/input_sorts/component_input_sort.rb +15 -13
  334. data/lib/decidim/api/input_sorts/user_entity_input_sort.rb +4 -4
  335. data/lib/decidim/api/interfaces/author_interface.rb +3 -2
  336. data/lib/decidim/api/interfaces/coauthorable_interface.rb +5 -5
  337. data/lib/decidim/api/interfaces/endorsable_interface.rb +2 -2
  338. data/lib/decidim/api/interfaces/participatory_space_interface.rb +5 -10
  339. data/lib/decidim/api/interfaces/taxonomizable_interface.rb +13 -0
  340. data/lib/decidim/api/types/amendment_type.rb +5 -7
  341. data/lib/decidim/api/types/area_api_type.rb +3 -3
  342. data/lib/decidim/api/types/attachment_type.rb +3 -3
  343. data/lib/decidim/api/types/category_type.rb +2 -2
  344. data/lib/decidim/api/types/component_type.rb +7 -0
  345. data/lib/decidim/api/types/decidim_type.rb +5 -1
  346. data/lib/decidim/api/types/fingerprint_type.rb +1 -1
  347. data/lib/decidim/api/types/localized_string_type.rb +1 -1
  348. data/lib/decidim/api/types/metric_history_type.rb +2 -0
  349. data/lib/decidim/api/types/metric_type.rb +1 -1
  350. data/lib/decidim/api/types/organization_type.rb +5 -0
  351. data/lib/decidim/api/types/participatory_space_link_type.rb +2 -2
  352. data/lib/decidim/api/types/participatory_space_manifest_type.rb +1 -1
  353. data/lib/decidim/api/types/quantifiable_translated_field_type.rb +1 -1
  354. data/lib/decidim/api/types/scope_api_type.rb +2 -3
  355. data/lib/decidim/api/types/session_type.rb +1 -2
  356. data/lib/decidim/api/types/taxonomy_type.rb +15 -0
  357. data/lib/decidim/api/types/trace_version_type.rb +2 -2
  358. data/lib/decidim/api/types/translated_field_type.rb +3 -5
  359. data/lib/decidim/api/types/user_group_type.rb +11 -20
  360. data/lib/decidim/api/types/user_type.rb +9 -18
  361. data/lib/decidim/asset_router/storage.rb +4 -4
  362. data/lib/decidim/assets/tailwind/tailwind.config.js.erb +2 -1
  363. data/lib/decidim/attachment_attributes.rb +1 -1
  364. data/lib/decidim/attribute_encryptor.rb +1 -1
  365. data/lib/decidim/attribute_object/nested_validator.rb +1 -1
  366. data/lib/decidim/attributes/integer_with_units.rb +27 -0
  367. data/lib/decidim/attributes/rich_text.rb +38 -0
  368. data/lib/decidim/attributes/time_with_zone.rb +11 -1
  369. data/lib/decidim/attributes.rb +4 -0
  370. data/lib/decidim/authorization_form_builder.rb +0 -7
  371. data/lib/decidim/coauthorable.rb +1 -1
  372. data/lib/decidim/command.rb +3 -3
  373. data/lib/decidim/commands/destroy_resource.rb +5 -1
  374. data/lib/decidim/commands/resource_handler.rb +2 -2
  375. data/lib/decidim/commands/restore_resource.rb +45 -0
  376. data/lib/decidim/commands/soft_delete_resource.rb +63 -0
  377. data/lib/decidim/component_manifest.rb +7 -0
  378. data/lib/decidim/content_parsers/blob_parser.rb +93 -0
  379. data/lib/decidim/content_parsers.rb +1 -0
  380. data/lib/decidim/content_renderers/blob_renderer.rb +90 -0
  381. data/lib/decidim/content_renderers.rb +1 -0
  382. data/lib/decidim/core/api.rb +2 -0
  383. data/lib/decidim/core/engine.rb +50 -8
  384. data/lib/decidim/core/menu.rb +0 -5
  385. data/lib/decidim/core/seeds.rb +42 -2
  386. data/lib/decidim/core/test/factories.rb +133 -2
  387. data/lib/decidim/core/test/shared_examples/admin_resource_gallery_examples.rb +85 -0
  388. data/lib/decidim/core/test/shared_examples/authorable_interface_examples.rb +1 -1
  389. data/lib/decidim/core/test/shared_examples/comments_examples.rb +408 -40
  390. data/lib/decidim/core/test/shared_examples/components_controller_hide_shared_examples.rb +11 -0
  391. data/lib/decidim/core/test/shared_examples/components_controller_reorder_shared_examples.rb +21 -0
  392. data/lib/decidim/core/test/shared_examples/download_open_data_shared_context.rb +36 -0
  393. data/lib/decidim/core/test/shared_examples/download_open_data_shared_examples.rb +45 -0
  394. data/lib/decidim/core/test/shared_examples/download_your_data_shared_examples.rb +17 -0
  395. data/lib/decidim/core/test/shared_examples/etiquette_validator_examples.rb +25 -0
  396. data/lib/decidim/core/test/shared_examples/follows_examples.rb +43 -0
  397. data/lib/decidim/core/test/shared_examples/has_taxonomies.rb +94 -0
  398. data/lib/decidim/core/test/shared_examples/manage_share_tokens_examples.rb +237 -0
  399. data/lib/decidim/core/test/shared_examples/map_examples.rb +2 -0
  400. data/lib/decidim/core/test/shared_examples/open_data_exporter_examples.rb +102 -0
  401. data/lib/decidim/core/test/shared_examples/participatory_space_members_page_examples.rb +46 -0
  402. data/lib/decidim/core/test/shared_examples/participatory_space_search_examples.rb +98 -0
  403. data/lib/decidim/core/test/shared_examples/permissions.rb +7 -7
  404. data/lib/decidim/core/test/shared_examples/preview_with_share_token_examples.rb +94 -0
  405. data/lib/decidim/core/test/shared_examples/reports_examples.rb +9 -5
  406. data/lib/decidim/core/test/shared_examples/resource_endorsed_event_examples.rb +2 -2
  407. data/lib/decidim/core/test/shared_examples/resource_search_examples.rb +75 -69
  408. data/lib/decidim/core/test/shared_examples/searchable_resources_shared_context.rb +1 -1
  409. data/lib/decidim/core/test/shared_examples/simple_event.rb +33 -19
  410. data/lib/decidim/core/test/shared_examples/social_share_examples.rb +25 -0
  411. data/lib/decidim/core/test/shared_examples/softdeleteable_components_examples.rb +173 -0
  412. data/lib/decidim/core/test/shared_examples/system_endorse_resource_examples.rb +54 -8
  413. data/lib/decidim/core/test/shared_examples/taxonomizable_interface_examples.rb +20 -0
  414. data/lib/decidim/core/test/shared_examples/taxonomizable_resource_examples.rb +37 -0
  415. data/lib/decidim/core/test/shared_examples/taxonomy_settings.rb +49 -0
  416. data/lib/decidim/core/test/shared_examples/translated_event_examples.rb +1 -1
  417. data/lib/decidim/core/test/shared_examples/versions_controller_examples.rb +2 -2
  418. data/lib/decidim/core/test/shared_examples/with_endorsable_permissions_examples.rb +1 -1
  419. data/lib/decidim/core/test.rb +9 -5
  420. data/lib/decidim/core/version.rb +1 -1
  421. data/lib/decidim/core.rb +58 -0
  422. data/lib/decidim/diffy_extension.rb +18 -0
  423. data/lib/decidim/download_your_data_serializers/download_your_data_identity_serializer.rb +0 -4
  424. data/lib/decidim/download_your_data_serializers/download_your_data_participatory_space_private_user_serializer.rb +3 -1
  425. data/lib/decidim/download_your_data_serializers/download_your_data_report_serializer.rb +2 -1
  426. data/lib/decidim/download_your_data_serializers/download_your_data_user_serializer.rb +6 -7
  427. data/lib/decidim/download_your_data_serializers.rb +19 -0
  428. data/lib/decidim/exporters/csv.rb +6 -0
  429. data/lib/decidim/exporters/export_manifest.rb +14 -0
  430. data/lib/decidim/exporters/pdf.rb +26 -22
  431. data/lib/decidim/filter_form_builder.rb +12 -15
  432. data/lib/decidim/form_builder.rb +7 -130
  433. data/lib/decidim/has_category.rb +5 -1
  434. data/lib/decidim/has_private_users.rb +4 -0
  435. data/lib/decidim/has_taxonomy_settings.rb +47 -0
  436. data/lib/decidim/has_upload_validations.rb +1 -1
  437. data/lib/decidim/legacy_form_builder.rb +118 -0
  438. data/lib/decidim/maintenance/import_models/application_record.rb +93 -0
  439. data/lib/decidim/maintenance/import_models/area.rb +82 -0
  440. data/lib/decidim/maintenance/import_models/area_type.rb +12 -0
  441. data/lib/decidim/maintenance/import_models/assembly_type.rb +49 -0
  442. data/lib/decidim/maintenance/import_models/categorization.rb +14 -0
  443. data/lib/decidim/maintenance/import_models/category.rb +107 -0
  444. data/lib/decidim/maintenance/import_models/participatory_process_type.rb +21 -0
  445. data/lib/decidim/maintenance/import_models/scope.rb +163 -0
  446. data/lib/decidim/maintenance/import_models.rb +19 -0
  447. data/lib/decidim/maintenance/taxonomy_importer.rb +132 -0
  448. data/lib/decidim/maintenance/taxonomy_plan.rb +78 -0
  449. data/lib/decidim/maintenance.rb +12 -0
  450. data/lib/decidim/map/autocomplete.rb +53 -8
  451. data/lib/decidim/map/dynamic_map.rb +1 -1
  452. data/lib/decidim/map/provider/geocoding/here.rb +9 -3
  453. data/lib/decidim/map.rb +3 -6
  454. data/lib/decidim/moderation_tools.rb +14 -3
  455. data/lib/decidim/newsletter_encryptor.rb +1 -1
  456. data/lib/decidim/paddable.rb +1 -1
  457. data/lib/decidim/participable.rb +10 -10
  458. data/lib/decidim/participatory_space_user.rb +14 -0
  459. data/lib/decidim/private_download_helper.rb +15 -0
  460. data/lib/decidim/query_extensions.rb +0 -26
  461. data/lib/decidim/record_encryptor.rb +51 -46
  462. data/lib/decidim/resourceable.rb +16 -0
  463. data/lib/decidim/seeds.rb +18 -20
  464. data/lib/decidim/settings_manifest.rb +36 -1
  465. data/lib/decidim/soft_deletable.rb +17 -0
  466. data/lib/decidim/taxonomizable.rb +72 -0
  467. data/lib/decidim/translatable_attributes.rb +6 -1
  468. data/lib/decidim/view_model.rb +0 -1
  469. data/lib/tasks/decidim_download_your_data_tasks.rake +2 -4
  470. data/lib/tasks/decidim_procfile.rake +6 -0
  471. data/lib/tasks/decidim_tasks.rake +1 -0
  472. data/lib/tasks/decidim_taxonomies.rake +155 -0
  473. data/lib/tasks/upgrade/{decidim_fix_categorization.rake → clean.rake} +52 -19
  474. data/lib/tasks/upgrade/decidim_active_storage_migration_tasks.rake +1 -1
  475. data/lib/tasks/upgrade/migrations.rake +91 -0
  476. metadata +217 -103
  477. data/app/cells/decidim/amendable/amenders_list/show.erb +0 -0
  478. data/app/cells/decidim/content_blocks/base/content.erb +0 -0
  479. data/app/cells/decidim/endorsers_list/full_endorsers_list.erb +0 -3
  480. data/app/cells/decidim/endorsers_list_button/content.erb +0 -1
  481. data/app/cells/decidim/endorsers_list_button_cell.rb +0 -15
  482. data/app/cells/decidim/scopes_picker/scope_picker_prompt.erb +0 -3
  483. data/app/cells/decidim/scopes_picker/scope_picker_values.erb +0 -5
  484. data/app/cells/decidim/scopes_picker/show.erb +0 -14
  485. data/app/cells/decidim/scopes_picker_cell.rb +0 -92
  486. data/app/commands/decidim/update_user_interests.rb +0 -41
  487. data/app/controllers/decidim/scopes_controller.rb +0 -84
  488. data/app/controllers/decidim/user_interests_controller.rb +0 -30
  489. data/app/forms/decidim/user_interest_scope_form.rb +0 -25
  490. data/app/forms/decidim/user_interests_form.rb +0 -17
  491. data/app/helpers/decidim/categories_helper.rb +0 -26
  492. data/app/helpers/decidim/scopes_helper.rb +0 -105
  493. data/app/packs/images/decidim/vendor/leaflet/layers-2x.png +0 -0
  494. data/app/packs/images/decidim/vendor/leaflet/layers.png +0 -0
  495. data/app/packs/images/decidim/vendor/leaflet/marker-icon-2x.png +0 -0
  496. data/app/packs/images/decidim/vendor/leaflet/marker-icon.png +0 -0
  497. data/app/packs/images/decidim/vendor/leaflet/marker-shadow.png +0 -0
  498. data/app/packs/stylesheets/decidim/legacy/MarkerCluster.Default.scss +0 -67
  499. data/app/packs/stylesheets/decidim/legacy/MarkerCluster.scss +0 -18
  500. data/app/packs/stylesheets/decidim/legacy/leaflet.scss +0 -705
  501. data/app/validators/scope_belongs_to_component_validator.rb +0 -16
  502. data/app/views/decidim/account/_user_groups.html.erb +0 -0
  503. data/app/views/decidim/scopes/_scopes_picker_input.html.erb +0 -19
  504. data/app/views/decidim/scopes/picker.html.erb +0 -53
  505. data/app/views/decidim/shared/_comments.html.erb +0 -0
  506. data/app/views/decidim/shared/_share_modal.html.erb +0 -26
  507. data/app/views/decidim/shared/_tags.html.erb +0 -1
  508. data/app/views/decidim/user_interests/_areas.html.erb +0 -14
  509. data/app/views/decidim/user_interests/_scopes.html.erb +0 -22
  510. data/app/views/decidim/user_interests/show.html.erb +0 -22
  511. data/config/initializers/foundation_rails_helper.rb +0 -4
  512. data/lib/decidim/core/test/shared_examples/has_category.rb +0 -38
  513. data/lib/decidim/core/test/shared_examples/has_scope.rb +0 -11
  514. data/lib/decidim/core/test/shared_examples/manage_component_share_tokens.rb +0 -83
  515. data/lib/decidim/core/test/shared_examples/preview_component_with_share_token_examples.rb +0 -49
  516. data/lib/decidim/core/test/shared_examples/scope_helper_examples.rb +0 -93
  517. data/lib/decidim/seven_zip_wrapper.rb +0 -29
@@ -1,41 +1,52 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require "decidim/seven_zip_wrapper"
4
-
5
3
  module Decidim
6
- # Public: Generates a 7z(seven zip) file with data files ready to be persisted
4
+ # Public: Generates a zip archive with data files ready to be persisted
7
5
  # somewhere so users can download their data.
8
6
  #
9
- # In fact, the 7z file wraps a ZIP file which finally contains the data files.
10
7
  class DownloadYourDataExporter
11
8
  DEFAULT_EXPORT_FORMAT = "CSV"
12
9
  ZIP_FILE_NAME = "download-your-data.zip"
13
10
 
11
+ include Decidim::TranslatableAttributes
12
+
14
13
  # Public: Initializes the class.
15
14
  #
16
15
  # user - The user to export the data from.
17
- # path - The String path where to write the zip file.
18
- # password - The password to protect the zip file.
16
+ # name - The name of the export in private area
19
17
  # export_format - The format of the data files inside the zip file. (CSV by default)
20
- def initialize(user, path, password, export_format = DEFAULT_EXPORT_FORMAT)
18
+ def initialize(user, name, export_format = DEFAULT_EXPORT_FORMAT)
21
19
  @user = user
22
- @path = File.expand_path path
23
20
  @export_format = export_format
24
- @password = password
21
+ @name = name
25
22
  end
26
23
 
24
+ # i18n-tasks-use t("decidim.download_your_data.show.download_your_data")
27
25
  def export
28
- tmpdir = Dir.mktmpdir("temporary-download-your-data-dir")
29
- user_data, user_attachments = data_and_attachments_for_user
30
- save_user_data(tmpdir, user_data)
31
- save_user_attachments(tmpdir, user_attachments)
32
-
33
- SevenZipWrapper.compress_and_encrypt(filename: @path, password: @password, input_directory: tmpdir)
26
+ user_export = user.private_exports.build
27
+ user_export.export_type = name
28
+ user_export.file.attach(io: data, filename: "#{name}.zip", content_type: "application/zip")
29
+ user_export.expires_at = Decidim.download_your_data_expiry_time.from_now
30
+ user_export.metadata = {}
31
+ user_export.save!
32
+ user_export
34
33
  end
35
34
 
36
35
  private
37
36
 
38
- attr_reader :user, :export_format
37
+ attr_reader :user, :export_format, :name
38
+
39
+ def data
40
+ user_data, user_attachments = data_and_attachments_for_user
41
+ buffer = Zip::OutputStream.write_buffer do |out|
42
+ save_user_data(out, user_data)
43
+ save_user_attachments(out, user_attachments)
44
+ save_readme(out)
45
+ end
46
+
47
+ buffer.rewind
48
+ buffer
49
+ end
39
50
 
40
51
  def data_and_attachments_for_user
41
52
  export_data = []
@@ -43,7 +54,8 @@ module Decidim
43
54
 
44
55
  download_your_data_entities.each do |object|
45
56
  klass = Object.const_get(object)
46
- export_data << [klass.model_name.name.parameterize.pluralize, Exporters.find_exporter(export_format).new(klass.user_collection(user), klass.export_serializer).export]
57
+ exporter = Exporters.find_exporter(export_format).new(klass.user_collection(user), klass.export_serializer)
58
+ export_data << [klass.model_name.name.parameterize.pluralize, exporter.export]
47
59
  attachments = klass.download_your_data_images(user)
48
60
  export_attachments << [klass.model_name.name.parameterize.pluralize, attachments.flatten] unless attachments.nil?
49
61
  end
@@ -55,30 +67,52 @@ module Decidim
55
67
  @download_your_data_entities ||= DownloadYourDataSerializers.data_entities
56
68
  end
57
69
 
58
- def save_user_data(tmpdir, user_data)
70
+ def save_user_data(output, user_data)
59
71
  user_data.each do |entity, exporter_data|
60
72
  next if exporter_data.read == "\n"
61
73
 
62
- file_name = File.join(tmpdir, "#{entity}-#{exporter_data.filename}")
63
- File.write(file_name, exporter_data.read)
74
+ output.put_next_entry("#{entity}-#{exporter_data.filename}")
75
+ output.write exporter_data.read
64
76
  end
65
77
  end
66
78
 
67
- def save_user_attachments(tmpdir, user_attachments)
79
+ def save_user_attachments(output, user_attachments)
68
80
  user_attachments.each do |entity, attachment_block|
69
81
  attachment_block.each do |attachment|
70
82
  next unless attachment.attached?
71
83
 
72
84
  blobs = attachment.is_a?(ActiveStorage::Attached::One) ? [attachment.blob] : attachment.blobs
73
85
  blobs.each do |blob|
74
- Dir.mkdir(File.join(tmpdir, entity.parameterize))
75
- file_name = File.join(tmpdir, entity.parameterize, blob.filename.to_s)
76
86
  blob.open do |blob_file|
77
- File.write(file_name, blob_file.read.force_encoding("UTF-8"))
87
+ output.put_next_entry("#{entity.parameterize}/#{blob.filename}")
88
+ output.write blob_file.read.force_encoding("UTF-8")
78
89
  end
79
90
  end
80
91
  end
81
92
  end
82
93
  end
94
+
95
+ def save_readme(output)
96
+ output.put_next_entry("README.md")
97
+ output.write readme
98
+ end
99
+
100
+ def readme
101
+ readme_file = "# #{I18n.t("decidim.download_your_data.help.core.title", organization: translated_attribute(user.organization.name))}\n\n"
102
+ readme_file << "#{I18n.t("decidim.download_your_data.help.core.description", user_name: "#{user.name} (#{user.nickname})")}\n\n"
103
+ help_definition = DownloadYourDataSerializers.help_definitions_for(user)
104
+
105
+ help_definition.each do |entity, headers|
106
+ next if headers.empty?
107
+
108
+ readme_file << "\n\n## #{entity}\n\n"
109
+
110
+ headers.each do |header, help_value|
111
+ readme_file << "* #{header}: #{help_value}\n"
112
+ end
113
+ end
114
+
115
+ readme_file
116
+ end
83
117
  end
84
118
  end
@@ -36,20 +36,29 @@ module Decidim
36
36
  return unless resource
37
37
  return unless event_class.types.include?(:email)
38
38
 
39
- followers.each do |recipient|
40
- next unless ["all", "followed-only"].include?(recipient.notification_types)
39
+ send_to_followers
40
+ send_to_affected_users
41
+ end
41
42
 
42
- send_email_to(recipient, user_role: :follower)
43
- end
43
+ private
44
44
 
45
+ def send_to_affected_users
45
46
  affected_users.each do |recipient|
46
47
  next unless ["all", "own-only"].include?(recipient.notification_types)
48
+ next if recipient.deleted? || recipient.blocked?
47
49
 
48
50
  send_email_to(recipient, user_role: :affected_user)
49
51
  end
50
52
  end
51
53
 
52
- private
54
+ def send_to_followers
55
+ followers.each do |recipient|
56
+ next unless ["all", "followed-only"].include?(recipient.notification_types)
57
+ next if recipient.deleted? || recipient.blocked?
58
+
59
+ send_email_to(recipient, user_role: :follower)
60
+ end
61
+ end
53
62
 
54
63
  attr_reader :event, :event_class, :resource, :followers, :affected_users, :extra
55
64
 
@@ -0,0 +1,272 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Decidim
4
+ # This class is used to manage the onboarding funnel for users that
5
+ # intended to perform an action in the platform but it requires some
6
+ # verification steps to be completed.
7
+ #
8
+ # The flow is as follows:
9
+ #
10
+ # 1. A visitor (logged out) tries to perform an action that requires to be logged in
11
+ # 2. The intended action is stored in a cookie (onboarding key)
12
+ # 3. The visitor logs in and the onboarding cookie data is merged into the User#extended_data column.
13
+ # 4. After logging in, the user is redirected to the onboarding pending authorizations page
14
+ # if there is any pending action stored in the User#extended_data column
15
+ # 5. The user completes the required authorizations
16
+ # 6. When the onboarding authorizations page detects that all the verification steps have been
17
+ # completed, it will remove the pending action from the User#extended_data column and redirect
18
+ # the user to the intended action.
19
+ #
20
+ class OnboardingManager
21
+ # The same key is set in onboarding_pending_action js file to manage the
22
+ # onboarding data in the cookie
23
+ DATA_KEY = "onboarding"
24
+
25
+ attr_reader :user
26
+
27
+ def initialize(user)
28
+ @user = user
29
+ end
30
+
31
+ delegate :ephemeral?, to: :user
32
+
33
+ # Checks if the onboarding data has an action and a model.
34
+ #
35
+ # Returns a boolean
36
+ def valid?
37
+ return if action.blank?
38
+
39
+ permissions_holder.present?
40
+ end
41
+
42
+ # Returns the action to be performed.
43
+ #
44
+ # Returns a string
45
+ def action
46
+ onboarding_action
47
+ end
48
+
49
+ # Returns the translation of the action to be performed if translation
50
+ # is found, otherwise the untranslated literal action key
51
+ #
52
+ # Returns a string
53
+ def action_text
54
+ @action_text ||= if component && I18n.exists?("#{component.manifest.name}.actions.#{action}", scope: "decidim.components")
55
+ I18n.t("#{component.manifest.name}.actions.#{action}", scope: "decidim.components")
56
+ elsif permissions_holder.respond_to?(:resource_manifest) &&
57
+ I18n.exists?("#{permissions_holder.resource_manifest.name}.actions.#{action}", scope: "decidim.resources")
58
+ I18n.t("#{permissions_holder.resource_manifest.name}.actions.#{action}", scope: "decidim.resources")
59
+ else
60
+ action
61
+ end
62
+ end
63
+
64
+ # Checks if there is any pending action for the user.
65
+ #
66
+ # Returns a boolean
67
+ def pending_action?
68
+ onboarding_data.present?
69
+ end
70
+
71
+ # Checks if the user has completed all the required authorizations.
72
+ #
73
+ # active_authorization_methods - A list of the active authorization methods for the user.
74
+ #
75
+ # Returns a boolean
76
+ def finished_verifications?(authorization_methods = active_authorization_methods)
77
+ (authorization_handlers - authorization_methods).empty?
78
+ end
79
+
80
+ # Returns the model related to the action in the onboarding process.
81
+ #
82
+ # Returns an ActiveRecord model
83
+ def model
84
+ return if onboarding_model.blank?
85
+
86
+ @model ||= GlobalID::Locator.locate(onboarding_model)
87
+ end
88
+
89
+ # Returns the permissions_holder if present related to the action in the
90
+ # onboarding process.
91
+ #
92
+ # Returns an ActiveRecord model
93
+ def permissions_holder
94
+ return model if onboarding_permissions_holder.blank?
95
+
96
+ @permissions_holder ||= GlobalID::Locator.locate(onboarding_permissions_holder)
97
+ end
98
+
99
+ # Returns the model name related to the action in the onboarding process.
100
+ #
101
+ # Returns a string
102
+ def model_name
103
+ return unless valid?
104
+
105
+ @model_name ||= (model.presence || permissions_holder).class.model_name
106
+ end
107
+
108
+ # Returns the resource title associated to the action. If the model is defined
109
+ # its title is used, if not the permissions holder title
110
+ #
111
+ # Returns a translations Hash or a String
112
+ def model_title
113
+ return unless valid?
114
+
115
+ @model_title ||= begin
116
+ resource = model.presence || permissions_holder
117
+
118
+ method = [:title, :name].find { |m| resource.respond_to?(m) }
119
+
120
+ resource.send(method) if method
121
+ end
122
+ end
123
+
124
+ # Filters the given authorizations that are required for the onboarding process.
125
+ #
126
+ # authorizations - An array of Decidim::Authorization objects
127
+ #
128
+ # Returns an array of Decidim::Authorization objects
129
+ def filter_authorizations(authorizations)
130
+ filtered_authorizations = authorizations.select { |authorization| authorization_handlers.include?(authorization.name) }
131
+
132
+ # By calling authorize on each authorization the path generated for each
133
+ # one will include the specific options of the action if available
134
+ filtered_authorizations.each do |authorization|
135
+ next unless authorization.is_a? Decidim::Verifications::Adapter
136
+
137
+ authorization.authorize(nil, permissions.dig("authorization_handlers", authorization.name, "options") || {}, model, permissions_holder)
138
+ end
139
+ filtered_authorizations
140
+ end
141
+
142
+ # Returns a hash which can be passed to action_authorized_to helper method
143
+ # to determine the permissions status of the action
144
+ #
145
+ # Returns a Hash
146
+ def action_authorized_resources
147
+ return {} unless valid?
148
+
149
+ {
150
+ resource: model,
151
+ permissions_holder: onboarding_permissions_holder.presence && permissions_holder
152
+ }
153
+ end
154
+
155
+ # Returns the path to redirect after finishing the verification process. The path
156
+ # can be obtained from the user onboarding redirect_path data or if a resource is
157
+ # present using a ResourceLocatorPresenter
158
+ #
159
+ # Returns a String
160
+ def finished_redirect_path
161
+ @finished_redirect_path ||= onboarding_data["redirect_path"].presence || model_path
162
+ end
163
+
164
+ def root_path
165
+ component_path || Decidim::Core::Engine.routes.url_helpers.root_path
166
+ end
167
+
168
+ def component_path
169
+ return if component.blank?
170
+
171
+ EngineRouter.main_proxy(component).root_path
172
+ end
173
+
174
+ def expired?
175
+ return unless ephemeral?
176
+
177
+ session_duration > Decidim.config.expire_session_after.to_i
178
+ end
179
+
180
+ # Time in seconds since the last login or creation of the user
181
+ #
182
+ # Returns an Integer
183
+ def session_duration
184
+ Time.current.to_i - (user.last_sign_in_at || user.created_at).to_i
185
+ end
186
+
187
+ # This method is used to determine if an ephemeral user has an onboarding
188
+ # page to be redirected or only an authorization is required to complete the
189
+ # verification
190
+ #
191
+ # Returns a Boolean
192
+ def available_authorization_selection_page?
193
+ return true unless valid? && ephemeral?
194
+
195
+ authorization_handlers.count > 1
196
+ end
197
+
198
+ private
199
+
200
+ def active_authorization_methods
201
+ @active_authorization_methods ||= Verifications::Authorizations.new(organization: user.organization, user:).pluck(:name)
202
+ end
203
+
204
+ # Returns the permissions for the action and model in the onboarding process.
205
+ #
206
+ # Returns a hash
207
+ def permissions
208
+ @permissions ||= permissions_holder&.permissions&.fetch(action, nil) || component&.permissions&.fetch(action, nil)
209
+ end
210
+
211
+ # Returns the authorization handlers for the action and model in the onboarding process.
212
+ #
213
+ # Returns an array of strings
214
+ def authorization_handlers
215
+ return [] unless permissions
216
+
217
+ permissions["authorization_handlers"]&.keys&.map(&:to_s) || []
218
+ end
219
+
220
+ # Returns the model component or the permissions holder when it is a component.
221
+ #
222
+ # Returns a Decidim::Component
223
+ def component
224
+ return permissions_holder if permissions_holder.is_a?(Decidim::Component)
225
+ return if model.blank?
226
+ return unless model.respond_to?(:component)
227
+
228
+ model.component
229
+ end
230
+
231
+ # Returns the onboarding data from the user's extended_data related to the model in gid format.
232
+ #
233
+ # Returns a string
234
+ def onboarding_model
235
+ onboarding_data["model"]
236
+ end
237
+
238
+ # Returns the onboarding data from the user's extended_data related to the permissions_holder
239
+ # in gid format. This attribute is optional and may not be present
240
+ #
241
+ # Returns a string
242
+ def onboarding_permissions_holder
243
+ onboarding_data["permissions_holder"]
244
+ end
245
+
246
+ # Returns the onboarding data from the user's extended_data related to the action.
247
+ #
248
+ # Returns a string
249
+ def onboarding_action
250
+ onboarding_data["action"]
251
+ end
252
+
253
+ # Returns the onboarding data from the user's extended_data.
254
+ #
255
+ # Returns a hash
256
+ def onboarding_data
257
+ user.extended_data[DATA_KEY] || {}
258
+ end
259
+
260
+ # Returns the path associated to the model using ResourceLocatorPresenter. If model is not
261
+ # present returns the root path
262
+ #
263
+ # Returns a String
264
+ def model_path
265
+ return Decidim::Core::Engine.routes.url_helpers.root_path if model.blank?
266
+
267
+ ResourceLocatorPresenter.new(model).url
268
+ rescue NoMethodError
269
+ Decidim::Core::Engine.routes.url_helpers.root_path
270
+ end
271
+ end
272
+ end
@@ -9,48 +9,86 @@ module Decidim
9
9
  class OpenDataExporter
10
10
  FILE_NAME_PATTERN = "%{host}-open-data-%{entity}.csv"
11
11
 
12
- attr_reader :organization, :path
12
+ attr_reader :organization, :path, :resource, :help_definition
13
+
14
+ include Decidim::TranslatableAttributes
13
15
 
14
16
  # Public: Initializes the class.
15
17
  #
16
18
  # organization - The Organization to export the data from.
17
19
  # path - The String path where to write the zip file.
18
- def initialize(organization, path)
20
+ # resource - The String of the component or participatory space to export. If nil, it will export all.
21
+ def initialize(organization, path, resource = nil)
19
22
  @organization = organization
20
23
  @path = File.expand_path path
24
+ @resource = resource
25
+ @help_definition = {}
21
26
  end
22
27
 
23
28
  def export
24
29
  dirname = File.dirname(path)
25
30
  FileUtils.mkdir_p(dirname) unless File.directory?(dirname)
26
- File.binwrite(path, data)
31
+ if resource.nil?
32
+ File.binwrite(path, data_for_all_resources)
33
+ else
34
+ File.write(path, data_for_resource(resource))
35
+ end
27
36
  end
28
37
 
29
38
  private
30
39
 
31
- def data
40
+ def data_for_all_resources
32
41
  buffer = Zip::OutputStream.write_buffer do |out|
42
+ core_data_manifests.each do |manifest|
43
+ add_file_to_output(out, format(FILE_NAME_PATTERN, { host: organization.host, entity: manifest.name }), data_for_core(manifest).read)
44
+ end
33
45
  open_data_component_manifests.each do |manifest|
34
- add_file_to_output(out, format(FILE_NAME_PATTERN, { host: organization.host, entity: manifest.name }), data_for_component(manifest))
46
+ add_file_to_output(out, format(FILE_NAME_PATTERN, { host: organization.host, entity: manifest.name }), data_for_component(manifest).read)
35
47
  end
36
48
  open_data_participatory_space_manifests.each do |manifest|
37
- add_file_to_output(out, format(FILE_NAME_PATTERN, { host: organization.host, entity: manifest.name }), data_for_participatory_space(manifest))
49
+ add_file_to_output(out, format(FILE_NAME_PATTERN, { host: organization.host, entity: manifest.name }), data_for_participatory_space(manifest).read)
38
50
  end
51
+
52
+ add_file_to_output(out, "README.md", readme)
53
+ add_file_to_output(out, "LICENSE.md", license)
39
54
  end
40
55
 
41
56
  buffer.string
42
57
  end
43
58
 
59
+ def data_for_core(export_manifest)
60
+ collection = export_manifest.collection.call(organization)
61
+ exporter = Decidim::Exporters::CSV.new(collection, export_manifest.serializer)
62
+
63
+ get_help_definition(:core, exporter, export_manifest) unless collection.empty?
64
+
65
+ exporter.export
66
+ end
67
+
68
+ def data_for_resource(resource)
69
+ export_manifest = (core_data_manifests + open_data_component_manifests + open_data_participatory_space_manifests)
70
+ .select { |manifest| manifest.name == resource.to_sym }.first
71
+
72
+ case export_manifest.respond_to?(:manifest) && export_manifest.manifest
73
+ when Decidim::ComponentManifest
74
+ data_for_component(export_manifest).read
75
+ when Decidim::ParticipatorySpaceManifest
76
+ data_for_participatory_space(export_manifest).read
77
+ else
78
+ data_for_core(export_manifest).read
79
+ end
80
+ end
81
+
44
82
  def data_for_component(export_manifest, col_sep = Decidim.default_csv_col_sep)
45
83
  headers = []
46
84
  collection = []
47
85
  ActiveRecord::Base.uncached do
48
- components.where(manifest_name: export_manifest.manifest.name).find_each do |component|
86
+ components.where(manifest_name: export_manifest.manifest.name).unscope(:order).find_each do |component|
49
87
  export_manifest.collection.call(component).find_in_batches(batch_size: 100) do |batch|
50
- exporter = Decidim::Exporters::CSV.new(batch, export_manifest.serializer)
88
+ serializer = export_manifest.open_data_serializer.nil? ? export_manifest.serializer : export_manifest.open_data_serializer
89
+ exporter = Decidim::Exporters::CSV.new(batch, serializer)
51
90
  headers.push(*exporter.headers)
52
91
  exported = exporter.export
53
-
54
92
  tmpdir = Dir::Tmpname.create(export_manifest.name.to_s) do
55
93
  # just get an empty file name
56
94
  end
@@ -59,6 +97,8 @@ module Decidim
59
97
  File.write(filename, exported.read)
60
98
 
61
99
  collection.push(filename)
100
+
101
+ get_help_definition(:components, exporter, export_manifest) unless collection.empty?
62
102
  end
63
103
  end
64
104
  end
@@ -76,16 +116,102 @@ module Decidim
76
116
  end
77
117
 
78
118
  def data_for_participatory_space(export_manifest)
79
- collection = participatory_spaces.filter { |space| space.manifest.name == export_manifest.manifest.name }.flat_map do |participatory_space|
119
+ collection = participatory_spaces.flat_map do |participatory_space|
80
120
  export_manifest.collection.call(participatory_space)
81
121
  end
122
+ serializer = export_manifest.open_data_serializer.nil? ? export_manifest.serializer : export_manifest.open_data_serializer
123
+ exporter = Decidim::Exporters::CSV.new(collection, serializer)
124
+ get_help_definition(:spaces, exporter, export_manifest) unless collection.empty?
125
+
126
+ exporter.export
127
+ end
128
+
129
+ def get_help_definition(manifest_type, exporter, export_manifest)
130
+ help_definition[manifest_type] = {} if help_definition[manifest_type].nil?
131
+ help_definition[manifest_type][export_manifest.name] = {} if help_definition[manifest_type][export_manifest.name].blank?
132
+ exporter.headers_without_locales.each do |header|
133
+ help_definition[manifest_type][export_manifest.name][header] = I18n.t("decidim.open_data.help.#{export_manifest.name}.#{header}")
134
+ end
135
+ end
136
+
137
+ def readme
138
+ "# #{I18n.t("decidim.open_data.help.core.title", organization: translated_attribute(organization.name))}\n\n
139
+ #{I18n.t("decidim.open_data.help.core.description")}\n\n
140
+ #{core_readme}
141
+ #{space_readme}
142
+ #{component_readme}
143
+ "
144
+ end
145
+
146
+ def core_readme
147
+ return unless help_definition.fetch(:core, false)
148
+
149
+ readme_file = "## #{I18n.t("decidim.open_data.help.core.main")}\n\n"
150
+ help_definition.fetch(:core, []).each do |element, headers|
151
+ readme_file << "### #{element}\n\n"
152
+
153
+ headers.each do |header, help_value|
154
+ readme_file << "* #{header}: #{help_value}\n"
155
+ end
156
+
157
+ readme_file << "\n\n"
158
+ end
159
+ readme_file
160
+ end
161
+
162
+ def space_readme
163
+ return unless help_definition.fetch(:spaces, false)
164
+
165
+ readme_file = "## #{I18n.t("decidim.open_data.help.core.spaces")}\n\n"
166
+
167
+ help_definition.fetch(:spaces, []).each do |space, headers|
168
+ readme_file << "### #{space}\n\n"
169
+
170
+ headers.each do |header, help_value|
171
+ readme_file << "* #{header}: #{help_value}\n"
172
+ end
173
+
174
+ readme_file << "\n\n"
175
+ end
176
+ readme_file
177
+ end
178
+
179
+ def component_readme
180
+ return unless help_definition.fetch(:components, false)
181
+
182
+ readme_file = "## #{I18n.t("decidim.open_data.help.core.components")}\n\n"
183
+
184
+ help_definition.fetch(:components, []).each do |component, headers|
185
+ readme_file << "### #{component}\n\n"
186
+
187
+ headers.each do |header, help_value|
188
+ readme_file << "* #{header}: #{help_value}\n"
189
+ end
190
+
191
+ readme_file << "\n\n"
192
+ end
82
193
 
83
- Decidim::Exporters::CSV.new(collection, export_manifest.serializer).export
194
+ readme_file
84
195
  end
85
196
 
86
- def add_file_to_output(output, file_name, data)
197
+ def license
198
+ link_database = "#{I18n.t("license_database_name", scope: "decidim.open_data.index.license")}: #{I18n.t("license_database_link", scope: "decidim.open_data.index.license")}"
199
+ link_contents = "#{I18n.t("license_contents_name", scope: "decidim.open_data.index.license")}: #{I18n.t("license_contents_link", scope: "decidim.open_data.index.license")}"
200
+
201
+ license_file = I18n.t("title", scope: "decidim.open_data.index.license")
202
+ license_file << "\n\n"
203
+ license_file << I18n.t("body_1_html", scope: "decidim.open_data.index.license", organization_name: translated_attribute(organization.name), link_database:, link_contents:)
204
+
205
+ license_file
206
+ end
207
+
208
+ def add_file_to_output(output, file_name, string)
87
209
  output.put_next_entry(file_name)
88
- output.write data.read
210
+ output.write string
211
+ end
212
+
213
+ def core_data_manifests
214
+ @core_data_manifests ||= Decidim.open_data_manifests.select(&:include_in_open_data)
89
215
  end
90
216
 
91
217
  def open_data_component_manifests
@@ -69,6 +69,19 @@ module Decidim
69
69
  end
70
70
  end
71
71
 
72
+ # Performs the given block and sets the author of the action without log.
73
+ #
74
+ # author - An object that implements `to_gid` or a String
75
+ #
76
+ # Returns whatever the given block returns.
77
+ def perform_action_without_log!(author)
78
+ PaperTrail.request(whodunnit: gid(author)) do
79
+ Decidim::ApplicationRecord.transaction do
80
+ block_given? ? yield : nil
81
+ end
82
+ end
83
+ end
84
+
72
85
  # Updates the `resource` with `update!` and sets the author of the version.
73
86
  #
74
87
  # resource - An ActiveRecord instance that implements `Decidim::Traceable`
@@ -60,7 +60,7 @@ module Decidim
60
60
  end
61
61
 
62
62
  def variant_path(key, options = {})
63
- variant_url(key, **options.merge(only_path: true))
63
+ variant_url(key, **options, only_path: true)
64
64
  end
65
65
 
66
66
  def remote_url=(url)