Module: RailsModuleUnification::ActiveSupportExtensions
- Defined in:
- lib/rails_module_unification/active_support_extensions.rb
Constant Summary collapse
- RESOURCE_SUFFIX_NAMES =
%w( Controller Serializer Operations Presenters Policy Policies ).freeze
- RESOURCE_SUFFIXES =
Join all the suffix names together with an “OR” operator
/(#{RESOURCE_SUFFIX_NAMES.join('|')})/- QUALIFIED_NAME_SPLIT =
split on any of the resource suffixes OR the ruby namespace seperator
/::|#{RESOURCE_SUFFIXES}/
Instance Method Summary collapse
- #load_from_path(file_path, qualified_name, from_mod, const_name) ⇒ Object
-
#load_missing_constant(from_mod, const_name) ⇒ Object
Load the constant named
const_namewhich is missing fromfrom_mod. -
#load_missing_constant_error(from_mod, const_name, e) ⇒ Object
the heavy liftign of Rails Module Unification is just adding some additional pathfinding / constat lookup logic when the default (super) can’t find what needs to be found.
-
#resource_path_from_qualified_name(qualified_name) ⇒ Object
A look for the possible places that various qualified names could be.
- #to_path(*args) ⇒ Object
Instance Method Details
#load_from_path(file_path, qualified_name, from_mod, const_name) ⇒ Object
19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 |
# File 'lib/rails_module_unification/active_support_extensions.rb', line 19 def load_from_path(file_path, qualified_name, from_mod, const_name) = File.(file_path) .sub!(/\.rb\z/, '') if loading.include?() raise "Circular dependency detected while autoloading constant #{qualified_name}" else require_or_load(, qualified_name) unless from_mod.const_defined?(const_name, false) raise LoadError, "Unable to autoload constant #{qualified_name}, expected #{file_path} to define it" end return from_mod.const_get(const_name) end end |
#load_missing_constant(from_mod, const_name) ⇒ Object
Load the constant named const_name which is missing from from_mod. If it is not possible to load the constant into from_mod, try its parent module using const_missing.
168 169 170 171 172 173 |
# File 'lib/rails_module_unification/active_support_extensions.rb', line 168 def load_missing_constant(from_mod, const_name) # always default to the actual implementation super rescue LoadError, NameError => e load_missing_constant_error(from_mod, const_name, e) end |
#load_missing_constant_error(from_mod, const_name, e) ⇒ Object
the heavy liftign of Rails Module Unification is just adding some additional pathfinding / constat lookup logic when the default (super) can’t find what needs to be found
182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 |
# File 'lib/rails_module_unification/active_support_extensions.rb', line 182 def load_missing_constant_error(from_mod, const_name, e) # examples # - Api::PostsController # - PostsController qualified_name = qualified_name_for(from_mod, const_name) file_path = resource_path_from_qualified_name(qualified_name) begin return load_from_path(file_path, qualified_name, from_mod, const_name) if file_path rescue LoadError, NameError => e # Recurse! # not found, check the parent at_the_top = from_mod.parent == from_mod return load_missing_constant_error(from_mod.parent, const_name, e) unless at_the_top raise e end name_error = NameError.new(e.) name_error.set_backtrace(caller.reject { |l| l.starts_with? __FILE__ }) raise name_error end |
#resource_path_from_qualified_name(qualified_name) ⇒ Object
The Lookup Rules:
-
all resources are plural
-
file_names can either be named after the type or traditional ruby/rails nameing i.e.: posts_controller.rb vs controller.rb
-
regular namespacing still applies. i.e: Api::V2::CategoriesController should be in
api/v2/categories/controller.rb
The Pattern:
-
namespace_a - api
-
namespace_b - v2
-
resource_name (plural) - posts
-
file_type.rb - controller.rb (or posts_controller.rb)
- operations.rb (or post_operations.rb) -
folder_type - operations/ (or post_operations/)
-
related namespaced classes - create.rb
-
-
-
-
A look for the possible places that various qualified names could be
All examples assume default resource directory (“resources”) and show the order of lookup
78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 |
# File 'lib/rails_module_unification/active_support_extensions.rb', line 78 def resource_path_from_qualified_name(qualified_name) # 1. break apart the qualified name into pieces that can easily be # manipulated # # Api::Posts # => Api, Posts # # Api::PostOperations::Create # => Api, Post, Operations, Create # # Api::PostsController # => Api, Posts, Controller # # Api::V2::PostOperations::Update # => Api, V2, Post, Operations, Update qualified_parts = qualified_name.split(QUALIFIED_NAME_SPLIT).reject(&:blank?) # based on the position of of the resource type name, # anything to the left will be the namespace, and anything # to the right will be the file path within the namespace # (may be obvious, but basically, we're 'pivoting' on RESOURCE_SUFFIX_NAMES) # # Given: Api, V2, Post, Operations, Update # ^ index_of_resource_type (3) index_of_resource_type = qualified_parts.index { |x| RESOURCE_SUFFIX_NAMES.include?(x) } # if this is not part of a resource, don't even bother return unless index_of_resource_type # Api, V2, Post, Operations, Update # => Operations resource_type = qualified_parts[index_of_resource_type] # Api, V2, Post, Operations, Update # => Posts # # Posts, Controller # => Posts original_resource_name = qualified_parts[index_of_resource_type - 1] resource_name = original_resource_name.pluralize # Posts_Controller # Post_Operations named_resource_type = "#{original_resource_name}_#{resource_type}" # Api, V2, Post, Operations, Update # => Api, V2 namespace_index = index_of_resource_type - 1 namespace = namespace_index < 1 ? '' : qualified_parts.take(namespace_index) # Api, V2, Post, Operations, Update # => Update class_index = index_of_resource_type + 1 class_path = class_index < 1 ? '' : qualified_parts.drop(class_index) # Finally, # build all the possible places that this file could be = [ # api/v2/posts/operations/update to_path(namespace, resource_name, resource_type, class_path), # api/v2/posts/post_operations/update to_path(namespace, resource_name, named_resource_type, class_path), # api/v2/posts/posts_controller to_path(namespace, resource_name, named_resource_type), # api/v2/posts/controller to_path(namespace, resource_name, resource_type) ].uniq file_path = '' .each do |path_option| file_path = search_for_file(path_option) break if file_path.present? end file_path end |
#to_path(*args) ⇒ Object
161 162 163 |
# File 'lib/rails_module_unification/active_support_extensions.rb', line 161 def to_path(*args) args.flatten.reject(&:blank?).map(&:underscore).join('/') end |