decidim-core 0.23.3 → 0.24.0.rc1

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of decidim-core might be problematic. Click here for more details.

Files changed (444) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +2 -2
  3. data/app/assets/images/decidim/icons.svg +1 -1
  4. data/app/assets/javascripts/decidim.js.es6 +1 -0
  5. data/app/assets/javascripts/decidim/core/bundle.js +1 -1
  6. data/app/assets/javascripts/decidim/core/bundle.js.map +1 -1
  7. data/app/assets/javascripts/decidim/editor.js.es6 +15 -4
  8. data/app/assets/javascripts/decidim/editor/history_override.js.es6 +117 -0
  9. data/app/assets/javascripts/decidim/editor/linebreak_module.js.es6 +83 -0
  10. data/app/assets/javascripts/decidim/editor/modified_backspace_offset1.js.es6 +98 -0
  11. data/app/assets/javascripts/decidim/editor/modified_backspace_offset_any.js.es6 +48 -0
  12. data/app/assets/javascripts/decidim/editor/modified_backspace_utils.js.es6 +31 -0
  13. data/app/assets/javascripts/decidim/editor/modified_enter.js.es6 +118 -0
  14. data/app/assets/javascripts/decidim/form_filter.component.js.es6 +25 -0
  15. data/app/assets/javascripts/decidim/geocoding/attach_input.js.es6 +41 -4
  16. data/app/assets/javascripts/decidim/input_character_counter.js.es6 +6 -1
  17. data/app/assets/javascripts/decidim/map/controller/markers.js.es6 +27 -9
  18. data/app/assets/javascripts/decidim/session_timeouter.js.es6 +78 -0
  19. data/app/assets/stylesheets/decidim/_variables.scss +9 -0
  20. data/app/assets/stylesheets/decidim/email.scss +14 -2
  21. data/app/assets/stylesheets/decidim/extras/_external-links.scss +0 -1
  22. data/app/assets/stylesheets/decidim/extras/_process_stats.scss +1 -1
  23. data/app/assets/stylesheets/decidim/extras/_results-per-page.scss +1 -0
  24. data/app/assets/stylesheets/decidim/modules/_author-avatar.scss +2 -2
  25. data/app/assets/stylesheets/decidim/modules/_buttons.scss +2 -2
  26. data/app/assets/stylesheets/decidim/modules/_cards.scss +18 -1
  27. data/app/assets/stylesheets/decidim/modules/_collapsible-list.scss +1 -1
  28. data/app/assets/stylesheets/decidim/modules/_comments.scss +5 -1
  29. data/app/assets/stylesheets/decidim/modules/_conference-speaker.scss +4 -3
  30. data/app/assets/stylesheets/decidim/modules/_data-picker.scss +7 -0
  31. data/app/assets/stylesheets/decidim/modules/_extra.scss +4 -0
  32. data/app/assets/stylesheets/decidim/modules/_forms.scss +4 -0
  33. data/app/assets/stylesheets/decidim/modules/_loading-spinner.scss +34 -0
  34. data/app/assets/stylesheets/decidim/modules/_modules.scss +1 -1
  35. data/app/assets/stylesheets/decidim/modules/_opinion-toggle.scss +1 -1
  36. data/app/assets/stylesheets/decidim/modules/_reveal.scss +6 -0
  37. data/app/assets/stylesheets/decidim/modules/_signup.scss +1 -0
  38. data/app/assets/stylesheets/decidim/modules/{_process-stats.scss → _space-stats.scss} +1 -1
  39. data/app/cells/decidim/activity_cell.rb +11 -8
  40. data/app/cells/decidim/announcement/show.erb +4 -4
  41. data/app/cells/decidim/announcement_cell.rb +22 -2
  42. data/app/cells/decidim/author/flag_user.erb +10 -0
  43. data/app/cells/decidim/author/profile_minicard.erb +1 -1
  44. data/app/cells/decidim/author_cell.rb +9 -1
  45. data/app/cells/decidim/card_m/top.erb +1 -1
  46. data/app/cells/decidim/card_m_cell.rb +1 -1
  47. data/app/cells/decidim/content_blocks/cta/show.erb +14 -0
  48. data/app/cells/decidim/content_blocks/cta_cell.rb +35 -0
  49. data/app/cells/decidim/content_blocks/cta_settings_form/show.erb +9 -0
  50. data/app/cells/decidim/content_blocks/cta_settings_form_cell.rb +13 -0
  51. data/app/cells/decidim/content_blocks/hero_cell.rb +14 -0
  52. data/app/cells/decidim/content_blocks/highlighted_elements/elements.erb +0 -0
  53. data/app/cells/decidim/content_blocks/highlighted_elements/heading.erb +0 -0
  54. data/app/cells/decidim/content_blocks/highlighted_elements/show.erb +4 -0
  55. data/app/cells/decidim/content_blocks/highlighted_elements_cell.rb +51 -0
  56. data/app/cells/decidim/content_blocks/highlighted_elements_settings_form/show.erb +3 -0
  57. data/app/cells/decidim/content_blocks/highlighted_elements_settings_form_cell.rb +28 -0
  58. data/app/cells/decidim/content_blocks/last_activity_cell.rb +11 -0
  59. data/app/cells/decidim/content_blocks/metrics_cell.rb +0 -6
  60. data/app/cells/decidim/content_blocks/stats_cell.rb +0 -6
  61. data/app/cells/decidim/date/show.erb +31 -0
  62. data/app/cells/decidim/date_cell.rb +50 -0
  63. data/app/cells/decidim/endorsement_buttons_cell.rb +3 -3
  64. data/app/cells/decidim/flag_modal/flag_user.erb +24 -0
  65. data/app/cells/decidim/flag_modal/show.erb +24 -0
  66. data/app/cells/decidim/flag_modal_cell.rb +25 -0
  67. data/app/cells/decidim/profile/inaccessible.erb +9 -0
  68. data/app/cells/decidim/profile_cell.rb +9 -1
  69. data/app/cells/decidim/profile_sidebar/show.erb +10 -1
  70. data/app/cells/decidim/profile_sidebar_cell.rb +1 -0
  71. data/app/cells/decidim/reported_content/show.erb +5 -0
  72. data/app/cells/decidim/reported_content_cell.rb +44 -0
  73. data/app/cells/decidim/scopes_picker/scope_picker_prompt.erb +3 -0
  74. data/app/cells/decidim/scopes_picker/scope_picker_values.erb +5 -0
  75. data/app/cells/decidim/scopes_picker/show.erb +14 -0
  76. data/app/cells/decidim/scopes_picker_cell.rb +93 -0
  77. data/app/cells/decidim/tos_page/announcement.erb +1 -1
  78. data/app/cells/decidim/tos_page/form.erb +15 -13
  79. data/app/cells/decidim/tos_page_cell.rb +10 -8
  80. data/app/cells/decidim/translation_bar_cell.rb +1 -1
  81. data/app/cells/decidim/user_conversation/conversation_header.erb +1 -1
  82. data/app/cells/decidim/user_conversations_cell.rb +1 -1
  83. data/app/commands/decidim/create_report.rb +11 -1
  84. data/app/commands/decidim/create_user_report.rb +67 -0
  85. data/app/commands/decidim/gallery_methods.rb +2 -2
  86. data/app/commands/decidim/multiple_attachments_methods.rb +14 -10
  87. data/app/commands/decidim/search.rb +5 -2
  88. data/app/commands/decidim/update_notifications_settings.rb +1 -0
  89. data/app/commands/decidim/update_user_group.rb +1 -0
  90. data/app/controllers/concerns/decidim/devise_controllers.rb +1 -0
  91. data/app/controllers/concerns/decidim/flaggable.rb +5 -1
  92. data/app/controllers/concerns/decidim/force_authentication.rb +11 -3
  93. data/app/controllers/concerns/decidim/locale_switcher.rb +5 -0
  94. data/app/controllers/concerns/decidim/needs_permission.rb +7 -1
  95. data/app/controllers/concerns/decidim/orderable.rb +3 -1
  96. data/app/controllers/concerns/decidim/user_blocked_checker.rb +26 -0
  97. data/app/controllers/decidim/application_controller.rb +22 -0
  98. data/app/controllers/decidim/devise/omniauth_registrations_controller.rb +3 -1
  99. data/app/controllers/decidim/devise/sessions_controller.rb +10 -6
  100. data/app/controllers/decidim/errors_controller.rb +1 -1
  101. data/app/controllers/decidim/locales_controller.rb +1 -1
  102. data/app/controllers/decidim/profiles_controller.rb +6 -0
  103. data/app/controllers/decidim/report_users_controller.rb +42 -0
  104. data/app/controllers/decidim/scopes_controller.rb +19 -10
  105. data/app/controllers/decidim/timeouts_controller.rb +28 -0
  106. data/app/controllers/decidim/user_activities_controller.rb +5 -0
  107. data/app/controllers/decidim/user_timeline_controller.rb +1 -0
  108. data/app/events/decidim/demoted_membership_event.rb +2 -2
  109. data/app/events/decidim/invited_to_group_event.rb +2 -2
  110. data/app/events/decidim/join_request_accepted_event.rb +2 -2
  111. data/app/events/decidim/join_request_created_event.rb +2 -2
  112. data/app/events/decidim/join_request_rejected_event.rb +2 -2
  113. data/app/events/decidim/promoted_to_admin_event.rb +2 -2
  114. data/app/events/decidim/removed_from_group_event.rb +2 -2
  115. data/app/forms/decidim/account_form.rb +1 -1
  116. data/app/forms/decidim/invite_user_form.rb +2 -2
  117. data/app/forms/decidim/messaging/conversation_form.rb +1 -1
  118. data/app/forms/decidim/messaging/message_form.rb +1 -1
  119. data/app/forms/decidim/notifications_settings_form.rb +20 -0
  120. data/app/forms/url_validator.rb +20 -0
  121. data/app/helpers/decidim/action_authorization_helper.rb +2 -0
  122. data/app/helpers/decidim/application_helper.rb +1 -2
  123. data/app/helpers/decidim/cells_helper.rb +6 -0
  124. data/app/helpers/decidim/filters_helper.rb +9 -0
  125. data/app/helpers/decidim/messaging/conversation_helper.rb +11 -1
  126. data/app/helpers/decidim/newsletters_helper.rb +7 -9
  127. data/app/helpers/decidim/scopes_helper.rb +1 -1
  128. data/app/jobs/decidim/block_user_job.rb +11 -0
  129. data/app/jobs/decidim/export_job.rb +2 -2
  130. data/app/jobs/decidim/find_and_update_descendants_job.rb +48 -0
  131. data/app/jobs/decidim/machine_translation_resource_job.rb +3 -0
  132. data/app/jobs/decidim/machine_translation_save_job.rb +29 -0
  133. data/app/jobs/decidim/update_search_indexes_job.rb +11 -0
  134. data/app/jobs/decidim/user_report_job.rb +11 -0
  135. data/app/mailers/decidim/block_user_mailer.rb +21 -0
  136. data/app/mailers/decidim/reported_mailer.rb +33 -3
  137. data/app/mailers/decidim/user_report_mailer.rb +21 -0
  138. data/app/models/decidim/area.rb +0 -1
  139. data/app/models/decidim/area_type.rb +1 -1
  140. data/app/models/decidim/attachment.rb +3 -2
  141. data/app/models/decidim/attachment_collection.rb +1 -1
  142. data/app/models/decidim/authorization.rb +4 -0
  143. data/app/models/decidim/category.rb +6 -1
  144. data/app/models/decidim/impersonation_log.rb +5 -7
  145. data/app/models/decidim/metric.rb +2 -2
  146. data/app/models/decidim/moderation.rb +12 -0
  147. data/app/models/decidim/organization.rb +5 -1
  148. data/app/models/decidim/scope.rb +5 -2
  149. data/app/models/decidim/scope_type.rb +1 -1
  150. data/app/models/decidim/share_token.rb +1 -1
  151. data/app/models/decidim/static_page.rb +11 -1
  152. data/app/models/decidim/static_page_topic.rb +4 -0
  153. data/app/models/decidim/user.rb +12 -1
  154. data/app/models/decidim/user_base_entity.rb +12 -12
  155. data/app/models/decidim/user_block.rb +10 -0
  156. data/app/models/decidim/user_group.rb +1 -0
  157. data/app/models/decidim/user_moderation.rb +22 -0
  158. data/app/models/decidim/user_report.rb +31 -0
  159. data/app/permissions/decidim/permissions.rb +1 -3
  160. data/app/permissions/decidim/report_user_permissions.rb +13 -0
  161. data/app/presenters/decidim/admin_log/impersonation_log_presenter.rb +39 -0
  162. data/app/presenters/decidim/admin_log/organization_presenter.rb +1 -2
  163. data/app/presenters/decidim/admin_log/user_moderation_presenter.rb +63 -0
  164. data/app/presenters/decidim/admin_log/user_presenter.rb +18 -9
  165. data/app/presenters/decidim/home_stats_presenter.rb +3 -3
  166. data/app/presenters/decidim/log/diff_presenter.rb +4 -3
  167. data/app/presenters/decidim/log/resource_presenter.rb +4 -4
  168. data/app/presenters/decidim/log/user_presenter.rb +1 -1
  169. data/app/presenters/decidim/menu_presenter.rb +12 -7
  170. data/app/presenters/decidim/nil_presenter.rb +0 -2
  171. data/app/presenters/decidim/user_presenter.rb +1 -1
  172. data/app/queries/decidim/metrics/blocked_users_metric_manage.rb +26 -0
  173. data/app/queries/decidim/metrics/followers_metric_manage.rb +3 -0
  174. data/app/queries/decidim/metrics/participants_metric_manage.rb +3 -0
  175. data/app/queries/decidim/metrics/reported_users_metric_manage.rb +26 -0
  176. data/app/queries/decidim/metrics/user_reports_metric_manage.rb +26 -0
  177. data/app/services/decidim/activity_search.rb +1 -3
  178. data/app/services/decidim/base_diff_renderer.rb +1 -1
  179. data/app/services/decidim/data_portability_exporter.rb +1 -0
  180. data/app/services/decidim/email_notification_generator.rb +2 -2
  181. data/app/services/decidim/events_manager.rb +1 -1
  182. data/app/services/decidim/open_data_exporter.rb +36 -7
  183. data/app/services/decidim/zip_stream/zip_stream_writer.rb +3 -3
  184. data/app/uploaders/decidim/application_uploader.rb +22 -2
  185. data/app/uploaders/decidim/attachment_uploader.rb +2 -4
  186. data/app/uploaders/decidim/downloader.rb +9 -0
  187. data/app/uploaders/decidim/image_uploader.rb +9 -21
  188. data/app/uploaders/decidim/open_data_uploader.rb +5 -0
  189. data/app/uploaders/decidim/record_image_uploader.rb +2 -2
  190. data/app/validators/etiquette_validator.rb +0 -7
  191. data/app/validators/uploader_content_type_validator.rb +4 -2
  192. data/app/views/decidim/block_user_mailer/notify.html.erb +7 -0
  193. data/app/views/decidim/devise/confirmations/new.html.erb +1 -1
  194. data/app/views/decidim/devise/registrations/edit.html.erb +1 -1
  195. data/app/views/decidim/devise/unlocks/new.html.erb +1 -1
  196. data/app/views/decidim/messaging/conversations/_conversation.html.erb +1 -1
  197. data/app/views/decidim/messaging/conversations/_messages.html.erb +1 -1
  198. data/app/views/decidim/messaging/conversations/_show.html.erb +4 -4
  199. data/app/views/decidim/notifications_settings/show.html.erb +11 -0
  200. data/app/views/decidim/pages/_standalone.html.erb +3 -1
  201. data/app/views/decidim/reported_mailer/report.html.erb +8 -8
  202. data/app/views/decidim/shared/_check_boxes_tree.html.erb +3 -3
  203. data/app/views/decidim/shared/_component_announcement.html.erb +2 -2
  204. data/app/views/decidim/shared/participatory_space_filters/_filters.html.erb +1 -1
  205. data/app/views/decidim/user_report_mailer/notify.html.erb +7 -0
  206. data/app/views/decidim/widgets/_data_picker.html.erb +1 -1
  207. data/app/views/layouts/decidim/_application.html.erb +1 -0
  208. data/app/views/layouts/decidim/_js_configuration.html.erb +1 -1
  209. data/app/views/layouts/decidim/_main_footer.html.erb +6 -4
  210. data/app/views/layouts/decidim/_timeout_modal.html.erb +21 -0
  211. data/config/initializers/active_support.rb +7 -0
  212. data/config/initializers/devise.rb +1 -1
  213. data/config/initializers/omniauth.rb +1 -1
  214. data/config/locales/ar.yml +1 -9
  215. data/config/locales/bg.yml +1 -15
  216. data/config/locales/ca.yml +42 -15
  217. data/config/locales/cs.yml +79 -14
  218. data/config/locales/de.yml +78 -13
  219. data/config/locales/el.yml +1 -16
  220. data/config/locales/en.yml +79 -14
  221. data/config/locales/eo.yml +0 -1
  222. data/config/locales/es-MX.yml +30 -15
  223. data/config/locales/es-PY.yml +30 -15
  224. data/config/locales/es.yml +30 -15
  225. data/config/locales/eu.yml +1 -9
  226. data/config/locales/fi-plain.yml +76 -12
  227. data/config/locales/fi.yml +76 -12
  228. data/config/locales/fr-CA.yml +79 -14
  229. data/config/locales/fr.yml +76 -11
  230. data/config/locales/gl.yml +1 -15
  231. data/config/locales/hu.yml +1 -14
  232. data/config/locales/id-ID.yml +1 -9
  233. data/config/locales/is-IS.yml +0 -2
  234. data/config/locales/it.yml +65 -14
  235. data/config/locales/ja.yml +1 -16
  236. data/config/locales/lv.yml +1 -14
  237. data/config/locales/nl.yml +38 -15
  238. data/config/locales/no.yml +7 -16
  239. data/config/locales/pl.yml +98 -23
  240. data/config/locales/pt-BR.yml +1 -9
  241. data/config/locales/pt.yml +1 -16
  242. data/config/locales/ro-RO.yml +20 -16
  243. data/config/locales/ru.yml +1 -10
  244. data/config/locales/sk.yml +1 -14
  245. data/config/locales/sr-CS.yml +0 -9
  246. data/config/locales/sv.yml +5 -16
  247. data/config/locales/tr-TR.yml +62 -13
  248. data/config/locales/uk.yml +1 -7
  249. data/config/locales/zh-CN.yml +1 -16
  250. data/config/routes.rb +9 -4
  251. data/db/migrate/20170713131206_add_admin_to_users.rb +1 -1
  252. data/db/migrate/20180206143340_fix_reference_for_all_resources.rb +2 -0
  253. data/db/migrate/20180314085339_rename_maximum_votes_per_proposal_to_threshold_per_proposal.rb +2 -2
  254. data/db/migrate/20200929171508_remove_show_statistics_from_organizations.rb +7 -0
  255. data/db/migrate/20201010124755_create_decidim_user_moderations.rb +12 -0
  256. data/db/migrate/20201010124756_create_decidim_user_reports.rb +16 -0
  257. data/db/migrate/20201010224433_add_suspension_fields_to_decidim_users.rb +8 -0
  258. data/db/migrate/20201011074641_create_decidim_user_suspensions.rb +14 -0
  259. data/db/migrate/20201011081626_add_current_suspension_id_to_decidim_users.rb +7 -0
  260. data/db/migrate/20201013071533_add_reported_content_to_moderations.rb +7 -0
  261. data/db/migrate/20201019074554_add_locale_to_moderation_reports.rb +7 -0
  262. data/db/migrate/20201127114444_encrypt_authorization_metadatas.rb +35 -0
  263. data/db/migrate/20201128130723_add_allow_public_access_to_static_pages.rb +17 -0
  264. data/db/migrate/20201218144706_update_table_block_user_functionality.rb +8 -0
  265. data/db/migrate/20201218145252_rename_decidim_user_fields_for_block_functionality.rb +9 -0
  266. data/db/migrate/20210208134328_add_email_on_moderations_to_users.rb +7 -0
  267. data/db/seeds.rb +14 -13
  268. data/lib/decidim/amendable.rb +3 -2
  269. data/{app/functions/decidim/core → lib/decidim/api/functions}/component_finder_base.rb +1 -1
  270. data/{app/functions/decidim/core → lib/decidim/api/functions}/component_list.rb +4 -4
  271. data/{app/functions/decidim/core → lib/decidim/api/functions}/component_list_base.rb +3 -1
  272. data/lib/decidim/api/functions/needs_api_default_order.rb +24 -0
  273. data/{app/functions/decidim/core → lib/decidim/api/functions}/needs_api_filter_and_order.rb +0 -0
  274. data/{app/functions/decidim/core → lib/decidim/api/functions}/participatory_space_finder_base.rb +2 -2
  275. data/{app/functions/decidim/core → lib/decidim/api/functions}/participatory_space_list_base.rb +3 -1
  276. data/{app/functions/decidim/core → lib/decidim/api/functions}/user_entity_finder.rb +1 -4
  277. data/{app/functions/decidim/core → lib/decidim/api/functions}/user_entity_list.rb +1 -3
  278. data/{app/types/decidim/core → lib/decidim/api/input_filters}/base_input_filter.rb +0 -0
  279. data/{app/types/decidim/core → lib/decidim/api/input_filters}/component_input_filter.rb +20 -22
  280. data/{app/types/decidim/core → lib/decidim/api/input_filters}/has_hastaggable_input_filter.rb +1 -1
  281. data/{app/types/decidim/core → lib/decidim/api/input_filters}/has_localized_input_filter.rb +3 -3
  282. data/{app/types/decidim/core → lib/decidim/api/input_filters}/has_publishable_input_filter.rb +6 -6
  283. data/{app/types/decidim/core → lib/decidim/api/input_filters}/has_timestamp_input_filter.rb +12 -12
  284. data/{app/types/decidim/core → lib/decidim/api/input_filters}/participatory_space_input_filter.rb +0 -1
  285. data/lib/decidim/api/input_filters/user_entity_input_filter.rb +89 -0
  286. data/{app/types/decidim/core → lib/decidim/api/input_sorts}/base_input_sort.rb +3 -2
  287. data/{app/types/decidim/core → lib/decidim/api/input_sorts}/component_input_sort.rb +10 -11
  288. data/{app/types/decidim/core → lib/decidim/api/input_sorts}/has_endorsable_input_sort.rb +2 -4
  289. data/{app/types/decidim/core → lib/decidim/api/input_sorts}/has_localized_input_sort.rb +3 -3
  290. data/{app/types/decidim/core → lib/decidim/api/input_sorts}/has_publishable_input_sort.rb +1 -1
  291. data/lib/decidim/api/input_sorts/has_timestamp_input_sort.rb +12 -0
  292. data/{app/types/decidim/core → lib/decidim/api/input_sorts}/participatory_space_input_sort.rb +1 -1
  293. data/{app/types/decidim/core → lib/decidim/api/input_sorts}/user_entity_input_sort.rb +4 -4
  294. data/lib/decidim/api/{amendable_entity_interface.rb → interfaces/amendable_entity_interface.rb} +5 -5
  295. data/lib/decidim/api/interfaces/amendable_interface.rb +17 -0
  296. data/lib/decidim/api/{attachable_interface.rb → interfaces/attachable_interface.rb} +3 -3
  297. data/lib/decidim/api/interfaces/author_interface.rb +32 -0
  298. data/lib/decidim/api/interfaces/authorable_interface.rb +23 -0
  299. data/lib/decidim/api/{categorizable_interface.rb → interfaces/categorizable_interface.rb} +3 -3
  300. data/lib/decidim/api/interfaces/coauthorable_interface.rb +30 -0
  301. data/lib/decidim/api/interfaces/component_interface.rb +22 -0
  302. data/lib/decidim/api/interfaces/endorsable_interface.rb +19 -0
  303. data/lib/decidim/api/{fingerprint_interface.rb → interfaces/fingerprint_interface.rb} +3 -3
  304. data/lib/decidim/api/interfaces/participatory_space_interface.rb +48 -0
  305. data/lib/decidim/api/interfaces/participatory_space_resourceable_interface.rb +25 -0
  306. data/lib/decidim/api/{scopable_interface.rb → interfaces/scopable_interface.rb} +3 -3
  307. data/lib/decidim/api/interfaces/timestamps_interface.rb +15 -0
  308. data/lib/decidim/api/interfaces/traceable_interface.rb +14 -0
  309. data/lib/decidim/api/scalars/date_time_type.rb +17 -0
  310. data/lib/decidim/api/scalars/date_type.rb +17 -0
  311. data/lib/decidim/api/types/amendment_type.rb +19 -0
  312. data/lib/decidim/api/types/area_api_type.rb +16 -0
  313. data/lib/decidim/api/types/area_type_type.rb +13 -0
  314. data/lib/decidim/api/types/attachment_type.rb +13 -0
  315. data/lib/decidim/api/types/category_type.rb +14 -0
  316. data/{app/types/decidim/core → lib/decidim/api/types}/component_type.rb +2 -4
  317. data/lib/decidim/api/types/coordinates_type.rb +21 -0
  318. data/lib/decidim/api/types/decidim_type.rb +13 -0
  319. data/lib/decidim/api/types/fingerprint_type.rb +12 -0
  320. data/lib/decidim/api/types/hashtag_type.rb +13 -0
  321. data/lib/decidim/api/types/localized_string_type.rb +13 -0
  322. data/lib/decidim/api/types/metric_history_type.rb +18 -0
  323. data/lib/decidim/api/types/metric_type.rb +13 -0
  324. data/lib/decidim/api/types/organization_type.rb +17 -0
  325. data/lib/decidim/api/types/participatory_space_link_type.rb +22 -0
  326. data/lib/decidim/api/types/participatory_space_type.rb +10 -0
  327. data/lib/decidim/api/types/scope_api_type.rb +16 -0
  328. data/lib/decidim/api/types/session_type.rb +22 -0
  329. data/lib/decidim/api/types/statistic_type.rb +20 -0
  330. data/lib/decidim/api/types/trace_version_type.rb +21 -0
  331. data/lib/decidim/api/types/translated_field_type.rb +36 -0
  332. data/lib/decidim/api/types/user_group_type.rb +64 -0
  333. data/lib/decidim/api/types/user_type.rb +67 -0
  334. data/lib/decidim/attachment_attributes.rb +57 -0
  335. data/lib/decidim/attributes.rb +1 -0
  336. data/lib/decidim/attributes/clean_string.rb +37 -0
  337. data/lib/decidim/coauthorable.rb +3 -3
  338. data/lib/decidim/component_manifest.rb +17 -0
  339. data/lib/decidim/content_block_manifest.rb +1 -1
  340. data/lib/decidim/content_parsers/hashtag_parser.rb +3 -6
  341. data/lib/decidim/content_parsers/user_group_parser.rb +2 -5
  342. data/lib/decidim/content_parsers/user_parser.rb +2 -5
  343. data/lib/decidim/content_processor.rb +1 -1
  344. data/lib/decidim/content_renderers/hashtag_renderer.rb +3 -6
  345. data/lib/decidim/core.rb +11 -3
  346. data/lib/decidim/core/api.rb +71 -16
  347. data/lib/decidim/core/engine.rb +35 -6
  348. data/lib/decidim/core/test.rb +2 -0
  349. data/lib/decidim/core/test/factories.rb +53 -15
  350. data/lib/decidim/core/test/shared_examples/admin_resource_gallery_examples.rb +1 -1
  351. data/lib/decidim/core/test/shared_examples/amendable/amendment_accepted_event_examples.rb +1 -1
  352. data/lib/decidim/core/test/shared_examples/amendable/amendment_promoted_event_examples.rb +1 -1
  353. data/lib/decidim/core/test/shared_examples/amendable/amendment_rejected_event_examples.rb +1 -1
  354. data/lib/decidim/core/test/shared_examples/comments_examples.rb +11 -9
  355. data/lib/decidim/core/test/shared_examples/component_type.rb +1 -1
  356. data/lib/decidim/core/test/shared_examples/controller_render_views.rb +28 -0
  357. data/lib/decidim/core/test/shared_examples/follows_examples.rb +1 -1
  358. data/lib/decidim/core/test/shared_examples/logo_email.rb +1 -1
  359. data/lib/decidim/core/test/shared_examples/map_examples.rb +1 -1
  360. data/lib/decidim/core/test/shared_examples/process_announcements_examples.rb +1 -1
  361. data/lib/decidim/core/test/shared_examples/reportable.rb +8 -0
  362. data/lib/decidim/core/test/shared_examples/rich_text_editor_examples.rb +1 -1
  363. data/lib/decidim/core/test/shared_examples/searchable_participatory_space_examples.rb +2 -3
  364. data/lib/decidim/core/test/shared_examples/searchable_results_examples.rb +46 -0
  365. data/lib/decidim/core/test/shared_examples/static_pages_examples.rb +43 -0
  366. data/lib/decidim/core/version.rb +1 -1
  367. data/lib/decidim/diffy_extension.rb +2 -2
  368. data/lib/decidim/exporters/csv.rb +3 -2
  369. data/lib/decidim/exporters/export_data.rb +1 -1
  370. data/lib/decidim/faker/internet.rb +17 -0
  371. data/lib/decidim/faker/localized.rb +43 -20
  372. data/lib/decidim/file_validator_humanizer.rb +6 -4
  373. data/lib/decidim/form_builder.rb +33 -22
  374. data/lib/decidim/gamification/badge_scorer.rb +3 -2
  375. data/lib/decidim/gamification/base_event.rb +2 -2
  376. data/lib/decidim/geocodable.rb +2 -0
  377. data/lib/decidim/has_category.rb +1 -1
  378. data/lib/decidim/has_private_users.rb +1 -1
  379. data/lib/decidim/importers/import_manifest.rb +36 -0
  380. data/lib/decidim/map/autocomplete.rb +11 -1
  381. data/lib/decidim/metric_operation.rb +5 -6
  382. data/lib/decidim/participatory_space_resourceable.rb +1 -3
  383. data/lib/decidim/query_extensions.rb +90 -69
  384. data/lib/decidim/randomable.rb +6 -1
  385. data/lib/decidim/record_encryptor.rb +131 -0
  386. data/lib/decidim/reportable.rb +26 -0
  387. data/lib/decidim/resource_manifest.rb +3 -0
  388. data/lib/decidim/resourceable.rb +6 -6
  389. data/lib/decidim/scopable.rb +1 -1
  390. data/lib/decidim/search_resource_fields_mapper.rb +1 -1
  391. data/lib/decidim/searchable.rb +9 -0
  392. data/lib/decidim/settings_manifest.rb +2 -1
  393. data/lib/decidim/shareable_with_token.rb +0 -1
  394. data/lib/decidim/translatable_attributes.rb +2 -1
  395. data/lib/decidim/translatable_resource.rb +10 -0
  396. data/lib/decidim/user_reportable.rb +33 -0
  397. data/lib/decidim/view_model.rb +26 -0
  398. data/lib/premailer/adapter/decidim.rb +4 -4
  399. data/lib/tasks/decidim_data_portability_tasks.rake +4 -4
  400. data/vendor/assets/javascripts/datepicker-locales/foundation-datepicker.ko.js +18 -0
  401. data/vendor/assets/javascripts/datepicker-locales/foundation-datepicker.vi.js +14 -0
  402. data/vendor/assets/javascripts/datepicker-locales/foundation-datepicker.zh-CN.js +14 -0
  403. data/vendor/assets/javascripts/datepicker-locales/foundation-datepicker.zh-TW.js +14 -0
  404. metadata +192 -101
  405. data/app/functions/decidim/core/participatory_space_finder.rb +0 -11
  406. data/app/functions/decidim/core/participatory_space_list.rb +0 -11
  407. data/app/types/decidim/core/amendment_type.rb +0 -26
  408. data/app/types/decidim/core/area_api_type.rb +0 -16
  409. data/app/types/decidim/core/area_type_type.rb +0 -14
  410. data/app/types/decidim/core/attachment_type.rb +0 -14
  411. data/app/types/decidim/core/category_type.rb +0 -16
  412. data/app/types/decidim/core/coordinates_type.rb +0 -19
  413. data/app/types/decidim/core/date_time_type.rb +0 -12
  414. data/app/types/decidim/core/date_type.rb +0 -13
  415. data/app/types/decidim/core/decidim_type.rb +0 -19
  416. data/app/types/decidim/core/fingerprint_type.rb +0 -15
  417. data/app/types/decidim/core/has_timestamp_input_sort.rb +0 -12
  418. data/app/types/decidim/core/hashtag_type.rb +0 -13
  419. data/app/types/decidim/core/localized_string_type.rb +0 -14
  420. data/app/types/decidim/core/metric_history_type.rb +0 -17
  421. data/app/types/decidim/core/metric_type.rb +0 -14
  422. data/app/types/decidim/core/organization_type.rb +0 -20
  423. data/app/types/decidim/core/participatory_space_link_type.rb +0 -24
  424. data/app/types/decidim/core/participatory_space_type.rb +0 -12
  425. data/app/types/decidim/core/scope_api_type.rb +0 -16
  426. data/app/types/decidim/core/session_type.rb +0 -19
  427. data/app/types/decidim/core/statistic_type.rb +0 -22
  428. data/app/types/decidim/core/trace_version_type.rb +0 -29
  429. data/app/types/decidim/core/translated_field_type.rb +0 -45
  430. data/app/types/decidim/core/user_entity_input_filter.rb +0 -81
  431. data/app/types/decidim/core/user_group_type.rb +0 -51
  432. data/app/types/decidim/core/user_type.rb +0 -52
  433. data/app/views/decidim/shared/_announcement.html.erb +0 -1
  434. data/app/views/decidim/shared/_flag_modal.html.erb +0 -20
  435. data/lib/decidim/api/amendable_interface.rb +0 -18
  436. data/lib/decidim/api/author_interface.rb +0 -29
  437. data/lib/decidim/api/authorable_interface.rb +0 -22
  438. data/lib/decidim/api/coauthorable_interface.rb +0 -29
  439. data/lib/decidim/api/component_interface.rb +0 -20
  440. data/lib/decidim/api/endorsable_interface.rb +0 -22
  441. data/lib/decidim/api/participatory_space_interface.rb +0 -47
  442. data/lib/decidim/api/participatory_space_resourceable_interface.rb +0 -21
  443. data/lib/decidim/api/timestamps_interface.rb +0 -21
  444. data/lib/decidim/api/traceable_interface.rb +0 -14
@@ -122,9 +122,10 @@ module Decidim
122
122
  end
123
123
 
124
124
  def recipients
125
- if @model.is_a?(User)
125
+ case @model
126
+ when User
126
127
  [@model]
127
- elsif @model.is_a?(UserGroup)
128
+ when UserGroup
128
129
  @model.users
129
130
  end
130
131
  end
@@ -25,11 +25,11 @@ module Decidim
25
25
  end
26
26
 
27
27
  def badge
28
- @badge ||= Gamification.find_badge(extra.dig("badge_name"))
28
+ @badge ||= Gamification.find_badge(extra["badge_name"])
29
29
  end
30
30
 
31
31
  def current_level
32
- extra.dig("current_level")
32
+ extra["current_level"]
33
33
  end
34
34
 
35
35
  def user
@@ -29,11 +29,13 @@ module Decidim
29
29
 
30
30
  private
31
31
 
32
+ # rubocop:disable Style/OptionalBooleanParameter
32
33
  def do_lookup(_reverse = false)
33
34
  RecordGeocoder.with_record(self) do
34
35
  super
35
36
  end
36
37
  end
38
+ # rubocop:enable Style/OptionalBooleanParameter
37
39
  end
38
40
 
39
41
  module RecordGeocoder
@@ -24,7 +24,7 @@ module Decidim
24
24
  def category_belongs_to_organization
25
25
  return unless category
26
26
 
27
- errors.add(:category, :invalid) unless component.categories.where(id: category.id).exists?
27
+ errors.add(:category, :invalid) unless component.categories.exists?(id: category.id)
28
28
  end
29
29
  end
30
30
  end
@@ -26,7 +26,7 @@ module Decidim
26
26
  id: public_spaces +
27
27
  private_spaces
28
28
  .joins(:participatory_space_private_users)
29
- .where("decidim_participatory_space_private_users.decidim_user_id = ?", user.id)
29
+ .where(decidim_participatory_space_private_users: { decidim_user_id: user.id })
30
30
  )
31
31
  else
32
32
  public_spaces
@@ -0,0 +1,36 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Decidim
4
+ module Importers
5
+ # For importing data from files to components. Every resource type should
6
+ # specify it's own creator, which will be responsible for producing (creating)
7
+ # and finishing (saving) the imported resource.
8
+ class ImportManifest
9
+ attr_reader :name, :manifest
10
+
11
+ # Initializes the manifest.
12
+ #
13
+ # name - The name of the export artifact. It should be unique in the
14
+ # space or component.
15
+ #
16
+ # manifest - The parent manifest where this import manifest belongs to.
17
+ #
18
+ def initialize(name, manifest)
19
+ @name = name.to_sym
20
+ @manifest = manifest
21
+ end
22
+
23
+ # Public: Sets the creator when an argument is provided, returns the
24
+ # stored creator otherwise.
25
+ def creator(creator = nil)
26
+ @creator ||= creator || Decidim::Admin::Import::Creator
27
+ end
28
+
29
+ DEFAULT_FORMATS = %w(CSV JSON Excel).freeze
30
+
31
+ def formats
32
+ DEFAULT_FORMATS
33
+ end
34
+ end
35
+ end
36
+ end
@@ -57,8 +57,18 @@ module Decidim
57
57
  @template.snippets.add(:head, @template.snippets.for(:geocoding))
58
58
  end
59
59
 
60
+ options[:value] ||= object.send(attribute) if object.respond_to?(attribute)
61
+ if object.respond_to?(:latitude) && object.respond_to?(:longitude)
62
+ point = [object.latitude, object.longitude]
63
+ options["data-coordinates"] ||= point.join(",")
64
+ end
65
+
60
66
  field(attribute, options) do |opts|
61
- builder.geocoding_field(@object_name, attribute, opts)
67
+ builder.geocoding_field(
68
+ @object_name,
69
+ attribute,
70
+ opts
71
+ )
62
72
  end
63
73
  end
64
74
  end
@@ -36,12 +36,11 @@ module Decidim
36
36
  # With 'metric_operation' and 'metric_name':
37
37
  # - Returns a single manifest related to that two params
38
38
  def for(metric_operation, metric_name = nil)
39
- list = if metric_name
40
- all.find { |manifest| manifest.metric_operation == metric_operation.to_s && manifest.metric_name == metric_name.to_s }
41
- else
42
- all.find_all { |manifest| manifest.metric_operation == metric_operation.to_s }
43
- end
44
- list
39
+ if metric_name
40
+ all.find { |manifest| manifest.metric_operation == metric_operation.to_s && manifest.metric_name == metric_name.to_s }
41
+ else
42
+ all.find_all { |manifest| manifest.metric_operation == metric_operation.to_s }
43
+ end
45
44
  end
46
45
 
47
46
  def all
@@ -54,9 +54,7 @@ module Decidim
54
54
  manifest = Decidim.find_participatory_space_manifest(participatory_space_name)
55
55
  return self.class.none unless manifest
56
56
 
57
- scope = manifest.participatory_spaces.call(organization)
58
-
59
- scope
57
+ manifest.participatory_spaces.call(organization)
60
58
  end
61
59
 
62
60
  # Links the given resources to this model, replaces any previous links with the same name.
@@ -1,8 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require "decidim/api/component_interface"
4
- require "decidim/api/participatory_space_interface"
5
-
6
3
  module Decidim
7
4
  # This module's job is to extend the API with custom fields related to
8
5
  # decidim-core.
@@ -12,91 +9,115 @@ module Decidim
12
9
  # type - A GraphQL::BaseType to extend.
13
10
  #
14
11
  # Returns nothing.
15
- def self.define(type)
16
- Decidim.participatory_space_manifests.each do |participatory_space_manifest|
17
- type.field participatory_space_manifest.name.to_s.camelize(:lower),
18
- type: type.types[participatory_space_manifest.query_type.constantize],
19
- description: "Lists all #{participatory_space_manifest.name}",
20
- function: participatory_space_manifest.query_list.constantize.new(manifest: participatory_space_manifest)
21
-
22
- type.field participatory_space_manifest.name.to_s.singularize.camelize(:lower),
23
- type: participatory_space_manifest.query_type.constantize,
24
- description: "Finds a #{participatory_space_manifest.name.to_s.singularize}",
25
- function: participatory_space_manifest.query_finder.constantize.new(manifest: participatory_space_manifest)
12
+ def self.included(type)
13
+ type.field :participatory_processes,
14
+ [Decidim::ParticipatoryProcesses::ParticipatoryProcessType],
15
+ null: true,
16
+ description: "Lists all participatory_processes" do
17
+ argument :filter, Decidim::ParticipatoryProcesses::ParticipatoryProcessInputFilter, "This argument let's you filter the results", required: false
18
+ argument :order, Decidim::ParticipatoryProcesses::ParticipatoryProcessInputSort, "This argument let's you order the results", required: false
26
19
  end
27
20
 
28
- type.field :component, Decidim::Core::ComponentInterface do
29
- description "Lists the components this space contains."
30
- argument :id, !types.ID, "The ID of the component to be found"
21
+ type.field :participatory_process,
22
+ Decidim::ParticipatoryProcesses::ParticipatoryProcessType,
23
+ null: true,
24
+ description: "Finds a participatory_process" do
25
+ argument :id, GraphQL::Types::ID, "The ID of the participatory space", required: false
26
+ argument :slug, String, "The slug of the participatory process", required: false
27
+ end
31
28
 
32
- resolve lambda { |_, args, ctx|
33
- component = Decidim::Component.published.find_by(id: args[:id])
34
- component&.organization == ctx[:current_organization] ? component : nil
35
- }
29
+ type.field :component, Decidim::Core::ComponentInterface, null: true do
30
+ description "Lists the components this space contains."
31
+ argument :id, GraphQL::Types::ID, required: true, description: "The ID of the component to be found"
36
32
  end
37
33
 
38
- type.field :session do
39
- type Core::SessionType
40
- description "Return's information about the logged in user"
34
+ type.field :session, Core::SessionType, description: "Return's information about the logged in user", null: true
41
35
 
42
- resolve lambda { |_obj, _args, ctx|
43
- ctx[:current_user]
44
- }
45
- end
36
+ type.field :decidim, Core::DecidimType, "Decidim's framework properties.", null: true
37
+
38
+ type.field :organization, Core::OrganizationType, "The current organization", null: true
46
39
 
47
- type.field :decidim, Core::DecidimType, "Decidim's framework properties." do
48
- resolve ->(_obj, _args, _ctx) { Decidim }
40
+ type.field :hashtags, [Core::HashtagType], null: true, description: "The hashtags for current organization" do
41
+ argument :name, GraphQL::Types::String, "The name of the hashtag", required: false
49
42
  end
50
43
 
51
- type.field :organization, Core::OrganizationType, "The current organization" do
52
- resolve ->(_obj, _args, ctx) { ctx[:current_organization] }
44
+ type.field :metrics, type: [Decidim::Core::MetricType], null: true do
45
+ argument :names, [GraphQL::Types::String], "The names of the metrics you want to retrieve", camelize: false, required: false
46
+ argument :space_type, GraphQL::Types::String, "The type of ParticipatorySpace you want to filter with", camelize: false, required: false
47
+ argument :space_id, GraphQL::Types::Int, "The ID of ParticipatorySpace you want to filter with", camelize: false, required: false
53
48
  end
54
49
 
55
- type.field :hashtags do
56
- type types[Core::HashtagType]
57
- description "The hashtags for current organization"
58
- argument :name, types.String, "The name of the hashtag"
50
+ type.field :user,
51
+ type: Core::AuthorInterface, null: true,
52
+ description: "A participant (user or group) in the current organization" do
53
+ argument :id, GraphQL::Types::ID, "The ID of the participant", required: false
54
+ argument :nickname, GraphQL::Types::String, "The @nickname of the participant", required: false
55
+ end
59
56
 
60
- resolve lambda { |_obj, args, ctx|
61
- Decidim::HashtagsResolver.new(ctx[:current_organization], args[:name]).hashtags
62
- }
57
+ type.field :users,
58
+ type: [Core::AuthorInterface], null: true,
59
+ description: "The participants (users or groups) for the current organization" do
60
+ argument :order, Decidim::Core::UserEntityInputSort, "Provides several methods to order the results", required: false
61
+ argument :filter, Decidim::Core::UserEntityInputFilter, "Provides several methods to filter the results", required: false
63
62
  end
63
+ end
64
64
 
65
- type.field :metrics do
66
- type types[Decidim::Core::MetricType]
67
- argument :names, types[types.String], "The names of the metrics you want to retrieve"
68
- argument :space_type, types.String, "The type of ParticipatorySpace you want to filter with"
69
- argument :space_id, types.Int, "The ID of ParticipatorySpace you want to filter with"
70
-
71
- resolve lambda { |_, args, ctx|
72
- manifests = if args[:names].blank?
73
- Decidim.metrics_registry.all
74
- else
75
- Decidim.metrics_registry.all.select do |manifest|
76
- args[:names].include?(manifest.metric_name.to_s)
77
- end
78
- end
79
- filters = {}
80
- if args[:space_type].present? && args[:space_id].present?
81
- filters[:participatory_space_type] = args[:space_type]
82
- filters[:participatory_space_id] = args[:space_id]
83
- end
65
+ def participatory_processes(filter: {}, order: {})
66
+ manifest = Decidim.participatory_space_manifests.select { |m| m.name == :participatory_processes }.first
67
+ Decidim::Core::ParticipatorySpaceListBase.new(manifest: manifest).call(object, { filter: filter, order: order }, context)
68
+ end
69
+
70
+ def participatory_process(id: nil, slug: nil)
71
+ manifest = Decidim.participatory_space_manifests.select { |m| m.name == :participatory_processes }.first
72
+ Decidim::Core::ParticipatorySpaceFinderBase.new(manifest: manifest).call(object, { id: id, slug: slug }, context)
73
+ end
74
+
75
+ def component(id: {})
76
+ component = Decidim::Component.published.find_by(id: id)
77
+ component&.organization == context[:current_organization] ? component : nil
78
+ end
79
+
80
+ def session
81
+ context[:current_user]
82
+ end
83
+
84
+ def decidim
85
+ Decidim
86
+ end
87
+
88
+ def organization
89
+ context[:current_organization]
90
+ end
84
91
 
85
- manifests.map do |manifest|
86
- Decidim::Core::MetricResolver.new(manifest.metric_name, ctx[:current_organization], filters)
92
+ def hashtags(name: nil)
93
+ Decidim::HashtagsResolver.new(context[:current_organization], name).hashtags
94
+ end
95
+
96
+ def metrics(names: [], space_type: nil, space_id: nil)
97
+ manifests = if names.blank?
98
+ Decidim.metrics_registry.all
99
+ else
100
+ Decidim.metrics_registry.all.select do |manifest|
101
+ names.include?(manifest.metric_name.to_s)
102
+ end
87
103
  end
88
- }
104
+ filters = {}
105
+ if space_type.present? && space_id.present?
106
+ filters[:participatory_space_type] = space_type
107
+ filters[:participatory_space_id] = space_id
89
108
  end
90
109
 
91
- type.field :user,
92
- type: Core::AuthorInterface,
93
- description: "A participant (user or group) in the current organization",
94
- function: Core::UserEntityFinder.new
110
+ manifests.map do |manifest|
111
+ Decidim::Core::MetricResolver.new(manifest.metric_name, context[:current_organization], filters)
112
+ end
113
+ end
95
114
 
96
- type.field :users,
97
- type: type.types[Core::AuthorInterface],
98
- description: "The participants (users or groups) for the current organization",
99
- function: Core::UserEntityList.new
115
+ def user(id: nil, nickname: nil)
116
+ Core::UserEntityFinder.new.call(object, { id: id, nickname: nickname }, context)
117
+ end
118
+
119
+ def users(filter: {}, order: {})
120
+ Core::UserEntityList.new.call(object, { filter: filter, order: order }, context)
100
121
  end
101
122
  end
102
123
  end
@@ -12,7 +12,12 @@ module Decidim
12
12
  def order_randomly(seed)
13
13
  transaction do
14
14
  connection.execute("SELECT setseed(#{connection.quote(seed)})")
15
- order(Arel.sql("RANDOM()")).load
15
+ # Include the record IDs as a base number for the order calculation
16
+ # in order to avoid PostgreSQL random ordering when the records are
17
+ # updated. PostgreSQL can randomly change the base ordering in case
18
+ # the records are changed which is not desired as we want consistent
19
+ # orders for the records.
20
+ order(arel_table[primary_key] * Arel.sql("RANDOM()")).load
16
21
  end
17
22
  end
18
23
  end
@@ -0,0 +1,131 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/concern"
4
+
5
+ module Decidim
6
+ # A concern that provides attribute encryption e.g. to active record models.
7
+ #
8
+ # Use this e.g. in models as follows:
9
+ #
10
+ # class Example < ApplicationRecord
11
+ # include Decidim::RecordEncryptor
12
+ #
13
+ # encrypt_attribute :name, type: :string
14
+ # encrypt_attribute :metadata, type: :hash
15
+ # end
16
+ module RecordEncryptor
17
+ extend ActiveSupport::Concern
18
+
19
+ included do
20
+ # Store the encrypted attributes in a class accessor
21
+ cattr_accessor :encrypted_attributes
22
+
23
+ before_save :ensure_encrypted_attributes if respond_to?(:before_save)
24
+ end
25
+
26
+ class_methods do
27
+ # Public: Defines an attribute that should be encrypted
28
+ def encrypt_attribute(attribute, type:)
29
+ self.encrypted_attributes ||= []
30
+ raise "The attribute #{attribute} is already defined as encrypted" if encrypted_attributes.include?(attribute)
31
+
32
+ encrypted_attributes << attribute
33
+
34
+ # Defines the suffix for the encrypt and decrypt methods. E.g. when
35
+ # the `type` is `:hash`, method `decrypt_hash_values` would be called
36
+ # for decryption and `encrypt_hash_values` would be called for
37
+ # encryption.
38
+ method_suffix = begin
39
+ case type
40
+ when :hash
41
+ "hash_values"
42
+ else
43
+ "value"
44
+ end
45
+ end
46
+
47
+ # Dynamically defines the getter and setter for the encrypted attribute.
48
+ # E.g. when called as `encrypt_attribute :name, type: :string`, this
49
+ # would define the following methods:
50
+ #
51
+ # def name
52
+ # decrypt_value(super)
53
+ # end
54
+ #
55
+ # def name=(value)
56
+ # super(encrypt_value(value))
57
+ # end
58
+ class_eval <<-RUBY, __FILE__, __LINE__ + 1
59
+ def #{attribute}
60
+ return @#{attribute}_decrypted if instance_variable_defined?(:@#{attribute}_decrypted)
61
+
62
+ encrypted_value = begin
63
+ if defined?(super)
64
+ super
65
+ elsif instance_variable_defined?(:@#{attribute})
66
+ @#{attribute}
67
+ end
68
+ end
69
+ @#{attribute}_decrypted = decrypt_#{method_suffix}(encrypted_value)
70
+ end
71
+
72
+ def #{attribute}=(value)
73
+ remove_instance_variable(:@#{attribute}_decrypted) if instance_variable_defined?(:@#{attribute}_decrypted)
74
+ encrypted_value = encrypt_#{method_suffix}(value)
75
+
76
+ if defined?(super)
77
+ super(encrypted_value)
78
+ else
79
+ @#{attribute} = encrypted_value
80
+ end
81
+ end
82
+ RUBY
83
+ end
84
+ end
85
+
86
+ private
87
+
88
+ # Re-assign the encrypted attributes before save so they are also saved when
89
+ # they are modified without calling the accessors. This could happen e.g.
90
+ # for hashes which are modified directly as follows:
91
+ #
92
+ # record = Example.find(1)
93
+ # record.metadata["foo"] = "bar"
94
+ # record.save!
95
+ #
96
+ # This will also clear the cached attributes during saving so that next time
97
+ # they are accessed, they will be updated according to the stored values.
98
+ def ensure_encrypted_attributes
99
+ self.class.encrypted_attributes.each do |attr|
100
+ send("#{attr}=", send(attr))
101
+ end
102
+ end
103
+
104
+ def decrypt_value(value)
105
+ Decidim::AttributeEncryptor.decrypt(value)
106
+ rescue ActiveSupport::MessageEncryptor::InvalidMessage
107
+ # Support for legacy unencrypted values. This is necessary e.g. when
108
+ # migrating the original unencrypted values to encrypted values.
109
+ value
110
+ end
111
+
112
+ def encrypt_value(value)
113
+ Decidim::AttributeEncryptor.encrypt(value)
114
+ end
115
+
116
+ def decrypt_hash_values(hash)
117
+ return hash unless hash.is_a?(Hash)
118
+
119
+ hash.transform_values { |value| ActiveSupport::JSON.decode(decrypt_value(value)) }
120
+ end
121
+
122
+ def encrypt_hash_values(hash)
123
+ return hash unless hash.is_a?(Hash)
124
+
125
+ # The values are stored in JSON encoded format in order to match the
126
+ # PostgreSQL adapter's default functionality as you can see at:
127
+ # https://git.io/JkdYJ
128
+ hash.transform_values { |value| encrypt_value(ActiveSupport::JSON.encode(value)) }
129
+ end
130
+ end
131
+ end