-
Notifications
You must be signed in to change notification settings - Fork 1.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
MONGOID-5734 Custom polymorphic types (backport to 8.1-stable) #5848
base: 8.1-stable
Are you sure you want to change the base?
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -102,6 +102,21 @@ def polymorphic? | |
@polymorphic ||= !!@options[:polymorphic] | ||
end | ||
|
||
# Returns the object responsible for converting polymorphic type references into | ||
# class objects, and vice versa. This is obtained via the `:polymorphic` option | ||
# that was given when the association was defined. | ||
# | ||
# See Mongoid::ModelResolver.resolver for how the `:polymorphic` option is | ||
# interpreted here. | ||
# | ||
# @raise KeyError if no such resolver has been registered under the given | ||
# identifier. | ||
# | ||
# @return [ nil | Mongoid::ModelResolver ] the resolver to use | ||
def resolver | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Would recommend to rename to |
||
@resolver ||= Mongoid::ModelResolver.resolver(@options[:polymorphic]) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Same here |
||
end | ||
|
||
# The name of the field used to store the type of polymorphic association. | ||
# | ||
# @return [ String ] The field used to store the type of polymorphic association. | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,41 @@ | ||
# frozen_string_literal: true | ||
|
||
module Mongoid | ||
module Association | ||
module Referenced | ||
# Implements the `with_polymorphic_criteria` shared behavior. | ||
# | ||
# @api private | ||
module WithPolymorphicCriteria | ||
# If the receiver represents a polymorphic association, applies | ||
# the polymorphic search criteria to the given `criteria` object. | ||
# | ||
# @param [ Mongoid::Criteria ] criteria the criteria to append to | ||
# if receiver is polymorphic. | ||
# @param [ Mongoid::Document ] base the document to use when resolving | ||
# the polymorphic type keys. | ||
# | ||
# @return [ Mongoid::Criteria] the resulting criteria, which may be | ||
# the same as the input. | ||
def with_polymorphic_criterion(criteria, base) | ||
if polymorphic? | ||
# 1. get the resolver for the inverse association | ||
resolver = klass.reflect_on_association(as).resolver | ||
|
||
# 2. look up the list of keys from the resolver, given base | ||
keys = resolver.keys_for(base) | ||
|
||
# 3. use equality if there is just one key, `in` if there are multiple | ||
if keys.many? | ||
criteria.where(type => { :$in => keys }) | ||
else | ||
criteria.where(type => keys.first) | ||
end | ||
else | ||
criteria | ||
end | ||
end | ||
end | ||
end | ||
end | ||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,53 @@ | ||
# frozen_string_literal: true | ||
|
||
module Mongoid | ||
module Errors | ||
# Raised when a polymorphic association is queried, but the type of the | ||
# association cannot be resolved. This usually happens when the data in | ||
# the database references a type that no longer exists. | ||
# | ||
# For example, consider the following model: | ||
# | ||
# class Manager | ||
# include Mongoid::Document | ||
# belongs_to :unit, polymorphic: true | ||
# end | ||
# | ||
# Imagine there is a document in the `managers` collection that looks | ||
# something like this: | ||
# | ||
# { _id: ..., unit_id: ..., unit_type: 'Department::Engineering' } | ||
# | ||
# If, at some point in your refactoring, you rename the `Department::Engineering` | ||
# model to something else, Mongoid will no longer be able to resolve the | ||
# type of this association, and asking for `manager.unit` will raise this | ||
# exception. | ||
# | ||
# To fix this exception, you can add an alias to the model class so that it | ||
# can still be found, even after renaming it: | ||
# | ||
# module Engineering | ||
# class Department | ||
# include Mongoid::Document | ||
# | ||
# identify_as 'Department::Engineering' | ||
# | ||
# # ... | ||
# end | ||
# end | ||
# | ||
# Better practice would be to use unique strings instead of class names to | ||
# identify these polymorphic types in the database (e.g. 'dept' instead of | ||
# 'Department::Engineering'). | ||
class UnrecognizedModelAlias < MongoidError | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It might be nice to rename these new errors to include the word |
||
def initialize(model_alias) | ||
super( | ||
compose_message( | ||
'unrecognized_model_alias', | ||
model_alias: model_alias.inspect | ||
) | ||
) | ||
end | ||
end | ||
end | ||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,27 @@ | ||
# frozen_string_literal: true | ||
|
||
module Mongoid | ||
module Errors | ||
# Raised when a model resolver is referenced, but not registered. | ||
# | ||
# class Manager | ||
# include Mongoid::Document | ||
# belongs_to :unit, polymorphic: :org | ||
# end | ||
# | ||
# If `:org` has not previously been registered as a model resolver, | ||
# Mongoid will raise UnrecognizedResolver when it tries to resolve | ||
# a manager's unit. | ||
class UnrecognizedResolver < MongoidError | ||
def initialize(resolver) | ||
super( | ||
compose_message( | ||
'unrecognized_resolver', | ||
resolver: resolver.inspect, | ||
resolvers: [ :default, *Mongoid::ModelResolver.resolvers.keys ].inspect | ||
) | ||
) | ||
end | ||
end | ||
end | ||
end |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This variable should be refactored to be called
@klass
instead of@class_name
, because it's not a name String--its an actualClass
object.