2014-01-08 04:39:24 +00:00
|
|
|
|
require "log4r"
|
|
|
|
|
|
|
|
|
|
require "vagrant/util"
|
2014-01-13 06:34:45 +00:00
|
|
|
|
require "vagrant/util/shell_quote"
|
2014-01-13 06:37:39 +00:00
|
|
|
|
require "vagrant/util/which"
|
2014-01-08 04:39:24 +00:00
|
|
|
|
|
|
|
|
|
module VagrantPlugins
|
|
|
|
|
module HostBSD
|
|
|
|
|
module Cap
|
|
|
|
|
class NFS
|
2019-10-06 04:40:54 +00:00
|
|
|
|
# On OS X 10.15, / is read-only and paths inside of /Users (and elsewhere) are mounted
|
|
|
|
|
# via a "firmlink" (which is a new invention in APFS). These must be resolved to their
|
|
|
|
|
# full path to be shareable via NFS.
|
|
|
|
|
# /Users/johnsmith/mycode becomes /System/Volumes/Data/Users/johnsmith/mycode
|
|
|
|
|
# we check to see if a path is mounted here with `df`, and prepend it.
|
|
|
|
|
#
|
|
|
|
|
# Firmlinks are only createable by the OS, so a hardcoded path should be fine, until
|
|
|
|
|
# Apple gets crazier. This wasn't supposed to be visible to applications anyway:
|
|
|
|
|
# https://developer.apple.com/videos/play/wwdc2019/710/?time=481
|
|
|
|
|
# see also https://github.com/hashicorp/vagrant/issues/10961
|
|
|
|
|
OSX_FIRMLINK_HACK = "/System/Volumes/Data"
|
|
|
|
|
|
2014-01-08 04:39:24 +00:00
|
|
|
|
def self.nfs_export(environment, ui, id, ips, folders)
|
|
|
|
|
nfs_exports_template = environment.host.capability(:nfs_exports_template)
|
|
|
|
|
nfs_restart_command = environment.host.capability(:nfs_restart_command)
|
|
|
|
|
logger = Log4r::Logger.new("vagrant::hosts::bsd")
|
|
|
|
|
|
|
|
|
|
nfs_checkexports! if File.file?("/etc/exports")
|
|
|
|
|
|
2019-10-06 04:40:54 +00:00
|
|
|
|
# Check to see if this folder is mounted 1) as APFS and 2) within the /System/Volumes/Data volume
|
|
|
|
|
# on OS X, which is a read-write "firmlink", and must be prepended so it can be shared via NFS
|
|
|
|
|
# we also need to directly mutate the :hostpath if we change it, so that it's mounted with the
|
|
|
|
|
# prefix.
|
|
|
|
|
logger.debug("Checking to see if NFS exports are in an APFS firmlink...")
|
|
|
|
|
nfs_check_folders_for_apfs folders
|
|
|
|
|
|
2014-01-08 04:39:24 +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 = {}
|
2014-02-22 02:40:10 +00:00
|
|
|
|
folders.sort_by { |_, opts| opts[:hostpath] }.each do |_, opts|
|
2014-01-08 04:39:24 +00:00
|
|
|
|
hostpath = opts[:hostpath].dup
|
|
|
|
|
hostpath.gsub!('"', '\"')
|
|
|
|
|
|
|
|
|
|
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
|
2014-02-26 16:04:21 +00:00
|
|
|
|
# the root is exported first. Also, remove duplicates so that
|
|
|
|
|
# checkexports will work properly.
|
2014-01-08 04:39:24 +00:00
|
|
|
|
dirmap.each do |dirs, _|
|
2014-02-26 16:04:21 +00:00
|
|
|
|
dirs.uniq!
|
2014-01-08 04:39:24 +00:00
|
|
|
|
dirs.sort_by! { |d| d.length }
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 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|
|
|
|
|
|
# mapall/maproot are mutually exclusive, so we have to check
|
|
|
|
|
# for both here.
|
|
|
|
|
if opt =~ /^mapall=/ || opt =~ /^maproot=/
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
output = Vagrant::Util::TemplateRenderer.render(nfs_exports_template,
|
2014-05-22 16:35:12 +00:00
|
|
|
|
uuid: id,
|
|
|
|
|
ips: ips,
|
|
|
|
|
folders: dirmap,
|
|
|
|
|
user: Process.uid)
|
2014-01-08 04:39:24 +00:00
|
|
|
|
|
|
|
|
|
# The sleep ensures that the output is truly flushed before any `sudo`
|
|
|
|
|
# commands are issued.
|
|
|
|
|
ui.info I18n.t("vagrant.hosts.bsd.nfs_export")
|
|
|
|
|
sleep 0.5
|
|
|
|
|
|
|
|
|
|
# First, clean up the old entry
|
|
|
|
|
nfs_cleanup(id)
|
|
|
|
|
|
2014-10-24 00:43:58 +00:00
|
|
|
|
# Only use "sudo" if we can't write to /etc/exports directly
|
|
|
|
|
sudo_command = ""
|
|
|
|
|
sudo_command = "sudo " if !File.writable?("/etc/exports")
|
|
|
|
|
|
2014-01-08 04:39:24 +00:00
|
|
|
|
# Output the rendered template into the exports
|
|
|
|
|
output.split("\n").each do |line|
|
2014-01-13 06:34:45 +00:00
|
|
|
|
line = Vagrant::Util::ShellQuote.escape(line, "'")
|
2014-10-24 00:43:58 +00:00
|
|
|
|
system(
|
|
|
|
|
"echo '#{line}' | " +
|
2017-06-29 23:11:18 +00:00
|
|
|
|
"#{sudo_command}/usr/bin/tee -a /etc/exports >/dev/null")
|
2014-01-08 04:39:24 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# We run restart here instead of "update" just in case nfsd
|
|
|
|
|
# is not starting
|
2014-01-13 06:34:45 +00:00
|
|
|
|
system(*nfs_restart_command)
|
2014-01-08 04:39:24 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def self.nfs_exports_template(environment)
|
2019-06-13 18:44:12 +00:00
|
|
|
|
"nfs/exports_bsd"
|
2014-01-08 04:39:24 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def self.nfs_installed(environment)
|
2014-01-13 06:37:39 +00:00
|
|
|
|
!!Vagrant::Util::Which.which("nfsd")
|
2014-01-08 04:39:24 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def self.nfs_prune(environment, ui, valid_ids)
|
|
|
|
|
return if !File.exist?("/etc/exports")
|
|
|
|
|
|
|
|
|
|
logger = Log4r::Logger.new("vagrant::hosts::bsd")
|
|
|
|
|
logger.info("Pruning invalid NFS entries...")
|
|
|
|
|
|
|
|
|
|
output = false
|
|
|
|
|
user = Process.uid
|
|
|
|
|
|
|
|
|
|
File.read("/etc/exports").lines.each do |line|
|
2015-01-20 14:04:42 +00:00
|
|
|
|
if id = line[/^# VAGRANT-BEGIN:( #{user})? ([\.\/A-Za-z0-9\-_:]+?)$/, 2]
|
2014-01-08 04:39:24 +00:00
|
|
|
|
if valid_ids.include?(id)
|
|
|
|
|
logger.debug("Valid ID: #{id}")
|
|
|
|
|
else
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
logger.info("Invalid ID, pruning: #{id}")
|
|
|
|
|
nfs_cleanup(id)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
rescue Errno::EACCES
|
|
|
|
|
raise Vagrant::Errors::NFSCantReadExports
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def self.nfs_restart_command(environment)
|
2014-01-13 06:34:45 +00:00
|
|
|
|
["sudo", "nfsd", "restart"]
|
2014-01-08 04:39:24 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
protected
|
|
|
|
|
|
|
|
|
|
def self.nfs_cleanup(id)
|
|
|
|
|
return if !File.exist?("/etc/exports")
|
|
|
|
|
|
|
|
|
|
# Escape sed-sensitive characters:
|
|
|
|
|
id = id.gsub("/", "\\/")
|
|
|
|
|
id = id.gsub(".", "\\.")
|
|
|
|
|
|
|
|
|
|
user = Process.uid
|
|
|
|
|
|
2014-10-24 00:43:58 +00:00
|
|
|
|
command = []
|
|
|
|
|
command << "sudo" if !File.writable?("/etc/exports")
|
|
|
|
|
command += [
|
|
|
|
|
"sed", "-E", "-e",
|
|
|
|
|
"/^# VAGRANT-BEGIN:( #{user})? #{id}/," +
|
|
|
|
|
"/^# VAGRANT-END:( #{user})? #{id}/ d",
|
|
|
|
|
"-ibak",
|
|
|
|
|
"/etc/exports"
|
|
|
|
|
]
|
|
|
|
|
|
2014-01-08 04:39:24 +00:00
|
|
|
|
# Use sed to just strip out the block of code which was inserted
|
|
|
|
|
# by Vagrant, and restart NFS.
|
2014-10-24 00:43:58 +00:00
|
|
|
|
system(*command)
|
2014-01-08 04:39:24 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def self.nfs_checkexports!
|
|
|
|
|
r = Vagrant::Util::Subprocess.execute("nfsd", "checkexports")
|
|
|
|
|
if r.exit_code != 0
|
|
|
|
|
raise Vagrant::Errors::NFSBadExports, output: r.stderr
|
|
|
|
|
end
|
|
|
|
|
end
|
2019-10-06 04:40:54 +00:00
|
|
|
|
|
|
|
|
|
def self.nfs_check_folders_for_apfs(folders)
|
|
|
|
|
folders.each do |_, opts|
|
2019-10-08 14:53:56 +00:00
|
|
|
|
# check to see if this path is mounted in an APFS filesystem, and if it's under the
|
|
|
|
|
# firmlink which must be prefixed. we need to use the OS X df — GNU won't notice.
|
|
|
|
|
is_mounted_apfs_command = "/bin/df -t apfs #{opts[:hostpath]}"
|
2019-10-06 04:40:54 +00:00
|
|
|
|
result = Vagrant::Util::Subprocess.execute(*Shellwords.split(is_mounted_apfs_command))
|
|
|
|
|
if (result.stdout.include? OSX_FIRMLINK_HACK)
|
|
|
|
|
opts[:hostpath].prepend(OSX_FIRMLINK_HACK)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
2014-01-08 04:39:24 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|