Module: RESTFramework::BaseModelControllerMixin
- Includes:
- BaseControllerMixin
- Included in:
- ModelControllerMixin, ReadOnlyModelControllerMixin
- Defined in:
- lib/rest_framework/controller_mixins/models.rb
Overview
This module provides the core functionality for controllers based on models.
Defined Under Namespace
Modules: ClassMethods
Constant Summary collapse
- BASE64_REGEX =
/data:(.*);base64,(.*)/
- BASE64_TRANSLATE =
->(field, value) { _, content_type, payload = value.match(BASE64_REGEX).to_a return { io: StringIO.new(Base64.decode64(payload)), content_type: content_type, filename: "image_#{field}#{Rack::Mime::MIME_TYPES.invert[content_type]}", } }
- RRF_BASE_MODEL_CONTROLLER_CONFIG =
{ # Core attributes related to models. model: nil, recordset: nil, # Attributes for configuring record fields. fields: nil, field_config: nil, action_fields: nil, # Options for what should be included/excluded from default fields. exclude_associations: false, include_active_storage: false, include_action_text: false, # Attributes for finding records. find_by_fields: nil, find_by_query_param: "find_by", # Attributes for create/update parameters. allowed_parameters: nil, allowed_action_parameters: nil, # Attributes for the default native serializer. native_serializer_config: nil, native_serializer_singular_config: nil, native_serializer_plural_config: nil, native_serializer_only_query_param: "only", native_serializer_except_query_param: "except", native_serializer_associations_limit: nil, native_serializer_associations_limit_query_param: "associations_limit", native_serializer_include_associations_count: false, # Attributes for default model filtering, ordering, and searching. filterset_fields: nil, ordering_fields: nil, ordering_query_param: "ordering", ordering_no_reorder: false, search_fields: nil, search_query_param: "search", search_ilike: false, # Options for association assignment. permit_id_assignment: true, permit_nested_attributes_assignment: true, allow_all_nested_attributes: false, # Option for `recordset.create` vs `Model.create` behavior. create_from_recordset: true, # Control if filtering is done before find. filter_recordset_before_find: true, # Control if bulk operations are done in a transaction and rolled back on error, or if all bulk # operations are attempted and errors simply returned in the response. bulk_transactional: false, # Control if bulk operations should be done in "batch" mode, using efficient queries, but also # skipping model validations/callbacks. bulk_batch_mode: false, }
Constants included from BaseControllerMixin
RESTFramework::BaseControllerMixin::RRF_BASE_CONTROLLER_CONFIG
Class Method Summary collapse
-
._rrf_bulk_transaction(&block) ⇒ Object
Create a transaction around the passed block, if configured.
- .included(base) ⇒ Object
Instance Method Summary collapse
- #_get_specific_action_config(action_config_key, generic_config_key) ⇒ Object
-
#get_allowed_parameters ⇒ Object
Get a list of parameters allowed for the current action.
-
#get_body_params(data: nil) ⇒ Object
(also: #get_create_params, #get_update_params)
Use strong parameters to filter the request body using the configured allowed parameters.
-
#get_fields ⇒ Object
Get a list of fields, taking into account the current action.
-
#get_filter_backends ⇒ Object
Get filtering backends, defaulting to using ‘ModelFilter`, `ModelOrderingFilter`, and `ModelSearchFilter`.
-
#get_find_by_fields ⇒ Object
Get a list of find_by fields for the current action.
-
#get_options_metadata ⇒ Object
Pass fields to get dynamic metadata based on which fields are available.
-
#get_record ⇒ Object
Get a single record by primary key or another column, if allowed.
-
#get_records ⇒ Object
Get the records this controller has access to after any filtering is applied.
-
#get_recordset ⇒ Object
Get the set of records this controller has access to.
-
#get_recordset_with_includes ⇒ Object
Get the recordset but with any associations included to avoid N+1 queries.
-
#get_serializer_class ⇒ Object
Get the configured serializer class, or ‘NativeSerializer` as a default.
Methods included from BaseControllerMixin
#api_response, #get_filtered_data, #options, #root, #rrf_error_handler, #serialize
Class Method Details
._rrf_bulk_transaction(&block) ⇒ Object
Create a transaction around the passed block, if configured. This is used primarily for bulk actions, but we include it here so it’s always available.
619 620 621 622 623 624 625 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 619 def self._rrf_bulk_transaction(&block) if self.bulk_transactional ActiveRecord::Base.transaction(&block) else yield end end |
.included(base) ⇒ Object
384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 384 def self.included(base) RESTFramework::BaseControllerMixin.included(base) return unless base.is_a?(Class) base.extend(ClassMethods) # Add class attributes (with defaults) unless they already exist. RRF_BASE_MODEL_CONTROLLER_CONFIG.each do |a, default| next if base.respond_to?(a) base.class_attribute(a) # Set default manually so we can still support Rails 4. Maybe later we can use the default # parameter on `class_attribute`. base.send(:"#{a}=", default) end end |
Instance Method Details
#_get_specific_action_config(action_config_key, generic_config_key) ⇒ Object
403 404 405 406 407 408 409 410 411 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 403 def _get_specific_action_config(action_config_key, generic_config_key) action_config = self.class.send(action_config_key)&.with_indifferent_access || {} action = self.action_name&.to_sym # Index action should use :list serializer if :index is not provided. action = :list if action == :index && !action_config.key?(:index) return (action_config[action] if action) || self.class.send(generic_config_key) end |
#get_allowed_parameters ⇒ Object
Get a list of parameters allowed for the current action.
430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 430 def get_allowed_parameters return @_get_allowed_parameters if defined?(@_get_allowed_parameters) @_get_allowed_parameters = self._get_specific_action_config( :allowed_action_parameters, :allowed_parameters, ) return @_get_allowed_parameters if @_get_allowed_parameters # For fields, automatically add `_id`/`_ids` and `_attributes` variations for associations. variations = [] hash_variations = {} reflections = self.class.get_model.reflections @_get_allowed_parameters = self.get_fields.map { |f| f = f.to_s # ActiveStorage Integration: `has_one_attached`. if reflections.key?("#{f}_attachment") next f end # ActiveStorage Integration: `has_many_attached`. if reflections.key?("#{f}_attachments") hash_variations[f] = [] next nil end # ActionText Integration. if reflections.key?("rich_test_#{f}") next f end # Return field if it's not an association. next f unless ref = reflections[f] if self.class.permit_id_assignment if ref.collection? hash_variations["#{f.singularize}_ids"] = [] elsif ref.belongs_to? variations << "#{f}_id" end end if self.class.permit_nested_attributes_assignment if self.class.allow_all_nested_attributes hash_variations["#{f}_attributes"] = {} else hash_variations["#{f}_attributes"] = self.class.get_field_config(f)[:sub_fields] end end # Associations are not allowed to be submitted in their bare form. next nil }.compact @_get_allowed_parameters += variations @_get_allowed_parameters << hash_variations return @_get_allowed_parameters end |
#get_body_params(data: nil) ⇒ Object Also known as: get_create_params, get_update_params
Use strong parameters to filter the request body using the configured allowed parameters.
505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 505 def get_body_params(data: nil) data ||= request.request_parameters # Filter the request body with strong params. body_params = ActionController::Parameters.new(data).permit(*self.get_allowed_parameters) # Filter primary key if configured. if self.class.filter_pk_from_request_body body_params.delete(self.class.get_model&.primary_key) end # Filter fields in `exclude_body_fields`. (self.class.exclude_body_fields || []).each { |f| body_params.delete(f) } # ActiveStorage Integration: Translate base64 encoded attachments to upload objects. # # rubocop:disable Layout/LineLength # # Good example base64 image: # data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABgAAAAYCAYAAADgdz34AAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAAApgAAAKYB3X3/OAAAABl0RVh0U29mdHdhcmUAd3d3Lmlua3NjYXBlLm9yZ5vuPBoAAANCSURBVEiJtZZPbBtFFMZ/M7ubXdtdb1xSFyeilBapySVU8h8OoFaooFSqiihIVIpQBKci6KEg9Q6H9kovIHoCIVQJJCKE1ENFjnAgcaSGC6rEnxBwA04Tx43t2FnvDAfjkNibxgHxnWb2e/u992bee7tCa00YFsffekFY+nUzFtjW0LrvjRXrCDIAaPLlW0nHL0SsZtVoaF98mLrx3pdhOqLtYPHChahZcYYO7KvPFxvRl5XPp1sN3adWiD1ZAqD6XYK1b/dvE5IWryTt2udLFedwc1+9kLp+vbbpoDh+6TklxBeAi9TL0taeWpdmZzQDry0AcO+jQ12RyohqqoYoo8RDwJrU+qXkjWtfi8Xxt58BdQuwQs9qC/afLwCw8tnQbqYAPsgxE1S6F3EAIXux2oQFKm0ihMsOF71dHYx+f3NND68ghCu1YIoePPQN1pGRABkJ6Bus96CutRZMydTl+TvuiRW1m3n0eDl0vRPcEysqdXn+jsQPsrHMquGeXEaY4Yk4wxWcY5V/9scqOMOVUFthatyTy8QyqwZ+kDURKoMWxNKr2EeqVKcTNOajqKoBgOE28U4tdQl5p5bwCw7BWquaZSzAPlwjlithJtp3pTImSqQRrb2Z8PHGigD4RZuNX6JYj6wj7O4TFLbCO/Mn/m8R+h6rYSUb3ekokRY6f/YukArN979jcW+V/S8g0eT/N3VN3kTqWbQ428m9/8k0P/1aIhF36PccEl6EhOcAUCrXKZXXWS3XKd2vc/TRBG9O5ELC17MmWubD2nKhUKZa26Ba2+D3P+4/MNCFwg59oWVeYhkzgN/JDR8deKBoD7Y+ljEjGZ0sosXVTvbc6RHirr2reNy1OXd6pJsQ+gqjk8VWFYmHrwBzW/n+uMPFiRwHB2I7ih8ciHFxIkd/3Omk5tCDV1t+2nNu5sxxpDFNx+huNhVT3/zMDz8usXC3ddaHBj1GHj/As08fwTS7Kt1HBTmyN29vdwAw+/wbwLVOJ3uAD1wi/dUH7Qei66PfyuRj4Ik9is+hglfbkbfR3cnZm7chlUWLdwmprtCohX4HUtlOcQjLYCu+fzGJH2QRKvP3UNz8bWk1qMxjGTOMThZ3kvgLI5AzFfo379UAAAAASUVORK5CYII= # # rubocop:enable Layout/LineLength self.class.get_model..keys.each do |k| next unless (body_params[k].is_a?(String) && body_params[k].match?(BASE64_REGEX)) || (body_params[k].is_a?(Array) && body_params[k].all? { |v| v.is_a?(String) && v.match?(BASE64_REGEX) }) if body_params[k].is_a?(Array) body_params[k] = body_params[k].map { |v| BASE64_TRANSLATE.call(k, v) } else body_params[k] = BASE64_TRANSLATE.call(k, body_params[k]) end end return body_params end |
#get_fields ⇒ Object
Get a list of fields, taking into account the current action.
414 415 416 417 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 414 def get_fields fields = self._get_specific_action_config(:action_fields, :fields) return self.class.get_fields(input_fields: fields) end |
#get_filter_backends ⇒ Object
Get filtering backends, defaulting to using ‘ModelFilter`, `ModelOrderingFilter`, and `ModelSearchFilter`.
496 497 498 499 500 501 502 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 496 def get_filter_backends return self.class.filter_backends || [ RESTFramework::ModelFilter, RESTFramework::ModelOrderingFilter, RESTFramework::ModelSearchFilter, ] end |
#get_find_by_fields ⇒ Object
Get a list of find_by fields for the current action.
425 426 427 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 425 def get_find_by_fields return self.class.find_by_fields end |
#get_options_metadata ⇒ Object
Pass fields to get dynamic metadata based on which fields are available.
420 421 422 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 420 def return self.class. end |
#get_record ⇒ Object
Get a single record by primary key or another column, if allowed. The return value is cached and exposed to the view as the ‘@record` instance variable.
590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 590 def get_record # Cache the result. return @record if instance_variable_defined?(:@record) recordset = self.get_recordset find_by_key = self.class.get_model.primary_key # Find by another column if it's permitted. if find_by_param = self.class.find_by_query_param.presence if find_by = params[find_by_param].presence find_by_fields = self.get_find_by_fields&.map(&:to_s) if !find_by_fields || find_by.in?(find_by_fields) find_by_key = find_by end end end # Filter recordset, if configured. if self.filter_recordset_before_find recordset = self.get_records end # Return the record. Route key is always `:id` by Rails convention. return @record = recordset.find_by!(find_by_key => request.path_parameters[:id]) end |
#get_records ⇒ Object
Get the records this controller has access to after any filtering is applied.
582 583 584 585 586 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 582 def get_records return @records if instance_variable_defined?(:@records) return @records = self.get_filtered_data(self.get_recordset_with_includes) end |
#get_recordset ⇒ Object
Get the set of records this controller has access to. The return value is cached and exposed to the view as the ‘@recordset` instance variable.
547 548 549 550 551 552 553 554 555 556 557 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 547 def get_recordset return @recordset if instance_variable_defined?(:@recordset) return (@recordset = self.class.recordset) if self.class.recordset # If there is a model, return that model's default scope (all records by default). if (model = self.class.get_model) return @recordset = model.all end return @recordset = nil end |
#get_recordset_with_includes ⇒ Object
Get the recordset but with any associations included to avoid N+1 queries.
560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 560 def get_recordset_with_includes reflections = self.class.get_model.reflections associations = self.get_fields.map { |f| if reflections.key?(f) f.to_sym elsif reflections.key?("rich_text_#{f}") :"rich_text_#{f}" elsif reflections.key?("#{f}_attachment") :"#{f}_attachment" elsif reflections.key?("#{f}_attachments") :"#{f}_attachments" end }.compact if associations.any? return self.get_recordset.includes(associations) end return self.get_recordset end |
#get_serializer_class ⇒ Object
Get the configured serializer class, or ‘NativeSerializer` as a default.
490 491 492 |
# File 'lib/rest_framework/controller_mixins/models.rb', line 490 def get_serializer_class return super || RESTFramework::NativeSerializer end |