Class: VagrantPlugins::VCloud::Action::HandleNATPortCollisions
- Inherits:
-
Object
- Object
- VagrantPlugins::VCloud::Action::HandleNATPortCollisions
- Defined in:
- lib/vagrant-vcloud/action/handle_nat_port_collisions.rb
Overview
This middleware class will detect and handle collisions with forwarded ports, whether that means raising an error or repairing them automatically.
Parameters it takes from the environment hash:
* `:port_collision_repair` - If true, it will attempt to repair
port collisions. If false, it will raise an exception when
there is a collision.
* `:port_collision_extra_in_use` - An array of ports that are
considered in use.
* `:port_collision_remap` - A hash remapping certain host ports
to other host ports.
Instance Method Summary collapse
- #call(env) ⇒ Object
-
#initialize(app, env) ⇒ HandleNATPortCollisions
constructor
A new instance of HandleNATPortCollisions.
Constructor Details
#initialize(app, env) ⇒ HandleNATPortCollisions
Returns a new instance of HandleNATPortCollisions.
23 24 25 26 27 28 |
# File 'lib/vagrant-vcloud/action/handle_nat_port_collisions.rb', line 23 def initialize(app, env) @app = app @logger = Log4r::Logger.new( 'vagrant_vcloud::action::handle_port_collisions' ) end |
Instance Method Details
#call(env) ⇒ Object
30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 |
# File 'lib/vagrant-vcloud/action/handle_nat_port_collisions.rb', line 30 def call(env) @logger.info('Detecting any forwarded port collisions...') # Determine a list of usable ports for repair usable_ports = Set.new(env[:machine].config.vm.usable_port_range) # Pass one, remove all defined host ports from usable ports with_forwarded_ports(env) do || usable_ports.delete([:host]) end cfg = env[:machine].provider_config cnx = cfg.vcloud_cnx.driver vapp_id = env[:machine].get_vapp_id @logger.debug('Getting VM info...') vm_name = cfg.name ? cfg.name.to_sym : env[:machine].name vm = cnx.get_vapp(vapp_id) vm_info = vm[:vms_hash][vm_name.to_sym] network_name = get_edge_network_name(env) vapp_edge_ip = cnx.get_vapp_edge_public_ip(vapp_id, network_name) @logger.debug('Getting edge gateway port forwarding rules...') edge_gateway_rules = cnx.get_edge_gateway_rules(cfg.vdc_edge_gateway, cfg.vdc_id) edge_dnat_rules = edge_gateway_rules.select {|r| (r[:rule_type] == 'DNAT' && r[:translated_ip] != vapp_edge_ip)} edge_ports_in_use = edge_dnat_rules.map{|r| r[:original_port].to_i}.to_set @logger.debug('Getting port forwarding rules...') vapp_nat_rules = cnx.get_vapp_port_forwarding_rules(vapp_id, network_name) ports_in_use = vapp_nat_rules.map{|r| r[:nat_external_port].to_i}.to_set # merge the vapp ports and the edge gateway ports together, all are in use ports_in_use = ports_in_use | edge_ports_in_use # Pass two, detect/handle any collisions with_forwarded_ports(env) do || guest_port = [:guest] host_port = [:host] # Find if there already is a NAT rule to guest_port of this VM if r = vapp_nat_rules.find { |rule| (rule[:vapp_scoped_local_id] == vm_info[:vapp_scoped_local_id] && rule[:nat_internal_port] == guest_port.to_s) } host_port = r[:nat_external_port].to_i @logger.info( "Found existing port forwarding rule #{host_port} to #{guest_port}" ) [:host] = host_port [:already_exists] = true else # If the port is open (listening for TCP connections) if ports_in_use.include?(host_port) if ![:auto_correct] raise Errors::ForwardPortCollision, :guest_port => guest_port.to_s, :host_port => host_port.to_s end @logger.info("Attempting to repair FP collision: #{host_port}") repaired_port = nil while !usable_ports.empty? # Attempt to repair the forwarded port repaired_port = usable_ports.to_a.sort[0] usable_ports.delete(repaired_port) # If the port is in use, then we can't use this either... if ports_in_use.include?(repaired_port) @logger.info( "Repaired port also in use: #{repaired_port}." + 'Trying another...' ) next end # We have a port so break out break end # If we have no usable ports then we can't repair if !repaired_port && usable_ports.empty? raise Errors::ForwardPortAutolistEmpty, :vm_name => env[:machine].name, :guest_port => guest_port.to_s, :host_port => host_port.to_s end # Modify the args in place [:host] = repaired_port @logger.info( "Repaired FP collision: #{host_port} to #{repaired_port}" ) # Notify the user env[:ui].info( I18n.t( 'vagrant.actions.vm.forward_ports.fixed_collision', :host_port => host_port.to_s, :guest_port => guest_port.to_s, :new_port => repaired_port.to_s ) ) end end end @app.call(env) end |