can_cleanup_param.rb 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. # Copyright (C) 2012-2016 Zammad Foundation, http://zammad-foundation.org/
  2. module ApplicationModel::CanCleanupParam
  3. extend ActiveSupport::Concern
  4. # methods defined here are going to extend the class, not the instance of it
  5. class_methods do
  6. =begin
  7. remove all not used model attributes of params
  8. result = Model.param_cleanup(params)
  9. for object creation, ignore id's
  10. result = Model.param_cleanup(params, true)
  11. returns
  12. result = params # params with valid attributes of model
  13. =end
  14. def param_cleanup(params, new_object = false, inside_nested = false)
  15. if params.respond_to?(:permit!)
  16. params = params.permit!.to_h
  17. end
  18. if params.nil?
  19. raise Exceptions::UnprocessableEntity, "No params for #{self}!"
  20. end
  21. # cleanup each member of array
  22. if params.is_a? Array
  23. return params.map { |elem| param_cleanup(elem, new_object, inside_nested) }
  24. end
  25. data = {}
  26. params.each do |key, value|
  27. data[key.to_s] = value
  28. end
  29. # ignore id for new objects
  30. if new_object && params[:id]
  31. data.delete('id')
  32. end
  33. # only use object attributes
  34. clean_params = ActiveSupport::HashWithIndifferentAccess.new
  35. new.attributes.each_key do |attribute|
  36. next if !data.key?(attribute)
  37. # check reference records, referenced by _id attributes
  38. reflect_on_all_associations.map do |assoc|
  39. class_name = assoc.options[:class_name]
  40. next if !class_name
  41. name = "#{assoc.name}_id"
  42. next if !data.key?(name)
  43. next if data[name].blank?
  44. next if assoc.klass.lookup(id: data[name])
  45. raise Exceptions::UnprocessableEntity, "Invalid value for param '#{name}': #{data[name].inspect}"
  46. end
  47. clean_params[attribute] = data[attribute]
  48. end
  49. clean_params['form_id'] = data['form_id'] if data.key?('form_id') && new.respond_to?('form_id')
  50. if inside_nested
  51. clean_params['id'] = params[:id] if params[:id].present?
  52. clean_params['_destroy'] = data['_destroy'] if data['_destroy'].present?
  53. end
  54. nested_attributes_options.each_key do |nested|
  55. nested_key = "#{nested}_attributes"
  56. target_klass = reflect_on_association(nested).klass
  57. next if data[nested_key].blank?
  58. nested_data = data[nested_key]
  59. if data.key? 'form_id'
  60. case nested_data
  61. when Array
  62. nested_data.each { |item| item['form_id'] = data['form_id'] }
  63. else
  64. nested_data['form_id'] = data['form_id']
  65. end
  66. end
  67. clean_params[nested_key] = target_klass.param_cleanup(nested_data, new_object, true)
  68. end
  69. # we do want to set this via database
  70. filter_unused_params(clean_params)
  71. end
  72. private
  73. =begin
  74. remove all not used params of object (per default :updated_at, :created_at, :updated_by_id and :created_by_id)
  75. if import mode is enabled, just do not used :action and :controller
  76. result = Model.filter_unused_params(params)
  77. returns
  78. result = params # params without listed attributes
  79. =end
  80. def filter_unused_params(data)
  81. params = %i[action controller updated_at created_at updated_by_id created_by_id updated_by created_by]
  82. if Setting.get('import_mode') == true
  83. params = %i[action controller]
  84. end
  85. params.each do |key|
  86. data.delete(key)
  87. end
  88. data
  89. end
  90. end
  91. =begin
  92. merge preferences param
  93. record = Model.find(123)
  94. new_preferences = record.param_preferences_merge(param_preferences)
  95. =end
  96. def param_preferences_merge(new_params)
  97. return new_params if new_params.blank?
  98. return new_params if preferences.blank?
  99. new_params[:preferences] = preferences.merge(new_params[:preferences] || {})
  100. new_params
  101. end
  102. end