2012-01-13 07:27:35 +00:00
|
|
|
require 'log4r'
|
|
|
|
|
2012-05-23 23:03:14 +00:00
|
|
|
require "vagrant"
|
2011-08-28 06:39:02 +00:00
|
|
|
require 'vagrant/util/platform'
|
2013-12-14 06:14:13 +00:00
|
|
|
require "vagrant/util/subprocess"
|
2011-08-28 06:39:02 +00:00
|
|
|
|
2012-04-19 05:20:45 +00:00
|
|
|
module VagrantPlugins
|
|
|
|
module HostBSD
|
2010-07-11 05:07:10 +00:00
|
|
|
# Represents a BSD host, such as FreeBSD and Darwin (Mac OS X).
|
2012-11-07 05:20:22 +00:00
|
|
|
class Host < Vagrant.plugin("2", :host)
|
2012-04-19 05:20:45 +00:00
|
|
|
include Vagrant::Util
|
|
|
|
include Vagrant::Util::Retryable
|
2010-07-13 05:10:17 +00:00
|
|
|
|
2011-12-12 07:22:44 +00:00
|
|
|
def self.match?
|
2012-04-19 05:20:45 +00:00
|
|
|
Vagrant::Util::Platform.darwin? || Vagrant::Util::Platform.bsd?
|
2011-12-12 07:22:44 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.precedence
|
|
|
|
# Set a lower precedence because this is a generic OS. We
|
|
|
|
# want specific distros to match first.
|
|
|
|
2
|
|
|
|
end
|
|
|
|
|
|
|
|
def initialize(*args)
|
|
|
|
super
|
|
|
|
|
2012-01-13 07:27:35 +00:00
|
|
|
@logger = Log4r::Logger.new("vagrant::hosts::bsd")
|
2011-12-12 07:22:44 +00:00
|
|
|
@nfs_restart_command = "sudo nfsd restart"
|
2012-01-30 19:40:29 +00:00
|
|
|
@nfs_exports_template = "nfs/exports"
|
2011-08-28 06:39:02 +00:00
|
|
|
end
|
|
|
|
|
2010-07-12 04:33:49 +00:00
|
|
|
def nfs?
|
2010-09-09 07:37:54 +00:00
|
|
|
retryable(:tries => 10, :on => TypeError) do
|
2010-09-30 06:38:07 +00:00
|
|
|
system("which nfsd > /dev/null 2>&1")
|
2010-07-17 05:00:49 +00:00
|
|
|
end
|
2010-07-12 04:33:49 +00:00
|
|
|
end
|
2010-07-13 05:10:17 +00:00
|
|
|
|
2013-07-18 04:01:49 +00:00
|
|
|
def nfs_export(id, ips, folders)
|
2013-12-14 06:14:13 +00:00
|
|
|
nfs_checkexports!
|
|
|
|
|
2013-07-10 21:19:57 +00:00
|
|
|
# We need to build up mapping of directories that are enclosed
|
|
|
|
# within each other because the exports file has to have subdirectories
|
|
|
|
# of an exported directory on the same line. e.g.:
|
|
|
|
#
|
|
|
|
# "/foo" "/foo/bar" ...
|
|
|
|
# "/bar"
|
|
|
|
#
|
|
|
|
# We build up this mapping within the following hash.
|
|
|
|
@logger.debug("Compiling map of sub-directories for NFS exports...")
|
|
|
|
dirmap = {}
|
|
|
|
folders.each do |_, opts|
|
2013-07-16 23:23:33 +00:00
|
|
|
hostpath = opts[:hostpath].dup
|
|
|
|
hostpath.gsub!('"', '\"')
|
2013-07-10 21:19:57 +00:00
|
|
|
|
|
|
|
found = false
|
|
|
|
dirmap.each do |dirs, diropts|
|
|
|
|
dirs.each do |dir|
|
|
|
|
if dir.start_with?(hostpath) || hostpath.start_with?(dir)
|
|
|
|
# TODO: verify opts and diropts are _identical_, raise an error
|
|
|
|
# if not. NFS mandates subdirectories have identical options.
|
|
|
|
dirs << hostpath
|
|
|
|
found = true
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
break if found
|
|
|
|
end
|
|
|
|
|
|
|
|
if !found
|
|
|
|
dirmap[[hostpath]] = opts.dup
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Sort all the keys by length so that the directory closest to
|
|
|
|
# the root is exported first.
|
|
|
|
dirmap.each do |dirs, _|
|
|
|
|
dirs.sort_by! { |d| d.length }
|
|
|
|
end
|
|
|
|
|
2013-09-01 20:08:02 +00:00
|
|
|
# Setup the NFS options
|
|
|
|
dirmap.each do |dirs, opts|
|
|
|
|
if !opts[:bsd__nfs_options]
|
|
|
|
opts[:bsd__nfs_options] = ["alldirs"]
|
|
|
|
end
|
|
|
|
|
|
|
|
hasmapall = false
|
|
|
|
opts[:bsd__nfs_options].each do |opt|
|
2013-11-23 19:36:20 +00:00
|
|
|
# mapall/maproot are mutually exclusive, so we have to check
|
|
|
|
# for both here.
|
2013-11-01 14:44:49 +00:00
|
|
|
if opt =~ /^mapall=/ || opt =~ /^maproot=/
|
2013-09-01 20:08:02 +00:00
|
|
|
hasmapall = true
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if !hasmapall
|
|
|
|
opts[:bsd__nfs_options] << "mapall=#{opts[:map_uid]}:#{opts[:map_gid]}"
|
|
|
|
end
|
|
|
|
|
|
|
|
opts[:bsd__compiled_nfs_options] = opts[:bsd__nfs_options].map do |opt|
|
|
|
|
"-#{opt}"
|
|
|
|
end.join(" ")
|
|
|
|
end
|
|
|
|
|
2013-07-10 21:19:57 +00:00
|
|
|
@logger.info("Exporting the following for NFS...")
|
|
|
|
dirmap.each do |dirs, opts|
|
|
|
|
@logger.info("NFS DIR: #{dirs.inspect}")
|
|
|
|
@logger.info("NFS OPTS: #{opts.inspect}")
|
|
|
|
end
|
|
|
|
|
2012-01-30 19:40:29 +00:00
|
|
|
output = TemplateRenderer.render(@nfs_exports_template,
|
2011-12-12 07:22:44 +00:00
|
|
|
:uuid => id,
|
2013-07-18 04:01:49 +00:00
|
|
|
:ips => ips,
|
2013-09-01 18:44:00 +00:00
|
|
|
:folders => dirmap,
|
2013-03-27 15:15:39 +00:00
|
|
|
:user => Process.uid)
|
2010-07-13 05:10:17 +00:00
|
|
|
|
2010-07-24 07:29:46 +00:00
|
|
|
# The sleep ensures that the output is truly flushed before any `sudo`
|
|
|
|
# commands are issued.
|
2012-01-13 07:33:17 +00:00
|
|
|
@ui.info I18n.t("vagrant.hosts.bsd.nfs_export")
|
2010-07-24 07:29:46 +00:00
|
|
|
sleep 0.5
|
|
|
|
|
2012-01-13 07:27:35 +00:00
|
|
|
# First, clean up the old entry
|
|
|
|
nfs_cleanup(id)
|
|
|
|
|
|
|
|
# Output the rendered template into the exports
|
2010-07-13 05:10:17 +00:00
|
|
|
output.split("\n").each do |line|
|
2013-07-16 23:23:33 +00:00
|
|
|
line.gsub!('"', '\"')
|
|
|
|
line.gsub!("'", "'\\\\''")
|
2013-09-11 21:22:13 +00:00
|
|
|
system(%Q[sudo -s -- "echo '#{line}' >> /etc/exports"])
|
2010-07-13 05:10:17 +00:00
|
|
|
end
|
2010-07-13 05:37:24 +00:00
|
|
|
|
|
|
|
# We run restart here instead of "update" just in case nfsd
|
|
|
|
# is not starting
|
2011-12-12 07:22:44 +00:00
|
|
|
system(@nfs_restart_command)
|
2010-07-13 05:10:17 +00:00
|
|
|
end
|
2010-07-14 05:30:54 +00:00
|
|
|
|
2012-01-13 07:27:35 +00:00
|
|
|
def nfs_prune(valid_ids)
|
2012-01-14 04:38:20 +00:00
|
|
|
return if !File.exist?("/etc/exports")
|
|
|
|
|
2012-01-13 07:27:35 +00:00
|
|
|
@logger.info("Pruning invalid NFS entries...")
|
|
|
|
|
2012-01-13 07:33:17 +00:00
|
|
|
output = false
|
2013-03-27 15:15:39 +00:00
|
|
|
user = Process.uid
|
2012-01-13 07:33:17 +00:00
|
|
|
|
2012-01-13 07:27:35 +00:00
|
|
|
File.read("/etc/exports").lines.each do |line|
|
2013-03-27 15:15:39 +00:00
|
|
|
if id = line[/^# VAGRANT-BEGIN:( #{user})? ([A-Za-z0-9-]+?)$/, 2]
|
2012-03-14 04:29:38 +00:00
|
|
|
if valid_ids.include?(id)
|
|
|
|
@logger.debug("Valid ID: #{id}")
|
2012-01-13 07:27:35 +00:00
|
|
|
else
|
2012-01-13 07:33:17 +00:00
|
|
|
if !output
|
|
|
|
# We want to warn the user but we only want to output once
|
|
|
|
@ui.info I18n.t("vagrant.hosts.bsd.nfs_prune")
|
|
|
|
output = true
|
|
|
|
end
|
|
|
|
|
2012-03-14 04:29:38 +00:00
|
|
|
@logger.info("Invalid ID, pruning: #{id}")
|
|
|
|
nfs_cleanup(id)
|
2012-01-13 07:27:35 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2013-09-01 17:25:00 +00:00
|
|
|
rescue Errno::EACCES
|
|
|
|
raise Vagrant::Errors::NFSCantReadExports
|
2012-01-13 07:27:35 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
protected
|
|
|
|
|
2013-12-14 06:14:13 +00:00
|
|
|
def nfs_checkexports!
|
|
|
|
r = Subprocess.execute("nfsd", "checkexports")
|
|
|
|
if r.exit_code != 0
|
|
|
|
raise Vagrant::Errors::NFSBadExports, output: r.stderr
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2011-12-12 07:22:44 +00:00
|
|
|
def nfs_cleanup(id)
|
2010-07-30 16:38:45 +00:00
|
|
|
return if !File.exist?("/etc/exports")
|
2010-07-14 05:30:54 +00:00
|
|
|
|
2013-02-09 00:18:57 +00:00
|
|
|
# Escape sed-sensitive characters:
|
|
|
|
id = id.gsub("/", "\\/")
|
|
|
|
id = id.gsub(".", "\\.")
|
|
|
|
|
2013-03-27 15:15:39 +00:00
|
|
|
user = Process.uid
|
|
|
|
|
2012-01-13 07:27:35 +00:00
|
|
|
# Use sed to just strip out the block of code which was inserted
|
|
|
|
# by Vagrant, and restart NFS.
|
2013-09-01 18:58:52 +00:00
|
|
|
system("sudo sed -E -e '/^# VAGRANT-BEGIN:( #{user})? #{id}/,/^# VAGRANT-END:( #{user})? #{id}/ d' -ibak /etc/exports")
|
2010-07-14 05:30:54 +00:00
|
|
|
end
|
2010-07-11 05:07:10 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|