diff --git a/.sync.yml b/.sync.yml index 0fee9dfd..65f7cb48 100644 --- a/.sync.yml +++ b/.sync.yml @@ -6,6 +6,8 @@ spec/spec_helper.rb: spec_overrides: "require 'spec_helper_methods'" Gemfile: optional: + ':development': + - gem: 'puppet-resource_api' ':test': - gem: 'ipaddress' - gem: 'rspec-its' diff --git a/Gemfile b/Gemfile index ca00eb53..dee52823 100644 --- a/Gemfile +++ b/Gemfile @@ -11,7 +11,7 @@ def location_for(place, fake_version = nil) end group :test do - gem 'puppetlabs_spec_helper', '~> 2.5.0', :require => false + gem 'puppetlabs_spec_helper', '~> 2.9.1', :require => false gem 'rspec-puppet', '~> 2.5', :require => false gem 'rspec-puppet-facts', :require => false gem 'rspec-puppet-utils', :require => false @@ -31,6 +31,7 @@ group :test do gem 'rack', '~> 1.0', :require => false if RUBY_VERSION < '2.2.2' gem 'parallel_tests', :require => false gem 'ipaddress', :require => false + gem 'net-ip', :require => false gem 'rspec-its', :require => false end @@ -39,6 +40,7 @@ group :development do gem 'travis-lint', :require => false gem 'guard-rake', :require => false gem 'overcommit', '>= 0.39.1', :require => false + gem 'puppet-resource_api', :require => false end group :system_tests do diff --git a/lib/facter/network.rb b/lib/facter/network.rb deleted file mode 100644 index 136d62da..00000000 --- a/lib/facter/network.rb +++ /dev/null @@ -1,68 +0,0 @@ -require 'facter' -require 'open-uri' -require 'timeout' - -# Facter 3 has new facts that we can use instead of trying to do it -# ourselves -def facter_3 - facter_version = Gem::Version.new(Facter.version) - version3 = Gem::Version.new('3.0.0') - facter_version >= version3 -end - -# Gateway -# Expected output: The ip address of the nexthop/default router -Facter.add('network_nexthop_ip') do - my_gw = nil - confine :kernel => :linux # rubocop:disable Style/HashSyntax - setcode do - gw_address = Facter::Util::Resolution.exec('ip route show 0/0') - # not all network configurations will have a nexthop. - # the ip tool expresses the presence of a nexthop with the word 'via' - if gw_address.include? ' via ' - my_gw = gw_address.split(%r{\s+})[2].match(%r{^(?:[0-9]{1,3}\.){3}[0-9]{1,3}$}).to_s - end - my_gw - end -end - -# Primary interface -# Expected output: The specific interface name that the node uses to communicate with the nexthop -Facter.add('network_primary_interface') do - confine :kernel => :linux # rubocop:disable Style/HashSyntax - setcode do - next Facter.fact(:networking).value['primary'] if facter_3 - gw_address = Facter::Util::Resolution.exec('ip route show 0/0') - # not all network configurations will have a nexthop. - # the ip tool expresses the presence of a nexthop with the word 'via' - if gw_address.include? ' via ' - my_gw = gw_address.split(%r{\s+})[2].match(%r{^(?:[0-9]{1,3}\.){3}[0-9]{1,3}$}).to_s - fun = Facter::Util::Resolution.exec("ip route get #{my_gw}").split("\n")[0] - # some network configurations simply have a link that all interactions are abstracted through - elsif gw_address.include? 'scope link' - # since we have no default route ip to determine where to send 'traffic not otherwise explicitly routed' - # lets just use 8.8.8.8 as far as a route goes. - fun = Facter::Util::Resolution.exec('ip route get 8.8.8.8').split("\n")[0] - end - fun.split(%r{dev\s+([^\s]*)\s+src\s+([^\s]*)})[1].to_s - end -end - -# Primary IP -# Expected output: The ipaddress configured on the interface that communicates with the nexthop -Facter.add('network_primary_ip') do - confine :kernel => :linux # rubocop:disable Style/HashSyntax - setcode do - next Facter.fact(:networking).value['ip'] if facter_3 - gw_address = Facter::Util::Resolution.exec('ip route show 0/0') - if gw_address.include? ' via ' - my_gw = gw_address.split(%r{\s+})[2].match(%r{^(?:[0-9]{1,3}\.){3}[0-9]{1,3}$}).to_s - fun = Facter::Util::Resolution.exec("ip route get #{my_gw}").split("\n")[0] - elsif gw_address.include? 'scope link' - # since we have no default route ip to determine where to send 'traffic not otherwise explicitly routed' - # lets just use 8.8.8.8 as far as a route goes and grab our IP from there. - fun = Facter::Util::Resolution.exec('ip route get 8.8.8.8').split("\n")[0] - end - fun.split(%r{dev\s+([^\s]*)\s+src\s+([^\s]*)})[2].to_s - end -end diff --git a/lib/puppet/provider/network_route/network_route.rb b/lib/puppet/provider/network_route/network_route.rb new file mode 100644 index 00000000..5e971c0c --- /dev/null +++ b/lib/puppet/provider/network_route/network_route.rb @@ -0,0 +1,95 @@ +require 'net/ip' +require 'puppet/resource_api/simple_provider' + +# Implementation for the network_route type using the Resource API. +class Puppet::Provider::NetworkRoute::NetworkRoute + # include PuppetX::FileMapper + + def routes_list + routes = [] + Net::IP.routes.each do |route| + routes.push(route.to_h) + end + routes + end + + def get(_context) + routes_list.map do |route| + default = if route[:prefix] == 'default' + true + else + false + end + + { + ensure: 'present', + prefix: route[:prefix], + default_route: default, + gateway: route[:via], + interface: route[:dev], + metric: route[:metric], + table: route[:table], + source: route[:src], + scope: route[:scope], + protocol: route[:proto], + mtu: route[:mtu] + }.compact! + end + end + + def puppet_munge(should) + should.delete(:ensure) + if should[:default_route] + should[:prefix] = 'default' + should.delete(:default_route) + else + should[:prefix] = should.delete(:prefix) + end + should[:via] = should.delete(:gateway) if should[:gateway] + should[:dev] = should.delete(:interface) if should[:interface] + should[:metric] = should.delete(:metric) + should[:table] = should.delete(:table) if should[:table] + should[:src] = should.delete(:source) if should[:source] + should[:scope] = should.delete(:scope) if should[:scope] + should[:proto] = should.delete(:protocol) + should[:mtu] = should.delete(:mtu) if should[:mtu] + should + end + + def set(context, changes) + changes.each do |name, change| + is = change.key?(:is) ? change[:is] : get_single(name) + should = change[:should] + + is = { prefix: name, ensure: 'absent' } if is.nil? + should = { prefix: name, ensure: 'absent' } if should.nil? + + if is[:ensure].to_s == 'absent' && should[:ensure].to_s == 'present' + create(context, name, should) + elsif is[:ensure] == should[:ensure] && is != should + update(context, name, should) + elsif is[:ensure].to_s == 'present' && should[:ensure].to_s == 'absent' + delete(context, name, should) + end + end + end + + def create(context, name, should) + puppet_munge(should) + route = Net::IP::Route.new(should) + Net::IP.routes.add(route) + end + + def update(context, name, should) + puppet_munge(should) + route = Net::IP::Route.new(should) + Net::IP.routes.flush(route.prefix) + Net::IP.routes.add(route) + end + + def delete(context, name, should) + puppet_munge(should) + route = Net::IP::Route.new(should) + Net::IP.routes.flush(route.prefix) + end +end diff --git a/lib/puppet/provider/network_route/redhat.rb b/lib/puppet/provider/network_route/redhat.rb deleted file mode 100644 index 163e0f96..00000000 --- a/lib/puppet/provider/network_route/redhat.rb +++ /dev/null @@ -1,104 +0,0 @@ -require 'ipaddr' -require 'puppetx/filemapper' - -Puppet::Type.type(:network_route).provide(:redhat) do - # RHEL network_route routes provider. - # - # This provider uses the filemapper mixin to map the routes file to a - # collection of network_route providers, and back. - # - # @see https://access.redhat.com/knowledge/docs/en-US/Red_Hat_Enterprise_Linux/6/html/Deployment_Guide/s1-networkscripts-static-routes.html - - include PuppetX::FileMapper - - desc 'RHEL style routes provider' - - confine osfamily: :redhat - defaultfor osfamily: :redhat - - has_feature :provider_options - - def select_file - "/etc/sysconfig/network-scripts/route-#{@resource[:interface]}" - end - - def self.target_files - Dir['/etc/sysconfig/network-scripts/route-*'] - end - - def self.parse_file(_filename, contents) - routes = [] - - lines = contents.split("\n") - lines.each do |line| - # Strip off any trailing comments - line.sub!(%r{#.*$}, '') - - if line =~ %r{^\s*#|^\s*$} - # Ignore comments and blank lines - next - end - - route = line.split(' ', 6) - if route.length < 4 - raise Puppet::Error, 'Malformed redhat route file, cannot instantiate network_route resources' - end - - new_route = {} - - new_route[:gateway] = route[2] - new_route[:interface] = route[4] - new_route[:options] = route[5] if route[5] - - if route[0] == 'default' - cidr_target = 'default' - - new_route[:name] = cidr_target - new_route[:network] = 'default' - new_route[:netmask] = '0.0.0.0' - else - # use the CIDR version of the target as :name - network, netmask = route[0].split('/') - cidr_target = "#{network}/#{IPAddr.new(netmask).to_i.to_s(2).count('1')}" - - new_route[:name] = cidr_target - new_route[:network] = network - new_route[:netmask] = netmask - end - - routes << new_route - end - - routes - end - - # Generate an array of sections - def self.format_file(_filename, providers) - contents = [] - contents << header - # Build routes - providers.sort_by(&:name).each do |provider| - [:network, :netmask, :gateway, :interface].each do |prop| - raise Puppet::Error, "#{provider.name} does not have a #{prop}." if provider.send(prop).nil? - end - contents << if provider.network == 'default' - "#{provider.network} via #{provider.gateway} dev #{provider.interface}" - else - "#{provider.network}/#{provider.netmask} via #{provider.gateway} dev #{provider.interface}" - end - contents << (provider.options == :absent ? "\n" : " #{provider.options}\n") - end - contents.join - end - - def self.header - str = <<-HEADER -# HEADER: This file is being managed by puppet. Changes to -# HEADER: routes that are not being managed by puppet will persist; -# HEADER: however changes to routes that are being managed by puppet will -# HEADER: be overwritten. In addition, file order is NOT guaranteed. -# HEADER: Last generated at: #{Time.now} -HEADER - str - end -end diff --git a/lib/puppet/provider/network_route/routes.rb b/lib/puppet/provider/network_route/routes.rb deleted file mode 100644 index 290bb4d7..00000000 --- a/lib/puppet/provider/network_route/routes.rb +++ /dev/null @@ -1,118 +0,0 @@ -require 'ipaddr' -require 'puppetx/filemapper' - -Puppet::Type.type(:network_route).provide(:routes) do - # Debian network_route routes provider. - # - # This provider uses the filemapper mixin to map the routes file to a - # collection of network_route providers, and back. - # - # @see http://wiki.debian.org/NetworkConfiguration - # @see http://packages.debian.org/squeeze/ifupdown-extras - - include PuppetX::FileMapper - - desc 'Debian routes style provider' - - confine osfamily: :debian - - # $ dpkg -S /etc/network/if-up.d/20static-routes - # ifupdown-extra: /etc/network/if-up.d/20static-routes - confine exists: '/etc/network/if-up.d/20static-routes' - - defaultfor osfamily: :debian - - has_feature :provider_options - - def select_file - '/etc/network/routes' - end - - def self.target_files - ['/etc/network/routes'] - end - - class MalformedRoutesError < Puppet::Error - def initialize(msg = nil) - msg = 'Malformed debian routes file; cannot instantiate network_route resources' if msg.nil? - super - end - end - - def self.raise_malformed - @failed = true - raise MalformedRoutesError - end - - def self.parse_file(_filename, contents) - # Build out an empty hash for new routes for storing their configs. - route_hash = Hash.new do |hash, key| - hash[key] = {} - hash[key][:name] = key - hash[key] - end - - lines = contents.split("\n") - lines.each do |line| - # Strip off any trailing comments - line.sub!(%r{#.*$}, '') - - if line =~ %r{^\s*#|^\s*$} - # Ignore comments and blank lines - next - end - - route = line.split(' ', 5) - - raise_malformed if route.length < 4 - - if route[0] == 'default' - name = 'default' - route_hash[name][:network] = 'default' - route_hash[name][:netmask] = '0.0.0.0' - else - # use the CIDR version of the target as :name - name = "#{route[0]}/#{IPAddr.new(route[1]).to_i.to_s(2).count('1')}" - route_hash[name][:network] = route[0] - route_hash[name][:netmask] = route[1] - end - route_hash[name][:gateway] = route[2] - route_hash[name][:interface] = route[3] - route_hash[name][:options] = route[4] if route[4] - end - - route_hash.values - end - - # Generate an array of sections - def self.format_file(_filename, providers) - contents = [] - contents << header - - # Build routes - providers.sort_by(&:name).each do |provider| - raise Puppet::Error, "#{provider.name} is missing the required parameter 'network'." if provider.network.nil? - raise Puppet::Error, "#{provider.name} is missing the required parameter 'netmask'." if provider.netmask.nil? - raise Puppet::Error, "#{provider.name} is missing the required parameter 'gateway'." if provider.gateway.nil? - raise Puppet::Error, "#{provider.name} is missing the required parameter 'interface'." if provider.interface.nil? - - netmask = (provider.name == 'default' ? '0.0.0.0' : provider.netmask) - - contents << "#{provider.network} #{netmask} #{provider.gateway} #{provider.interface}" - contents << (provider.options == :absent ? "\n" : " #{provider.options}\n") - end - - contents.join - end - - def self.header - str = <<-HEADER -# HEADER: This file is being managed by puppet. Changes to -# HEADER: routes that are not being managed by puppet will persist; -# HEADER: however changes to routes that are being managed by puppet will -# HEADER: be overwritten. In addition, file order is NOT guaranteed. -# HEADER: Last generated at: #{Time.now} -HEADER - str - end -end diff --git a/lib/puppet/type/network_route.rb b/lib/puppet/type/network_route.rb index 03289264..b4ff9ed3 100644 --- a/lib/puppet/type/network_route.rb +++ b/lib/puppet/type/network_route.rb @@ -1,81 +1,58 @@ -require 'ipaddr' -require_relative '../../puppet_x/voxpupuli/utils.rb' - -Puppet::Type.newtype(:network_route) do - @doc = 'Manage non-volatile route configuration information' - - include PuppetX::Voxpupuli::Utils - - ensurable - - newparam(:name) do - isnamevar - desc 'The name of the network route' - end - - newproperty(:network) do - isrequired - desc 'The target network address' - validate do |value| - unless value == 'default' - a = PuppetX::Voxpupuli::Utils.try { IPAddr.new(value) } - raise("Invalid value for network: #{value}") unless a - end - end - end - - newproperty(:netmask) do - isrequired - desc 'The subnet mask to apply to the route' - - validate do |value| - unless value.length <= 3 || PuppetX::Voxpupuli::Utils.try { IPAddr.new(value) } - raise("Invalid value for argument netmask: #{value}") - end - end - - munge do |value| - # '255.255.255.255'.to_i will return 255, so we try to convert it back: - if value.to_i.to_s == value - # what are the chances someone is using /16 for their IPv6 network? - addr = value.to_i <= 32 ? '255.255.255.255' : 'ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff' - IPAddr.new(addr).mask(value.strip.to_i).to_s - elsif PuppetX::Voxpupuli::Utils.try { IPAddr.new(value).ipv6? } - IPAddr.new('ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff').mask(value).to_s - elsif PuppetX::Voxpupuli::Utils.try { IPAddr.new(value).ipv4? } - IPAddr.new('255.255.255.255').mask(value).to_s - else - raise("Invalid value for argument netmask: #{value}") - end - end - end - - newproperty(:gateway) do - isrequired - desc 'The gateway to use for the route' - - validate do |value| - begin - IPAddr.new(value) - rescue ArgumentError - raise("Invalid value for gateway: #{value}") - end - end - end - - newproperty(:interface) do - isrequired - desc 'The interface to use for the route' - end - - # `:options` provides an arbitrary passthrough for provider properties, so - # that provider specific behavior doesn't clutter up the main type but still - # allows for more powerful actions to be taken. - newproperty(:options, required_features: :provider_options) do - desc 'Provider specific options to be passed to the provider' - - validate do |value| - raise ArgumentError, "#{self.class} requires a string for the options property" unless value.is_a?(String) - end - end -end +require 'puppet/resource_api' + +Puppet::ResourceApi.register_type( + name: 'network_route', + docs: <<-EOS, + Manage non-volatile route configuration information. + EOS + attributes: { + ensure: { + type: 'Enum[present, absent]', + desc: 'Whether the network route should be present or absent on the target system.', + default: 'present', + }, + prefix: { + type: 'String', + desc: 'The destination prefix/network of the route.', + behaviour: :namevar, + }, + default_route: { + type: 'Optional[Boolean]', + desc: 'Whether the route is default or not.', + }, + gateway: { + type: 'Optional[String]', + desc: 'The gateway to use for the route.', + }, + interface: { + type: 'Optional[String]', + desc: 'The interface to use for the route.', + }, + metric: { + type: 'String', + desc: 'preference value of the route. NUMBER is an arbitrary 32bit number.', + default: '100', + }, + table: { + type: 'Optional[String]', + desc: 'table to add this route.', + }, + source: { + type: 'Optional[String]', + desc: 'the source address to prefer using when sending to the destinations covered by route prefix.', + }, + scope: { + type: 'Optional[Enum["global", "nowhere", "host", "link", "site"]]', + desc: 'scope of the destinations covered by the route prefix.', + }, + protocol: { + type: 'Enum["static", "redirect", "kernel", "boot", "ra", "dhcp"]', + desc: 'routing protocol identifier of this route.', + default: 'static', + }, + mtu: { + type: 'Optional[String]', + desc: 'the MTU along the path to destination.', + }, + }, +) diff --git a/manifests/init.pp b/manifests/init.pp index 5465c580..3ad5cd46 100644 --- a/manifests/init.pp +++ b/manifests/init.pp @@ -1,58 +1,38 @@ -# = Class: network +# Manage network configuration and routing on Linux systems with the +# network_route and network_config resources. This also installs the +# packages and gems needed to manage these settings. # -# Install the packages and gems required by the network_route and network_config resources +# @summary Manage linux network configuration and routes # -# == Parameters +# @author Vox Pupuli # -# [*ifupdown_extra*] +# @example +# include network # -# The name of the ifupdown-extra package +# @param ifupdown_extra +# The name of the ifupdown-extra package # -# Default: ifupdown-extra +# @param ifupdown_extra_provider +# The provider of the ifupdown-extra package # -# [*ifupdown_extra_provider*] +# @param manage_ifupdown_extra +# Whether this class should manage the ifupdown-extra package # -# The provider of the ifupdown-extra package +# @param ensure_ifupdown_extra +# What state the ifupdown-extra package should be in # -# Default: undef +# @param ipaddress +# The name of the ipaddress gems # -# [*manage_ifupdown_extra*] +# @param ipaddress_provider +# The provider of the ipaddress gem # -# Whether this class should manage the ifupdown-extra package +# @param manage_ipaddress +# Whether this class should manage the ipaddress gem # -# Default: true +# @param ensure_ipaddress +# What state the ifupdown-extra package should be in # -# [*ensure_ifupdown_extra*] -# -# What state the ifupdown-extra package should be in -# -# Default: present -# -# [*ipaddress*] -# -# The name of the ipaddress gems -# -# Default: ipaddress -# -# [*ipaddress_provider*] -# -# The provider of the ipaddress gem -# -# Default: gem -# -# [*manage_ipaddress*] -# -# Whether this class should manage the ipaddress gem -# -# Default: true -# -# [*ensure_ipaddress*] -# -# What state the ifupdown-extra package should be in -# -# Default: present -# - class network( $ifupdown_extra = 'ifupdown-extra', $ifupdown_extra_provider = undef, diff --git a/manifests/route.pp b/manifests/route.pp new file mode 100644 index 00000000..aa08518b --- /dev/null +++ b/manifests/route.pp @@ -0,0 +1,101 @@ +# Deploys a network route to the system using ip route +# and adds the route to a file for persistence. +# +# @summary Manages network routes in both the table and persistent file +# +# @author Vox Pupuli +# +# @example +# network::route { '192.168.232.0/24': +# ensure => present +# network => '192.168.232.0', +# netmask => '24', +# gateway => '192.168.100.1', +# } +# +# @param network +# IPv4 Network or Host ip address the route is pointing to. +# @param netmask +# IPv4 CIDR address for the network/host destination route. If using host, +# then the netmask should be '32' +# @param ensure +# Ensure the route exists (or not) +# @param default_route +# Is the route the default route? +# @param metric +# Set the routing metric. +# @param protocol +# Set the iproute2 protocol. +# Valid values are: 'static', 'dhcp', 'redirect', 'kernel', 'boot', 'ra' +# @param gateway +# The network gateway to route through. +# @param interface +# The Network Interface to route through. +# @param table +# The iproute2 routing table to apply the route to. +# @param source +# The source IP for which to apply the route to. +# @param scope +# The iproute2 scope to apply. +# Valid values: 'global', 'nowhere', 'host', 'link', 'site' +# @param mtu +# The MTU to apply to the route. +# +define network::route( + Stdlib::Ipv4 $network, + Stdlib::Ipv4 $netmask, + Enum['present', 'absent'] $ensure = 'present', + Boolean $default_route = false, + String $metric = '100', + Enum['static', 'redirect', 'kernel', 'boot', 'ra', 'dhcp'] $protocol = 'static', + Optional[Stdlib::Ipv4] $gateway, + Optional[String] $interface, + Optional[String] $table, + Optional[Stdlib::Ipv4] $source, + Optional[Enum['global', 'nowhere', 'host', 'link', 'site']] $scope, + Optional[String] $mtu, +) { + if $gateway == undef && $interface == undef { + fail('Route must have a gateway or interface!') + } + + $route_file = case $facts['os']['name'] { + 'Debian': { + '/etc/network/routes' + } + 'Ubuntu': { + if $facts['os']['release']['full'] == '18.04' { + '/etc/network/routes' + } else { + '/etc/network/routes' + } + } + /RedHat|CentOS/: { + "/etc/sysconfig/network-scripts/route-${interface}" + } + default: { + fail("Network::Route is not compatible with ${facts['os']['family']}!") + } + } + + file { $route_file: + ensure => $ensure, + owner => 'root', + group => 'root', + mode => '0755', + content => template("network/routes/${facts['os']['family'].downcase}.erb"), + } + + network_route { "${network}/${netmask}": + ensure => $ensure, + default_route => $default_route, + gateway => $gateway, + interface => $interface, + metric => $metric, + table => $table, + source => $source, + scope => $scope, + protocol => $protocol, + mtu => $mtu, + } +} diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb index c4d49f4a..e0030522 100644 --- a/spec/spec_helper.rb +++ b/spec/spec_helper.rb @@ -1,3 +1,7 @@ +RSpec.configure do |c| + c.mock_with :rspec +end + require 'puppetlabs_spec_helper/module_spec_helper' require 'rspec-puppet-facts' include RspecPuppetFacts @@ -26,7 +30,7 @@ default_facts.merge!(YAML.load(File.read(File.expand_path('../default_facts.yml', __FILE__)))) if File.exist?(File.expand_path('../default_facts.yml', __FILE__)) default_facts.merge!(YAML.load(File.read(File.expand_path('../default_module_facts.yml', __FILE__)))) if File.exist?(File.expand_path('../default_module_facts.yml', __FILE__)) c.default_facts = default_facts - c.mock_with :mocha + #c.mock_with :rspec end require 'spec_helper_methods' diff --git a/spec/unit/facts/network_nexthop_ip_spec.rb b/spec/unit/facts/network_nexthop_ip_spec.rb deleted file mode 100644 index 471664eb..00000000 --- a/spec/unit/facts/network_nexthop_ip_spec.rb +++ /dev/null @@ -1,40 +0,0 @@ -require 'spec_helper' - -describe 'network_nexthop_ip' do - before do - Facter.fact(:kernel).stubs(:value).returns('linux') - end - context 'on a Linux host' do - before do - Facter::Util::Resolution.stubs(:exec).with('ip route show 0/0').returns('default via 1.2.3.4 dev eth0') - end - it 'execs ip and determine the next hop' do - expect(Facter.fact(:network_nexthop_ip).value).to eq('1.2.3.4') - end - end - context 'on an OpenVZ VM' do - before do - Facter.clear - Facter.fact(:kernel).stubs(:value).returns('linux') - Facter.fact(:virtual).stubs(:value).returns('openvz') - Facter::Util::Resolution.stubs(:exec) - end - context 'which has the default route via a veth device' do - before do - Facter.fact(:macaddress).stubs(:value).returns(nil) - Facter::Util::Resolution.stubs(:exec).with('ip route show 0/0').returns('default via 1.2.3.4 dev eth0') - end - it 'execs ip and determine the next hop' do - expect(Facter.fact(:network_nexthop_ip).value).to eq('1.2.3.4') - end - end - context 'with only venet interfaces' do - before do - Facter::Util::Resolution.stubs(:exec).with('ip route show 0/0').returns('default dev venet0 scope link') - end - it 'does not exist' do - expect(Facter.fact(:network_nexthop_ip).value).to be_nil - end - end - end -end diff --git a/spec/unit/facts/network_primary_interface_spec.rb b/spec/unit/facts/network_primary_interface_spec.rb deleted file mode 100644 index 90d5f01f..00000000 --- a/spec/unit/facts/network_primary_interface_spec.rb +++ /dev/null @@ -1,51 +0,0 @@ -require 'spec_helper' - -describe 'network_primary_interface' do - before do - Facter.clear - Facter.fact(:kernel).stubs(:value).returns('linux') - end - context 'on a Linux host with Facter 2.x' do - before do - Facter::Util::Resolution.stubs(:exec).with('ip route show 0/0').returns('default via 1.2.3.4 dev eth0') - Facter::Util::Resolution.stubs(:exec).with('ip route get 1.2.3.4').returns('1.2.3.4 dev eth0 src 1.2.3.99\n - cache mtu 1500 advmss 1460 hoplimit 64') - end - it 'execs ip and determines the primary interface' do - expect(Facter.fact(:network_primary_interface).value).to eq('eth0') - end - end - context 'on an OpenVZ VM with Facter 2.x' do - before do - Facter.stubs(:version).returns('2.4.6') - Facter.fact(:kernel).stubs(:value).returns('linux') - Facter.fact(:virtual).stubs(:value).returns('openvz') - Facter::Util::Resolution.stubs(:exec) - end - context 'with only venet devices' do - before do - Facter::Util::Resolution.stubs(:exec).with('ip route show 0/0').returns('default dev venet0 scope link') - Facter::Util::Resolution.stubs(:exec).with('ip route get 8.8.8.8').returns('8.8.8.8 dev venet0 src 1.2.3.99\n - cache mtu 1500 advmss 1460 hoplimit 64') - end - it 'execs ip and determines the primary interface' do - expect(Facter.fact(:network_primary_interface).value).to eq('venet0') - end - end - end - context 'on a Linux host with Facter 3' do - before do - Facter::Util::Resolution.expects(:exec).never - interface_fact = Facter.fact(:network_primary_interface) - networking_fact = Facter::Util::Fact.new(:networking) - networking_fact.expects(:value).returns('primary' => 'eth1') - Facter.expects(:fact).with(:networking).returns(networking_fact) - Facter.expects(:fact).with(:network_primary_interface).returns(interface_fact) - end - it 'uses the built-in facts to determine the primary interface' do - # (rski) For some reason with ruby 1.9.3 this doesn't work in the before block - Facter.stubs(:version).returns('3.0.0') - expect(Facter.fact(:network_primary_interface).value).to eq('eth1') - end - end -end diff --git a/spec/unit/facts/network_primary_ip_spec.rb b/spec/unit/facts/network_primary_ip_spec.rb deleted file mode 100644 index 86fab556..00000000 --- a/spec/unit/facts/network_primary_ip_spec.rb +++ /dev/null @@ -1,52 +0,0 @@ -require 'spec_helper' - -describe 'network_primary_ip' do - before do - Facter.clear - Facter.fact(:kernel).stubs(:value).returns('linux') - end - context 'on a Linux host with Facter 2.x' do - before do - Facter.stubs(:version).returns('2.4.6') - Facter::Util::Resolution.stubs(:exec).with('ip route show 0/0').returns('default via 1.2.3.4 dev eth0').once - Facter::Util::Resolution.stubs(:exec).with('ip route get 1.2.3.4').returns("1.2.3.4 dev eth0 src 1.2.3.99\n - cache mtu 1500 advmss 1460 hoplimit 64").once - end - it 'execs ip and determine the primary ip address' do - expect(Facter.fact(:network_primary_ip).value).to eq('1.2.3.99') - end - end - context 'on an OpenVZ VM with Facter 2.x' do - before do - Facter.clear - Facter.stubs(:version).returns('2.4.6') - Facter.fact(:virtual).stubs(:value).returns('openvz') - Facter::Util::Resolution.stubs(:exec) - end - context 'with only venet devices' do - before do - Facter::Util::Resolution.stubs(:exec).with('ip route show 0/0').returns('default dev venet0 scope link') - Facter::Util::Resolution.stubs(:exec).with('ip route get 8.8.8.8').returns("8.8.8.8 dev venet0 src 1.2.3.99\n - cache mtu 1500 advmss 1460 hoplimit 64") - end - it 'execs ip and determine the primary ip address' do - expect(Facter.fact(:network_primary_ip).value).to eq('1.2.3.99') - end - end - end - context 'on a Linux host with Facter 3.x' do - before do - Facter::Util::Resolution.expects(:exec).never - ip_fact = Facter.fact(:network_primary_ip) - networking_fact = Facter::Util::Fact.new(:networking) - networking_fact.expects(:value).returns('ip' => '2.3.4.5') - Facter.expects(:fact).with(:networking).returns(networking_fact) - Facter.expects(:fact).with(:network_primary_ip).returns(ip_fact) - end - it 'uses the built-in facts to resolve the primary ip address' do - # (rski) For some reason this doesn't work with ruby 1.9.3 - Facter.stubs(:version).returns('3.0.0') - expect(Facter.fact(:network_primary_ip).value).to eq('2.3.4.5') - end - end -end diff --git a/spec/unit/provider/network_route/redhat_spec.rb b/spec/unit/provider/network_route/redhat_spec.rb deleted file mode 100644 index eabbe687..00000000 --- a/spec/unit/provider/network_route/redhat_spec.rb +++ /dev/null @@ -1,171 +0,0 @@ -require 'spec_helper' - -describe Puppet::Type.type(:network_route).provider(:redhat) do - def fixture_data(file) - basedir = File.join(PROJECT_ROOT, 'spec', 'fixtures', 'provider', 'network_route', 'redhat') - File.read(File.join(basedir, file)) - end - - describe 'when parsing' do - describe 'a simple well formed file' do - let(:data) { described_class.parse_file('', fixture_data('simple_routes')) } - - it 'parses out normal ipv4 network routes' do - expect(data.find { |h| h[:name] == '172.17.67.0/30' }).to eq( - name: '172.17.67.0/30', - network: '172.17.67.0', - netmask: '255.255.255.252', - gateway: '172.18.6.2', - interface: 'vlan200' - ) - end - it 'parses out ipv6 network routes' do - expect(data.find { |h| h[:name] == '2a01:4f8:211:9d5:53::/96' }).to eq( - name: '2a01:4f8:211:9d5:53::/96', - network: '2a01:4f8:211:9d5:53::', - netmask: 'ffff:ffff:ffff:ffff:ffff:ffff::', - gateway: '2a01:4f8:211:9d5::2', - interface: 'vlan200' - ) - end - - it 'parses out default routes' do - expect(data.find { |h| h[:name] == 'default' }).to eq( - name: 'default', - network: 'default', - netmask: '0.0.0.0', - gateway: '10.0.0.1', - interface: 'eth1' - ) - end - end - - describe 'an advanced, well formed file' do - let :data do - described_class.parse_file('', fixture_data('advanced_routes')) - end - - it 'parses out normal ipv4 network routes' do - expect(data.find { |h| h[:name] == '2a01:4f8:211:9d5:53::/96' }).to eq( - name: '2a01:4f8:211:9d5:53::/96', - network: '2a01:4f8:211:9d5:53::', - netmask: 'ffff:ffff:ffff:ffff:ffff:ffff::', - gateway: '2a01:4f8:211:9d5::2', - interface: 'vlan200', - options: 'table 200' - ) - end - - it 'parses out normal ipv6 network routes' do - expect(data.find { |h| h[:name] == '172.17.67.0/30' }).to eq( - name: '172.17.67.0/30', - network: '172.17.67.0', - netmask: '255.255.255.252', - gateway: '172.18.6.2', - interface: 'vlan200', - options: 'table 200' - ) - end - end - - describe 'an invalid file' do - it 'fails' do - expect do - described_class.parse_file('', "192.168.1.1/30 via\n") - end.to raise_error - end - end - end - - describe 'when formatting' do - let :route1_provider do - stub( - 'route1_provider', - name: '172.17.67.0/30', - network: '172.17.67.0', - netmask: '30', - gateway: '172.18.6.2', - interface: 'vlan200', - options: 'table 200' - ) - end - - let :route2_provider do - stub( - 'lo_provider', - name: '172.28.45.0/30', - network: '172.28.45.0', - netmask: '30', - gateway: '172.18.6.2', - interface: 'eth0', - options: 'table 200' - ) - end - - let :defaultroute_provider do - stub( - 'defaultroute_provider', - name: 'default', - network: 'default', - netmask: '', - gateway: '10.0.0.1', - interface: 'eth1', - options: 'table 200' - ) - end - - let :nooptions_provider do - stub( - 'nooptions_provider', - name: 'default', - network: 'default', - netmask: '', - gateway: '10.0.0.1', - interface: 'eth2', - options: :absent - ) - end - - let :content do - described_class.format_file('', [route1_provider, route2_provider, defaultroute_provider, nooptions_provider]) - end - - describe 'writing the route line' do - describe 'For standard (non-default) routes' do - it 'writes a single line for the route' do - expect(content.scan(%r{^172.17.67.0\/30 .*$}).length).to eq(1) - end - - it 'writes 7 fields' do - expect(content.scan(%r{^172.17.67.0\/30 .*$}).first.split(' ').length).to eq(7) - end - - it 'has the correct fields appended' do - expect(content.scan(%r{^172.17.67.0\/30 .*$}).first).to include('172.17.67.0/30 via 172.18.6.2 dev vlan200 table 200') - end - - it 'fails if the netmask property is not defined' do - route2_provider.unstub(:netmask) - route2_provider.stubs(:netmask).returns nil - expect { content }.to raise_exception - end - - it 'fails if the gateway property is not defined' do - route2_provider.unstub(:gateway) - route2_provider.stubs(:gateway).returns nil - expect { content }.to raise_exception - end - end - end - - describe 'for default routes' do - it 'has the correct fields appended' do - expect(content.scan(%r{^default .*$}).first).to include('default via 10.0.0.1 dev eth1') - end - - it 'does not contain the word absent when no options are defined' do - expect(content).not_to match(%r{absent}) - end - end - end -end diff --git a/spec/unit/provider/network_route/routes_spec.rb b/spec/unit/provider/network_route/routes_spec.rb deleted file mode 100644 index 4ac23ebf..00000000 --- a/spec/unit/provider/network_route/routes_spec.rb +++ /dev/null @@ -1,194 +0,0 @@ -require 'spec_helper' - -describe Puppet::Type.type(:network_route).provider(:routes) do - def fixture_data(file) - basedir = File.join(PROJECT_ROOT, 'spec', 'fixtures', 'provider', 'network_route', 'routes_spec') - File.read(File.join(basedir, file)) - end - - describe 'when parsing' do - it 'parses out simple ipv4 iface lines' do - fixture = fixture_data('simple_routes') - data = described_class.parse_file('', fixture) - - expect(data.find { |h| h[:name] == '172.17.67.0/24' }).to eq( - name: '172.17.67.0/24', - network: '172.17.67.0', - netmask: '255.255.255.0', - gateway: '172.18.6.2', - interface: 'vlan200' - ) - end - - it 'names default routes "default" and have a 0.0.0.0 netmask' do - fixture = fixture_data('simple_routes') - data = described_class.parse_file('', fixture) - - expect(data.find { |h| h[:name] == 'default' }).to eq( - name: 'default', - network: 'default', - netmask: '0.0.0.0', - gateway: '172.18.6.2', - interface: 'vlan200' - ) - end - - it 'parses out simple ipv6 iface lines' do - fixture = fixture_data('simple_routes') - data = described_class.parse_file('', fixture) - - expect(data.find { |h| h[:name] == '2a01:4f8:211:9d5:53::/96' }).to eq( - name: '2a01:4f8:211:9d5:53::/96', - network: '2a01:4f8:211:9d5:53::', - netmask: 'ffff:ffff:ffff:ffff:ffff:ffff::', - gateway: '2a01:4f8:211:9d5::2', - interface: 'vlan200' - ) - end - - it 'parses out advanced routes' do - fixture = fixture_data('advanced_routes') - data = described_class.parse_file('', fixture) - - expect(data.find { |h| h[:name] == '172.17.67.0/24' }).to eq( - name: '172.17.67.0/24', - network: '172.17.67.0', - netmask: '255.255.255.0', - gateway: '172.18.6.2', - interface: 'vlan200', - options: 'table 200' - ) - end - it 'parses out advanced ipv6 iface lines' do - fixture = fixture_data('advanced_routes') - data = described_class.parse_file('', fixture) - - expect(data.find { |h| h[:name] == '2a01:4f8:211:9d5:53::/96' }).to eq( - name: '2a01:4f8:211:9d5:53::/96', - network: '2a01:4f8:211:9d5:53::', - netmask: 'ffff:ffff:ffff:ffff:ffff:ffff::', - gateway: '2a01:4f8:211:9d5::2', - interface: 'vlan200', - options: 'table 200' - ) - end - - describe 'when reading an invalid routes file' do - it 'with missing options should fail' do - expect do - described_class.parse_file('', "192.168.1.1 255.255.255.0 172.16.0.1\n") - end.to raise_error - end - end - end - - describe 'when formatting' do - let :route1_provider do - stub( - 'route1_provider', - name: '172.17.67.0', - network: '172.17.67.0', - netmask: '255.255.255.0', - gateway: '172.18.6.2', - interface: 'vlan200', - options: 'table 200' - ) - end - - let :route2_provider do - stub( - 'lo_provider', - name: '172.28.45.0', - network: '172.28.45.0', - netmask: '255.255.255.0', - gateway: '172.18.6.2', - interface: 'eth0', - options: 'table 200' - ) - end - - let :content do - described_class.format_file('', [route1_provider, route2_provider]) - end - - describe 'writing the route line' do - it 'writes a single line for the route' do - expect(content.scan(%r{^172.17.67.0 .*$}).length).to eq(1) - end - it 'writes all 6 fields' do - expect(content.scan(%r{^172.17.67.0 .*$}).first.split(' ').length).to eq(6) - end - - it 'has the correct fields appended' do - expect(content.scan(%r{^172.17.67.0 .*$}).first).to include('172.17.67.0 255.255.255.0 172.18.6.2 vlan200 table 200') - end - - it 'fails if the netmask property is not defined' do - route2_provider.unstub(:netmask) - route2_provider.stubs(:netmask).returns nil - expect { content }.to raise_exception - end - - it 'fails if the gateway property is not defined' do - route2_provider.unstub(:gateway) - route2_provider.stubs(:gateway).returns nil - expect { content }.to raise_exception - end - end - end - describe 'when formatting simple files' do - let :route1_provider do - stub( - 'route1_provider', - name: '172.17.67.0', - network: '172.17.67.0', - netmask: '255.255.255.0', - gateway: '172.18.6.2', - interface: 'vlan200', - options: :absent - ) - end - - let :route2_provider do - stub( - 'lo_provider', - name: '172.28.45.0', - network: '172.28.45.0', - netmask: '255.255.255.0', - gateway: '172.18.6.2', - interface: 'eth0', - options: :absent - ) - end - - let :content do - described_class.format_file('', [route1_provider, route2_provider]) - end - - describe 'writing the route line' do - it 'writes a single line for the route' do - expect(content.scan(%r{^172.17.67.0 .*$}).length).to eq(1) - end - - it 'writes only fields' do - expect(content.scan(%r{^172.17.67.0 .*$}).first.split(' ').length).to eq(4) - end - - it 'has the correct fields appended' do - expect(content.scan(%r{^172.17.67.0 .*$}).first).to include('172.17.67.0 255.255.255.0 172.18.6.2 vlan200') - end - - it 'fails if the netmask property is not defined' do - route2_provider.unstub(:netmask) - route2_provider.stubs(:netmask).returns nil - expect { content }.to raise_exception - end - - it 'fails if the gateway property is not defined' do - route2_provider.unstub(:gateway) - route2_provider.stubs(:gateway).returns nil - expect { content }.to raise_exception - end - end - end -end diff --git a/spec/unit/puppet/provider/network_route/network_route_spec.rb b/spec/unit/puppet/provider/network_route/network_route_spec.rb new file mode 100644 index 00000000..e0967090 --- /dev/null +++ b/spec/unit/puppet/provider/network_route/network_route_spec.rb @@ -0,0 +1,111 @@ +require 'spec_helper' + +# nsure_module_defined('Puppet::Provider::NetworkRoute') +module Puppet::Provider::NetworkRoute; end +require 'puppet/provider/network_route/network_route' + +RSpec.describe Puppet::Provider::NetworkRoute::NetworkRoute do + subject(:provider) { described_class.new } + + let(:context) { instance_double('Puppet::ResourceApi::BaseContext', 'context') } + let(:routes) { instance_double('Net::IP::Route::Collection', 'routes') } + let(:netiproute) { instance_double('Net::IP::Route', prefix: 'route') } + + before(:each) do + allow(Net::IP::Route).to receive(:new).with('should').and_return(netiproute) + allow(Net::IP::Route::Collection).to receive(:new).with('main').and_return(routes) + end + + let(:route) do + [ + { + prefix: 'default', + via: '10.0.2.2', + dev: 'enp0s3', + proto: 'dhcp', + metric: '100' + } + ] + end + let(:network_route) do + [ + { + default_route: true, + ensure: 'present', + gateway: '10.0.2.2', + interface: 'enp0s3', + metric: '100', + prefix: 'default', + protocol: 'dhcp' + } + ] + end + + describe '#puppet_munge(should)' do + let(:should) { network_route[0] } + + it 'should parse network_route into iproute2 keys' do + expect(provider.puppet_munge(should)).to eq( + { + dev: 'enp0s3', + metric: '100', + prefix: 'default', + proto: 'dhcp', + via: '10.0.2.2', + } + ) + end + end + + describe '#get' do + before(:each) do + allow(provider).to receive(:routes_list).and_return(route) # rubocop:disable RSpec/SubjectStub + end + + it 'processes resources' do + expect(provider.get(context)).to eq( + [{default_route: true, + ensure: 'present', + gateway: '10.0.2.2', + interface: 'enp0s3', + metric: '100', + prefix: 'default', + protocol: 'dhcp'}] + ) + end + end + + describe '#create(context, name, should)' do + before(:each) do + allow(provider).to receive(:puppet_munge).with('should').and_return('munged') + end + + it 'creates the resource' do + expect(routes).to receive(:add).with(netiproute) + provider.create(context, 'default', 'should') + end + end + + describe '#update(context, name, should)' do + before(:each) do + allow(provider).to receive(:puppet_munge).with('should').and_return('munged') + end + + it 'updates the resource' do + expect(routes).to receive(:flush).with(netiproute.prefix) + expect(routes).to receive(:add).with(netiproute) + provider.update(context, 'default', 'should') + end + end + + describe 'delete(context, name, should)' do + before(:each) do + allow(provider).to receive(:puppet_munge).with('should').and_return('munged') + end + + it 'deletes the resource' do + expect(routes).to receive(:flush).with(netiproute.prefix) + provider.delete(context, 'default', 'should') + end + end +end diff --git a/spec/unit/puppet/type/network_route_spec.rb b/spec/unit/puppet/type/network_route_spec.rb new file mode 100644 index 00000000..c38ee02b --- /dev/null +++ b/spec/unit/puppet/type/network_route_spec.rb @@ -0,0 +1,186 @@ +require 'spec_helper' +require 'puppet/type/network_route' + +RSpec.describe 'the network_route type' do + it 'loads' do + expect(Puppet::Type.type(:network_route)).not_to be_nil + end + + context 'prefix is default' do + let(:resource) do + Puppet::Type.type('network_route').new( + prefix: 'default', + default_route: true, + ) + end + + it 'prefix is set' do + expect(resource[:prefix]).to eq 'default' + end + + it 'name is set to prefix' do + expect(resource[:name]).to eq 'default' + end + + it 'default_route should be true' do + expect(resource[:default_route]).to eq :true + end + + it 'metric should be 100' do + expect(resource[:metric]).to eq '100' + end + + it 'protocol should be static' do + expect(resource[:protocol]).to eq 'static' + end + end + + context 'with a non-default route' do + let(:resource) do + Puppet::Type.type('network_route').new( + prefix: '10.155.255.0/24', + default_route: false, + metric: '600', + source: '10.155.255.110' + ) + end + + it 'prefix is set to network address' do + expect(resource[:prefix]).to eq '10.155.255.0/24' + end + + it 'default_route is false' do + expect(resource[:default_route]).to eq :false + end + + it 'metric is set to 600' do + expect(resource[:metric]).to eq '600' + end + + it 'source is set to 10.155.255.110' do + expect(resource[:source]).to eq '10.155.255.110' + end + end + + context 'with gateway set' do + let(:resource) do + Puppet::Type.type('network_route').new( + prefix: 'default', + gateway: '10.155.255.1' + ) + end + + it 'gateway should be 10.155.255.1' do + expect(resource[:gateway]).to eq '10.155.255.1' + end + end + + context 'with interface set' do + let(:resource) do + Puppet::Type.type('network_route').new( + prefix: 'default', + interface: 'eth0' + ) + end + + it 'interface should be eth0' do + expect(resource[:interface]).to eq 'eth0' + end + end + + context 'with metric set' do + let(:resource) do + Puppet::Type.type('network_route').new( + prefix: 'default', + metric: '400' + ) + end + + it 'metric should be 400' do + expect(resource[:metric]).to eq '400' + end + end + + context 'with table set' do + let(:resource) do + Puppet::Type.type('network_route').new( + prefix: 'default', + table: 'main' + ) + end + + it 'table should be main' do + expect(resource[:table]).to eq 'main' + end + end + + context 'with source set' do + let(:resource) do + Puppet::Type.type('network_route').new( + prefix: 'default', + source: '10.155.255.10' + ) + end + + it 'source should be 10.155.255.10' do + expect(resource[:source]).to eq '10.155.255.10' + end + end + + context 'with scope set' do + let(:resource) do + Puppet::Type.type('network_route').new( + prefix: 'default', + scope: 'link' + ) + end + + it 'scope should be link' do + expect(resource[:scope]).to eq 'link' + end + end + + context 'with protocol set' do + let(:resource) do + Puppet::Type.type('network_route').new( + prefix: 'default', + protocol: 'boot' + ) + end + + it 'protocol should be boot' do + expect(resource[:protocol]).to eq 'boot' + end + end + + context 'with mtu set' do + let(:resource) do + Puppet::Type.type('network_route').new( + prefix: 'default', + mtu: '1500' + ) + end + + it 'mtu should be 1500' do + expect(resource[:mtu]).to eq '1500' + end + end + + context 'with invalid scope' do + it 'should raise an error' do + expect { Puppet::Type.type('network_route').new( + prefix: 'default', + scope: 'fail' + )}.to raise_error(Puppet::ResourceError) + end + end + + context 'with invalid protocol' do + it 'should raise an error' do + expect { Puppet::Type.type('network_route').new( + prefix: 'default', + protocol: 'fail' + )}.to raise_error(Puppet::ResourceError) + end + end +end diff --git a/spec/unit/type/network_route_spec.rb b/spec/unit/type/network_route_spec.rb deleted file mode 100644 index 85566c13..00000000 --- a/spec/unit/type/network_route_spec.rb +++ /dev/null @@ -1,74 +0,0 @@ -require 'spec_helper' - -describe Puppet::Type.type(:network_route) do - before do - provider_class = stub 'provider class', name: 'fake', suitable?: true, supports_parameter?: true - provider_class.stubs(:new) - - Puppet::Type.type(:network_route).stubs(:defaultprovider).returns provider_class - Puppet::Type.type(:network_route).stubs(:provider).returns provider_class - end - - describe 'when validating the attribute' do - describe :name do # rubocop:disable RSpec/DescribeSymbol - it { expect(Puppet::Type.type(:network_route).attrtype(:name)).to eq(:param) } - end - - [:ensure, :network, :netmask, :gateway, :interface, :options].each do |property| - describe property do - it { expect(Puppet::Type.type(:network_route).attrtype(property)).to eq(:property) } - end - end - - it 'use the name parameter as the namevar' do - expect(Puppet::Type.type(:network_route).key_attributes).to eq([:name]) - end - - describe 'ensure' do - it 'is an ensurable value' do - expect(Puppet::Type.type(:network_route).propertybyname(:ensure).ancestors).to be_include(Puppet::Property::Ensure) - end - end - end - - describe 'when validating the attribute value' do - describe 'network' do - it 'validates the network as an IP address' do - expect do - Puppet::Type.type(:network_route).new(name: '192.168.1.0/24', network: 'not an ip address', netmask: '255.255.255.0', gateway: '23.23.23.42', interface: 'eth0') - end.to raise_error - end - end - - describe 'netmask' do - it 'fails if an invalid netmask is used' do - expect do - Puppet::Type.type(:network_route).new(name: '192.168.1.0/24', network: '192.168.1.0', netmask: 'This is clearly not a netmask', gateway: '23.23.23.42', interface: 'eth0') - end.to raise_error - end - - it 'converts netmasks of the CIDR form' do - r = Puppet::Type.type(:network_route).new(name: '192.168.1.0/24', network: '192.168.1.0', netmask: '24', gateway: '23.23.23.42', interface: 'eth0') - expect(r[:netmask]).to eq('255.255.255.0') - end - - it 'converts IPv6 netmasks of the CIDR form' do - r = Puppet::Type.type(:network_route).new(name: 'lxd bridge', network: 'fd58:281b:6eef:eb3d::', netmask: '64', gateway: 'fd58:281b:6eef:eb3d::1', interface: 'lxdbr0') - expect(r[:netmask]).to eq('ffff:ffff:ffff:ffff::') - end - - it 'converts netmasks of the expanded netmask form' do - r = Puppet::Type.type(:network_route).new(name: '192.168.1.0/24', network: '192.168.1.0', netmask: '255.255.128.0', gateway: '23.23.23.42', interface: 'eth0') - expect(r[:netmask]).to eq('255.255.128.0') - end - end - - describe 'gateway' do - it 'validates as an IP address' do - expect do - Puppet::Type.type(:network_route).new(name: '192.168.1.0/24', network: '192.168.1.0', netmask: '255.255.255.0', gateway: 'not an ip address', interface: 'eth0') - end.to raise_error - end - end - end -end diff --git a/templates/routes/redhat.erb b/templates/routes/redhat.erb new file mode 100644 index 00000000..0a1909e6 --- /dev/null +++ b/templates/routes/redhat.erb @@ -0,0 +1,23 @@ +<% require 'net/ip' + route = { + prefix: @prefix, + proto: @protocol, + metric: @metric, + dev: @interface, + via: @gateway, + src: @source, + scope: @scope, + mtu: @mtu, + table: @table, + } + + route.each do |k, v| + if v.nil? + route.delete(k) + end + end + + new_route = Net::IP::Route.new(new_route) +%> + +<%= new_route.prefix %> <%= new_route.to_params %> \ No newline at end of file