class Bullet::Rack

def append_to_html_body(response_body, content)

def append_to_html_body(response_body, content)
  body = response_body.dup
  content = content.html_safe if content.respond_to?(:html_safe)
  if body.include?('</body>')
    position = body.rindex('</body>')
    body.insert(position, content)
  else
    body << content
  end
end

def call(env)

def call(env)
  return @app.call(env) unless Bullet.enable?
  Bullet.start_request
  status, headers, response = @app.call(env)
  response_body = nil
  if Bullet.notification?
    if Bullet.inject_into_page? && !file?(headers) && !sse?(headers) && !empty?(response) && status == 200
      if html_request?(headers, response)
        response_body = response_body(response)
        with_security_policy_nonce(headers) do |nonce|
          response_body = append_to_html_body(response_body, footer_note) if Bullet.add_footer
          response_body = append_to_html_body(response_body, Bullet.gather_inline_notifications)
          if Bullet.add_footer && !Bullet.skip_http_headers
            response_body = append_to_html_body(response_body, xhr_script(nonce))
          end
        end
        headers['Content-Length'] = response_body.bytesize.to_s
      elsif !Bullet.skip_http_headers
        set_header(headers, 'X-bullet-footer-text', Bullet.footer_info.uniq) if Bullet.add_footer
        set_header(headers, 'X-bullet-console-text', Bullet.text_notifications) if Bullet.console_enabled?
      end
    end
    Bullet.perform_out_of_channel_notifications(env)
  end
  [status, headers, response_body ? [response_body] : response]
ensure
  Bullet.end_request
end

def details_attributes

def details_attributes
  <<~EOF
    id="bullet-footer" data-is-bullet-footer
    style="cursor: pointer; position: fixed; left: 0px; bottom: 0px; z-index: 9999; background: #fdf2f2; color: #9b1c1c; font-size: 12px; border-radius: 0px 8px 0px 0px; border: 1px solid #9b1c1c;"
  EOF
end

def empty?(response)

fix issue if response's body is a Proc
def empty?(response)
  # response may be ["Not Found"], ["Move Permanently"], etc, but
  # those should not happen if the status is 200
  return true if !response.respond_to?(:body) && !response.respond_to?(:first)
  body = response_body(response)
  body.nil? || body.empty?
end

def file?(headers)

def file?(headers)
  headers['Content-Transfer-Encoding'] == 'binary' || headers['Content-Disposition']
end

def footer_console_message

def footer_console_message
  if Bullet.console_enabled?
    "<br/><span style='font-style: italic;'>See 'Uniform Notifier' in JS Console for Stacktrace</span>"
  end
end

def footer_content_attributes

def footer_content_attributes
  <<~EOF
    style="padding: 8px; border-top: 1px solid #9b1c1c;"
  EOF
end

def footer_note

def footer_note
  "<details #{details_attributes}><summary #{summary_attributes}>Bullet Warnings</summary><div #{footer_content_attributes}>#{Bullet.footer_info.uniq.join('<br>')}#{footer_console_message}</div></details>"
end

def html_request?(headers, response)

def html_request?(headers, response)
  headers['Content-Type']&.include?('text/html')
end

def initialize(app)

def initialize(app)
  @app = app
end

def response_body(response)

def response_body(response)
  if response.respond_to?(:body)
    Array === response.body ? response.body.first : response.body
  elsif response.respond_to?(:first)
    response.first
  end
end

def set_header(headers, header_name, header_array)

def set_header(headers, header_name, header_array)
  # Many proxy applications such as Nginx and AWS ELB limit
  # the size a header to 8KB, so truncate the list of reports to
  # be under that limit
  header_array.pop while header_array.to_json.length > 8 * 1024
  headers[header_name] = header_array.to_json
end

def sse?(headers)

def sse?(headers)
  headers['Content-Type'] == 'text/event-stream'
end

def summary_attributes

def summary_attributes
  <<~EOF
    style="font-weight: 600; padding: 2px 8px"
  EOF
end

def with_security_policy_nonce(headers)

def with_security_policy_nonce(headers)
  matched = (headers['Content-Security-Policy'] || '').match(NONCE_MATCHER)
  nonce = matched[:nonce] if matched
  if nonce
    console_enabled = UniformNotifier.console
    alert_enabled = UniformNotifier.alert
    UniformNotifier.console = { attributes: { nonce: nonce } } if console_enabled
    UniformNotifier.alert = { attributes: { nonce: nonce } } if alert_enabled
    yield nonce
    UniformNotifier.console = console_enabled
    UniformNotifier.alert = alert_enabled
  else
    yield
  end
end

def xhr_script(nonce = nil)

Make footer work for XHR requests by appending data to the footer
def xhr_script(nonce = nil)
  script = File.read("#{__dir__}/bullet_xhr.js")
  if nonce
    "<script type='text/javascript' nonce='#{nonce}'>#{script}</script>"
  else
    "<script type='text/javascript'>#{script}</script>"
  end
end