class Opal::CLI

def builder

def builder
  @builder ||= create_builder
end

def compiler_option_names

def compiler_option_names
  %w[
    method_missing
    arity_check
    dynamic_require_severity
    source_map_enabled
    irb_enabled
    inline_operators
    enable_source_location
    parse_comments
  ]
end

def create_builder

def create_builder
  builder = Opal::Builder.new(
    stubs: stubs,
    compiler_options: compiler_options,
    missing_require_severity: missing_require_severity,
  )
  # --include
  builder.append_paths(*load_paths)
  # --gem
  gems.each { |gem_name| builder.use_gem gem_name }
  # --require
  requires.each { |required| builder.build(required) }
  # --preload
  preload.each { |path| builder.build_require(path) }
  # --verbose
  builder.build_str '$VERBOSE = true', '(flags)' if verbose
  # --debug
  builder.build_str '$DEBUG = true', '(flags)' if debug
  # --eval / stdin / file
  evals_or_file { |source, filename| builder.build_str(source, filename) }
  # --no-exit
  builder.build_str 'Kernel.exit', '(exit)' unless no_exit
  builder
end

def evals_or_file

evals, stdin or a filepath.
Internal: Yields a string of source code and the proper filename for either
def evals_or_file
  # --library
  return if lib_only
  if evals.any?
    yield evals.join("\n"), '-e'
  else
    if file and (filename != '-' or evals.empty?)
      yield file.read, filename
    end
  end
end

def initialize options = nil

def initialize options = nil
  options ||= {}
  # Runner
  @runner_type = options.delete(:runner)     || :nodejs
  @port        = options.delete(:port)       || 3000
  @options     = options
  @compile     = !!options.delete(:compile)
  @sexp        = options.delete(:sexp)
  @file        = options.delete(:file)
  @map         = options.delete(:map)
  @no_exit     = options.delete(:no_exit)
  @lib_only    = options.delete(:lib_only)
  @argv        = options.delete(:argv)       || []
  @evals       = options.delete(:evals)      || []
  @load_paths  = options.delete(:load_paths) || []
  @gems        = options.delete(:gems)       || []
  @stubs       = options.delete(:stubs)      || []
  @preload     = options.delete(:preload)    || []
  @output      = options.delete(:output)     || self.class.stdout || $stdout
  @verbose     = options.fetch(:verbose, false); options.delete(:verbose)
  @debug       = options.fetch(:debug, false);   options.delete(:debug)
  @filename    = options.fetch(:filename) { @file && @file.path }; options.delete(:filename)
  @requires    = options.delete(:requires)   || []
  @missing_require_severity = options.delete(:missing_require_severity) { Opal::Config.missing_require_severity }
  @requires.unshift('opal') unless options.delete(:skip_opal_require)
  @compiler_options = Hash[
    *compiler_option_names.map do |option|
      key = option.to_sym
      next unless options.has_key? key
      value = options.delete(key)
      [key, value]
    end.compact.flatten
  ]
  raise ArgumentError, "no libraries to compile" if @lib_only and @requires.length == 0
  raise ArgumentError, "no runnable code provided (evals or file)" if @evals.empty? and @file.nil? and not(@lib_only)
  raise ArgumentError, "can't accept evals or file in `library only` mode" if (@evals.any? or @file) and @lib_only
  raise ArgumentError, "unknown options: #{options.inspect}" unless @options.empty?
end

def puts(*args)

def puts(*args)
  output.puts(*args)
end

def run

def run
  return show_sexp if @sexp
  compiled_source = builder.to_s
  File.write @map, builder.source_map if @map
  return puts compiled_source if @compile
  runner.run(compiled_source, argv)
  @exit_status = runner.exit_status
end

def runner

def runner
  @runner ||= begin
    const_name = @runner_type.to_s.capitalize
    CliRunners.const_defined?(const_name) or
      raise ArgumentError, "unknown runner: #{@runner_type.inspect}"
    CliRunners.const_get(const_name).new(output: output, port: port)
  end
end

def show_sexp

def show_sexp
  evals_or_file do |contents, filename|
    buffer = ::Opal::Source::Buffer.new(filename)
    buffer.source = contents
    sexp = Opal::Parser.default_parser.parse(buffer)
    puts sexp.inspect
  end
end