# frozen_string_literal: true module Extension module Commands class Serve < ExtensionCommand DEFAULT_PORT = 39351 options do |parser, flags| parser.on("-t", "--[no-]tunnel", "Establish an ngrok tunnel") { |tunnel| flags[:tunnel] = tunnel } end class RuntimeConfiguration include SmartProperties property! :tunnel_url, accepts: String, default: "" property! :tunnel_requested, accepts: [true, false], reader: :tunnel_requested?, default: true property! :port, accepts: (1...(2**16)), default: DEFAULT_PORT end def call(_args, _command_name) config = RuntimeConfiguration.new( tunnel_requested: tunnel_requested? ) ShopifyCli::Result .success(config) .then(&method(:find_available_port)) .then(&method(:start_tunnel_if_required)) .then(&method(:serve)) .unwrap { |error| raise error } end def self.help <<~HELP Serve your extension in a local simulator for development. Usage: {{command:#{ShopifyCli::TOOL_NAME} serve}} Options: {{command:--tunnel=TUNNEL}} Establish an ngrok tunnel (default: false) HELP end private def tunnel_requested? tunnel = options.flags[:tunnel] tunnel.nil? || !!tunnel end def find_available_port(runtime_configuration) return runtime_configuration unless specification_handler.choose_port?(@ctx) chosen_port = Tasks::ChooseNextAvailablePort .call(from: runtime_configuration.port) .unwrap { |_error| @ctx.abort(@ctx.message("serve.no_available_ports_found")) } runtime_configuration.tap { |c| c.port = chosen_port } end def start_tunnel_if_required(runtime_configuration) return runtime_configuration unless specification_handler.establish_tunnel?(@ctx) return runtime_configuration unless runtime_configuration.tunnel_requested? return start_tunnel(runtime_configuration) if can_start_tunnel?(runtime_configuration) @ctx.abort(@ctx.message("serve.tunnel_already_running")) end def can_start_tunnel?(runtime_configuration) return true if ShopifyCli::Tunnel.urls.empty? ShopifyCli::Tunnel.running_on?(runtime_configuration.port) end def start_tunnel(runtime_configuration) tunnel_url = ShopifyCli::Tunnel.start(@ctx, port: runtime_configuration.port) runtime_configuration.tap { |c| c.tunnel_url = tunnel_url } end def serve(runtime_configuration) specification_handler.serve( context: @ctx, tunnel_url: runtime_configuration.tunnel_url, port: runtime_configuration.port ) runtime_configuration end end end end