Alex Ruiz Estradera 24a60a49d9 Fix the neutron-port resource with binding options
Parameters of type dict must be at the end of the
shell command to not cause conflict with the network name

current shell generated

/usr/bin/neutron port-create --format=shell --name=testport \
--fixed-ip ip_address=172.19.0.2 --binding:host_id=aio \
--binding:profile type=dict interface_name=veth1 net-edge1-gw1

this fails since it matches the first word without "--" to the network
name

correct call

/usr/bin/neutron port-create --format=shell --name=testport \
--fixed-ip ip_address=172.19.0.2 --binding:host_id=aio net-edge1-gw1 \
--binding:profile type=dict interface_name=veth1

Change-Id: Iaa24d52f4f42e0535ad2d0aed1c0a67178af091a
2016-08-25 21:33:01 +00:00

212 lines
5.9 KiB
Ruby

require File.join(File.dirname(__FILE__), "..","..","..",
"puppet/provider/neutron")
Puppet::Type.type(:neutron_port).provide(
:neutron,
:parent => Puppet::Provider::Neutron
) do
desc <<-EOT
Neutron provider to manage neutron_port type.
Assumes that the neutron service is configured on the same host.
EOT
#TODO No security group support
mk_resource_methods
def self.instances
list_neutron_resources("port").collect do |id|
attrs = get_neutron_resource_attrs("port", id)
attrs["name"] = attrs["id"] if attrs["name"].empty?
new(
:ensure => :present,
:name => attrs["name"],
:id => attrs["id"],
:status => attrs["status"],
:tenant_id => attrs["tenant_id"],
:network_id => attrs["network_id"],
:admin_state_up => attrs["admin_state_up"],
:network_name => get_network_name(attrs["network_id"]),
:subnet_name => get_subnet_name(parse_subnet_id(attrs["fixed_ips"])),
:subnet_id => parse_subnet_id(attrs["fixed_ips"]),
:ip_address => parse_ip_address(attrs["fixed_ips"]),
:binding_profile => parse_binding_profile_interface_name(attrs["binding:profile"]),
:binding_host_id => attrs["binding:host_id"],
)
end
end
def self.prefetch(resources)
instances_ = instances
resources.keys.each do |name|
if provider = instances_.find{ |instance| instance.name == name }
resources[name].provider = provider
end
end
end
def exists?
@property_hash[:ensure] == :present
end
def create
opts = Array.new
dict_opts = Array.new
if @resource[:admin_state_up] == "False"
opts << "--admin-state-down"
end
if @resource[:ip_address]
# The spec says that multiple ip addresses may be specified, but this
# doesn't seem to work yet.
opts << "--fixed-ip"
opts << resource[:ip_address].map{|ip|"ip_address=#{ip}"}.join(',')
end
if @resource[:subnet_name]
# The spec says that multiple subnets may be specified, but this doesn't
# seem to work yet.
opts << "--fixed-ip"
opts << resource[:subnet_name].map{|s|"subnet_id=#{s}"}.join(',')
end
if @resource[:tenant_name]
tenant_id = self.class.get_tenant_id(
@resource.catalog,
@resource[:tenant_name]
)
opts << "--tenant_id=#{tenant_id}"
elsif @resource[:tenant_id]
opts << "--tenant_id=#{@resource[:tenant_id]}"
end
if @resource[:binding_host_id]
opts << "--binding:host_id=#{@resource[:binding_host_id]}"
end
if @resource[:binding_profile]
binding_profile_opts = @resource[:binding_profile].map{|k,v| "#{k}=#{v}"}.join(' ')
dict_opts << "--binding:profile"
dict_opts << "type=dict"
dict_opts << "#{binding_profile_opts}"
end
results = auth_neutron(
"port-create",
"--format=shell",
"--name=#{resource[:name]}",
opts,
resource[:network_name],
dict_opts
)
attrs = self.class.parse_creation_output(results)
@property_hash = {
:ensure => :present,
:name => resource[:name],
:id => attrs["id"],
:status => attrs["status"],
:tenant_id => attrs["tenant_id"],
:network_id => attrs["network_id"],
:admin_state_up => attrs["admin_state_up"],
:network_name => resource[:network_name],
:subnet_name => resource[:subnet_name],
:subnet_id => self.class.parse_subnet_id(attrs["fixed_ips"]),
:ip_address => self.class.parse_ip_address(attrs["fixed_ips"]),
:binding_profile => self.class.parse_binding_profile_interface_name(attrs["binding:profile"]),
:binding_host_id => attrs["binding:host_id"],
}
end
def destroy
auth_neutron("port-delete", name)
@property_hash[:ensure] = :absent
end
def admin_state_up=(value)
auth_neutron("port-update", "--admin-state-up=#{value}", name)
end
private
def self.get_network_name(network_id_)
if network_id_
network_instances = Puppet::Type.type("neutron_network").instances
network_name = network_instances.find do |instance|
instance.provider.id == network_id_
end.provider.name
end
network_name
end
def get_network_name(network_id_)
@get_network_name ||= self.class.get_network_name(network_id_)
end
def self.get_subnet_name(subnet_id_)
if subnet_id_
subnet_ids = Array(subnet_id_)
subnet_instances = Puppet::Type.type("neutron_subnet").instances
subnet_names = subnet_instances.collect do |instance|
if subnet_ids.include?(instance.provider.id)
instance.provider.name
else
nil
end
end.compact
if subnet_names.length > 1
subnet_names
else
subnet_names.first
end
end
end
def get_subnet_name(subnet_id_)
@subnet_name ||= self.class.subnet_name(subnet_id_)
end
def self.parse_subnet_id(fixed_ips_)
subnet_ids = Array(fixed_ips_).collect do |json|
match_data = /\{"subnet_id": "(.*)", /.match(json)
if match_data
match_data[1]
else
nil
end
end.compact
if subnet_ids.length > 1
subnet_ids
else
subnet_ids.first
end
end
def self.parse_ip_address(fixed_ips_)
ip_addresses = Array(fixed_ips_).collect do |json|
match_data = /"ip_address": "(.*)"\}/.match(json)
if match_data
match_data[1]
else
nil
end
end.compact
if ip_addresses.length > 1
ip_addresses
else
ip_addresses.first
end
end
def self.parse_binding_profile_interface_name(binding_profile_)
match_data = /\{"interface_name": "(.*)"\}/.match(binding_profile_)
if match_data
match_data[1]
else
nil
end
end
end