Module: GraphQL::Client::ViewModule
- Defined in:
- lib/graphql/client/view_module.rb
Overview
Allows a magic namespace to map to app/views/*/.erb files to retrieve statically defined GraphQL definitions.
# app/views/users/show.html.erb
<%grapql
fragment UserFragment on User { }
%>
# Loads graphql section from app/views/users/show.html.erb
Views::Users::Show::UserFragment
Instance Attribute Summary collapse
-
#client ⇒ Object
Returns the value of attribute client.
-
#load_path ⇒ Object
(also: #path)
Public: Directory to retrieve nested GraphQL definitions from.
-
#source_path ⇒ Object
Public: if this module was defined by a view.
Class Method Summary collapse
-
.extract_graphql_section(src) ⇒ Object
Public: Extract GraphQL section from ERB template.
-
.valid_constant_name?(name) ⇒ Boolean
Internal: Check if name is a valid Ruby constant identifier.
Instance Method Summary collapse
-
#const_missing(name) ⇒ Object
Public: Implement constant missing hook to autoload View ERB statics.
-
#eager_load! ⇒ Object
Public: Eager load module and all subdependencies.
- #load_and_set_module(name) ⇒ Object
-
#load_module(name) ⇒ Object
Internal: Initialize new module for constant name and load ERB statics.
- #placeholder_module(name) ⇒ Object
Instance Attribute Details
#client ⇒ Object
Returns the value of attribute client.
20 21 22 |
# File 'lib/graphql/client/view_module.rb', line 20 def client @client end |
#load_path ⇒ Object Also known as: path
Public: Directory to retrieve nested GraphQL definitions from.
Returns absolute String path under app/views.
79 80 81 |
# File 'lib/graphql/client/view_module.rb', line 79 def load_path @load_path end |
#source_path ⇒ Object
Public: if this module was defined by a view
Returns absolute String path under app/views.
86 87 88 |
# File 'lib/graphql/client/view_module.rb', line 86 def source_path @source_path end |
Class Method Details
.extract_graphql_section(src) ⇒ Object
Public: Extract GraphQL section from ERB template.
src - String ERB text
Returns String GraphQL query and line number or nil or no section was defined.
28 29 30 31 32 |
# File 'lib/graphql/client/view_module.rb', line 28 def self.extract_graphql_section(src) query_string = src.scan(/<%graphql([^%]+)%>/).flatten.first return nil unless query_string [query_string, Regexp.last_match.pre_match.count("\n") + 1] end |
.valid_constant_name?(name) ⇒ Boolean
Internal: Check if name is a valid Ruby constant identifier.
name - String or Symbol constant name
Examples
valid_constant_name?("Foo") #=> true
valid_constant_name?("404") #=> false
Returns true if name is a valid constant, otherwise false if name would result in a “NameError: wrong constant name”.
72 73 74 |
# File 'lib/graphql/client/view_module.rb', line 72 def self.valid_constant_name?(name) name.to_s =~ /^[A-Z][a-zA-Z0-9_]*$/ end |
Instance Method Details
#const_missing(name) ⇒ Object
Public: Implement constant missing hook to autoload View ERB statics.
name - String or Symbol constant name
Returns module or raises NameError if missing.
145 146 147 |
# File 'lib/graphql/client/view_module.rb', line 145 def const_missing(name) load_and_set_module(name) || super end |
#eager_load! ⇒ Object
Public: Eager load module and all subdependencies.
Use in production when cache_classes is true.
Traverses all app/views/*/.erb and loads all static constants defined in ERB files.
Examples
Views.eager_load!
Returns nothing.
46 47 48 49 50 51 52 53 54 55 56 57 58 59 |
# File 'lib/graphql/client/view_module.rb', line 46 def eager_load! return unless File.directory?(load_path) Dir.entries(load_path).sort.each do |entry| next if entry == "." || entry == ".." name = entry.sub(/(\.\w+)+$/, "").camelize.to_sym if ViewModule.valid_constant_name?(name) mod = const_defined?(name, false) ? const_get(name) : load_and_set_module(name) mod.eager_load! if mod end end nil end |
#load_and_set_module(name) ⇒ Object
127 128 129 130 131 132 133 134 135 136 137 138 |
# File 'lib/graphql/client/view_module.rb', line 127 def load_and_set_module(name) placeholder = placeholder_module(name) const_set(name, placeholder) if placeholder mod = load_module(name) return placeholder unless mod remove_const(name) if placeholder const_set(name, mod) mod.unloadable if mod.respond_to?(:unloadable) mod end |
#load_module(name) ⇒ Object
Internal: Initialize new module for constant name and load ERB statics.
name - String or Symbol constant name.
Examples
Views::Users.load_module(:Profile)
Views::Users::Profile.load_module(:Show)
Returns new Module implementing Loadable concern.
98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 |
# File 'lib/graphql/client/view_module.rb', line 98 def load_module(name) pathname = ActiveSupport::Inflector.underscore(name.to_s) path = Dir[File.join(load_path, "{#{pathname},_#{pathname}}{.*}")].sort.map { |fn| File.(fn) }.first return if !path || File.extname(path) != ".erb" contents = File.read(path) query, lineno = ViewModule.extract_graphql_section(contents) return unless query mod = client.parse(query, path, lineno) mod.extend(ViewModule) mod.load_path = File.join(load_path, pathname) mod.source_path = path mod.client = client mod end |
#placeholder_module(name) ⇒ Object
116 117 118 119 120 121 122 123 124 125 |
# File 'lib/graphql/client/view_module.rb', line 116 def placeholder_module(name) dirname = File.join(load_path, ActiveSupport::Inflector.underscore(name.to_s)) return nil unless Dir.exist?(dirname) Module.new.tap do |mod| mod.extend(ViewModule) mod.load_path = dirname mod.client = client end end |