Class: Jets::Controller::Rendering::RackRenderer

Inherits:
Object
  • Object
show all
Defined in:
lib/jets/controller/rendering/rack_renderer.rb

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(controller, options = {}) ⇒ RackRenderer

Returns a new instance of RackRenderer.



8
9
10
11
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 8

def initialize(controller, options={})
  @controller = controller
  @options = options
end

Instance Attribute Details

#controllerObject (readonly)

Returns the value of attribute controller.



7
8
9
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 7

def controller
  @controller
end

Class Method Details

.find_app_helper_classesObject

Does not include ApplicationHelper, will include ApplicationHelper explicitly first.



209
210
211
212
213
214
215
216
217
218
219
220
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 209

def find_app_helper_classes
  klasses = []
  expression = "#{Jets.root}/app/helpers/**/*"
  Dir.glob(expression).each do |path|
    next unless File.file?(path)
    class_name = path.sub("#{Jets.root}/app/helpers/","").sub(/\.rb/,'')
    unless class_name == "application_helper"
      klasses << class_name.classify.constantize # autoload
    end
  end
  klasses
end

.setup!Object



189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 189

def setup!
  require "action_controller"
  require "jets/overrides/rails"

  # Load helpers
  # Assign local variable because scoe in the `:action_view do` changes
  app_helper_classes = find_app_helper_classes
  ActiveSupport.on_load :action_view do
    include ApplicationHelper # include first
    app_helper_classes.each do |helper_class|
      include helper_class
    end
  end

  ActionController::Base.append_view_path("#{Jets.root}/app/views")

  setup_webpacker if Jets.webpacker?
end

.setup_webpackerObject



222
223
224
225
226
227
228
229
230
231
232
233
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 222

def setup_webpacker
  require 'webpacker'
  require 'webpacker/helper'

  ActiveSupport.on_load :action_controller do
    ActionController::Base.helper Webpacker::Helper
  end

  ActiveSupport.on_load :action_view do
    include Webpacker::Helper
  end
end

Instance Method Details

#controller_instance_variablesObject



136
137
138
139
140
141
142
143
144
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 136

def controller_instance_variables
  instance_vars = @controller.instance_variables.inject({}) do |vars, v|
    k = v.to_s.sub(/^@/,'') # @var => var
    vars[k] = @controller.instance_variable_get(v)
    vars
  end
  instance_vars[:event] = event
  instance_vars
end

#default_template_nameObject

Example: posts/index



42
43
44
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 42

def default_template_name
  "#{template_namespace}/#{@controller.meth}"
end

#rackify_headers(headers) ⇒ Object

Takes headers and adds HTTP_ to front of the keys because that is what rack does to the headers passed from a request. This seems to be the standard when testing with curl and inspecting the headers in a Rack app. Example: gist.github.com/tongueroo/94f22f6c261c8999e4f4f776547e2ee3

This is useful for:

ActionController::Base.renderer.new(renderer_options)

renderer_options are rack normalized headers.

Example input (from api gateway)

{"host"=>"localhost:8888",
"user-agent"=>"curl/7.53.1",
"accept"=>"*/*",
"version"=>"HTTP/1.1",
"x-amzn-trace-id"=>"Root=1-5bde5b19-61d0d4ab4659144f8f69e38f"}

Example output:

{"HTTP_HOST"=>"localhost:8888",
"HTTP_USER_AGENT"=>"curl/7.53.1",
"HTTP_ACCEPT"=>"*/*",
"HTTP_VERSION"=>"HTTP/1.1",
"HTTP_X_AMZN_TRACE_ID"=>"Root=1-5bde5b19-61d0d4ab4659144f8f69e38f"}


103
104
105
106
107
108
109
110
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 103

def rackify_headers(headers)
  results = {}
  headers.each do |k,v|
    rack_key = 'HTTP_' + k.gsub('-','_').upcase
    results[rack_key] = v
  end
  results
end

#renderObject

Example response:

[200, {"my-header" = > "value" }, "my body" ]

Returns rack triplet



18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 18

def render
  # we do some normalization here
  status = normalize_status_code(@options[:status])

  base64 = normalized_base64_option(@options)

  headers = @options[:headers] || {}
  set_content_type!(status, headers)
  # x-jets-base64 to convert this Rack triplet to a API Gateway hash structure later
  headers["x-jets-base64"] = base64 ? 'yes' : 'no' # headers values must be Strings

  # Rails rendering does heavy lifting
  if (status)
    body = StringIO.new
  else
    renderer = ActionController::Base.renderer.new(renderer_options)
    body = renderer.render(render_options)
    body = StringIO.new(body)
  end

  [status, headers, body] # triplet
end

#render_optionsObject



112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 112

def render_options
  # nomralize the template option
  template = @options[:template]
  if template and !template.include?('/')
    template = "#{template_namespace}/#{template}"
  end
  template ||= default_template_name
  # ready to override @options[:template]
  @options[:template] = template if @options[:template]

  render_options = {
    template: template, # weird: template needs to be set no matter because it
      # sets the name which is used in lookup_context.rb:209:in `normalize_name'
    layout: @options[:layout],
    assigns: controller_instance_variables,
  }
  types = %w[json inline plain file xml body action].map(&:to_sym)
  types.each do |type|
    render_options[type] = @options[type] if @options[type]
  end

  render_options
end

#renderer_optionsObject

default options:

https://github.com/rails/rails/blob/master/actionpack/lib/action_controller/renderer.rb#L41-L47


53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 53

def renderer_options
  options = {
    # script_name: "", # unfortunately doesnt seem to effect relative_url_root like desired
    # input: ""
  }

  origin = headers["origin"]
  if origin
    uri = URI.parse(origin)
    options[:https] = uri.scheme == "https"
  end

  # Important to not use rack_headers as local variable instead of headers.
  # headers is a method that gets deleted to controller.headers and using it
  # seems to cause issues.
  rack_headers = rackify_headers(headers)
  options.merge!(rack_headers)

  # Note @options[:method] uses @options vs options on purpose
  @options[:method] = event["httpMethod"].downcase if event["httpMethod"]
  options
end

#template_namespaceObject

PostsController => “posts” is the namespace



47
48
49
# File 'lib/jets/controller/rendering/rack_renderer.rb', line 47

def template_namespace
  @controller.class.to_s.sub('Controller','').underscore.pluralize
end