guests/freebsd: Configure predictable networks in a single command

This commit refactors the freebsd networking to:

1. Use predictable network naming
2. Properly handle DHCP vs static networks on up and reload [GH-5852]
3. Perform all networking configuration in a single command to prevent
   partial configuration.
This commit is contained in:
Seth Vargo 2016-06-05 13:15:19 -04:00
parent a444110993
commit 2b08151977
No known key found for this signature in database
GPG Key ID: 905A90C2949E8787
4 changed files with 92 additions and 23 deletions

View File

@ -9,40 +9,57 @@ module VagrantPlugins
include Vagrant::Util include Vagrant::Util
def self.configure_networks(machine, networks) def self.configure_networks(machine, networks)
# Remove any previous network additions to the configuration file. options = { shell: "sh" }
machine.communicate.sudo("sed -i '' -e '/^#VAGRANT-BEGIN/,/^#VAGRANT-END/ d' /etc/rc.conf", {shell: "sh"}) comm = machine.communicate
networks.each do |network| commands = []
# Determine the interface prefix... interfaces = []
command = "ifconfig -a | grep -o ^[0-9a-z]*"
result = "" # Remove any previous network additions to the configuration file.
ifname = "" commands << "sed -i'' -e '/^#VAGRANT-BEGIN/,/^#VAGRANT-END/ d' /etc/rc.conf"
machine.communicate.execute(command) do |type, data|
result << data if type == :stdout comm.sudo("ifconfig -a | grep -o ^[0-9a-z]* | grep -v '^lo'", options) do |_, stdout|
if result.split(/\n/).any?{|i| i.match(/vtnet*/)} interfaces = stdout.split("\n")
ifname = "vtnet#{network[:interface]}"
else
ifname = "em#{network[:interface]}"
end
end end
networks.each.with_index do |network, i|
network[:device] = interfaces[network[:interface]]
entry = TemplateRenderer.render("guests/freebsd/network_#{network[:type]}", entry = TemplateRenderer.render("guests/freebsd/network_#{network[:type]}",
options: network, ifname: ifname) options: network,
)
remote_path = "/tmp/vagrant-network-#{network[:device]}-#{Time.now.to_i}-#{i}"
Tempfile.open("vagrant-freebsd-configure-networks") do |f| Tempfile.open("vagrant-freebsd-configure-networks") do |f|
f.binmode f.binmode
f.write(entry) f.write(entry)
f.fsync f.fsync
f.close f.close
machine.communicate.upload(f.path, "/tmp/vagrant-network-entry") comm.upload(f.path, remote_path)
end end
machine.communicate.sudo("su -m root -c 'cat /tmp/vagrant-network-entry >> /etc/rc.conf'", {shell: "sh"}) commands << <<-EOH.gsub(/^ {14}/, '')
machine.communicate.sudo("rm -f /tmp/vagrant-network-entry", {shell: "sh"}) cat '#{remote_path}' >> /etc/rc.conf
rm -f '#{remote_path}'
EOH
# Restart interface so it loads configuration stored in /etc/rc.conf # If the network is DHCP, then we have to start the dhclient, unless
machine.communicate.sudo("service netif restart #{ifname}", {shell: "sh"}) # it is already running. See GH-5852 for more information
end if network[:type].to_sym == :dhcp
file = "/var/run/dhclient.#{network[:device]}.pid"
commands << <<-EOH.gsub(/^ {16}/, '')
if ! test -f '#{file}' || ! kill -0 $(cat '#{file}'); then
dhclient '#{network[:device]}'
fi
EOH
end
# For some reason, this returns status 1... every time
commands << "/etc/rc.d/netif restart '#{network[:device]}' || true"
end
comm.sudo(commands.join("\n"), options)
end end
end end
end end

View File

@ -1,3 +1,4 @@
#VAGRANT-BEGIN #VAGRANT-BEGIN
ifconfig_<%= ifname %>="DHCP" ifconfig_<%= options[:device] %>="DHCP"
synchronous_dhclient="YES"
#VAGRANT-END #VAGRANT-END

View File

@ -1,5 +1,5 @@
#VAGRANT-BEGIN #VAGRANT-BEGIN
ifconfig_<%= ifname %>="inet <%= options[:ip] %> netmask <%= options[:netmask] %>" ifconfig_<%= options[:device] %>="inet <%= options[:ip] %> netmask <%= options[:netmask] %>"
<% if options[:gateway] %> <% if options[:gateway] %>
default_router="<%= options[:gateway] %>" default_router="<%= options[:gateway] %>"
<% end %> <% end %>

View File

@ -0,0 +1,51 @@
require_relative "../../../../base"
describe "VagrantPlugins::GuestFreeBSD::Cap::ConfigureNetworks" do
let(:described_class) do
VagrantPlugins::GuestFreeBSD::Plugin
.components
.guest_capabilities[:freebsd]
.get(:configure_networks)
end
let(:machine) { double("machine") }
let(:comm) { VagrantTests::DummyCommunicator::Communicator.new(machine) }
before do
allow(machine).to receive(:communicate).and_return(comm)
comm.stub_command("ifconfig -a | grep -o ^[0-9a-z]* | grep -v '^lo'",
stdout: "em1\nem2")
end
after do
comm.verify_expectations!
end
describe ".configure_networks" do
let(:network_1) do
{
interface: 0,
type: "dhcp",
}
end
let(:network_2) do
{
interface: 1,
type: "static",
ip: "33.33.33.10",
netmask: "255.255.0.0",
gateway: "33.33.0.1",
}
end
it "creates and starts the networks" do
described_class.configure_networks(machine, [network_1, network_2])
expect(comm.received_commands[1]).to match(/dhclient 'em1'/)
expect(comm.received_commands[1]).to match(/\/etc\/rc.d\/netif restart 'em1'/)
expect(comm.received_commands[1]).to_not match(/dhclient 'em2'/)
expect(comm.received_commands[1]).to match(/\/etc\/rc.d\/netif restart 'em2'/)
end
end
end