# frozen_string_literal: true

require "uri"
require "net/http"
require "rack"
require_relative "initializer_hooks"
require_relative "server/middleware"
require_relative "server/checker"
require_relative "server/timer"
require_relative "server/puma"

module CypressRails
  class Server
    class << self
      def ports
        @ports ||= {}
      end
    end

    attr_reader :app, :host, :port

    def initialize(app,
      host:,
      port:,
      reportable_errors: [Exception],
      extra_middleware: [])
      @app = app
      @extra_middleware = extra_middleware
      @server_thread = nil # suppress warnings
      @host = host
      @reportable_errors = reportable_errors
      @port = port
      @port ||= Server.ports[port_key]
      @port ||= find_available_port(host)
      @checker = Checker.new(@host, @port)
      @initializer_hooks = InitializerHooks.instance
    end

    def reset_error!
      middleware.clear_error
    end

    def error
      middleware.error
    end

    def using_ssl?
      @checker.ssl?
    end

    def responsive?
      return false if @server_thread&.join(0)

      res = @checker.request { |http| http.get("/__identify__") }

      res.body == app.object_id.to_s if res.is_a?(Net::HTTPSuccess) || res.is_a?(Net::HTTPRedirection)
    rescue SystemCallError, Net::ReadTimeout, OpenSSL::SSL::SSLError
      false
    end

    def wait_for_pending_requests
      timer = Timer.new(60)
      while pending_requests?
        raise "Requests did not finish in 60 seconds: #{middleware.pending_requests}" if timer.expired?

        sleep 0.01
      end
    end

    def boot
      unless responsive?
        Server.ports[port_key] = port

        @server_thread = Thread.new {
          Puma.create(middleware, port, host)
        }

        timer = Timer.new(60)
        until responsive?
          raise "Rack application timed out during boot" if timer.expired?

          @server_thread.join(0.1)
          @initializer_hooks.run(:after_server_start)
        end
      end

      self
    end

    private

    def middleware
      @middleware ||= Middleware.new(app, @reportable_errors, @extra_middleware)
    end

    def port_key
      app.object_id # as opposed to middleware.object_id if multiple instances
    end

    def pending_requests?
      middleware.pending_requests?
    end

    def find_available_port(host)
      server = TCPServer.new(host, 0)
      port = server.addr[1]
      server.close

      # Workaround issue where some platforms (mac, ???) when passed a host
      # of '0.0.0.0' will return a port that is only available on one of the
      # ip addresses that resolves to, but the next binding to that port requires
      # that port to be available on all ips
      server = TCPServer.new(host, port)
      port
    rescue Errno::EADDRINUSE
      retry
    ensure
      server&.close
    end
  end
end