1
0
Fork 0
mirror of synced 2024-11-26 18:15:35 -05:00
dotbot/plugins/link.py

249 lines
10 KiB
Python
Raw Normal View History

2016-11-19 10:42:02 -05:00
import os
2018-01-27 02:27:44 -05:00
import glob
2016-11-19 10:42:02 -05:00
import shutil
import dotbot
import subprocess
import copy
2016-11-19 10:42:02 -05:00
2014-03-19 23:07:30 -04:00
2016-01-16 22:00:15 -05:00
class Link(dotbot.Plugin):
2014-03-19 23:07:30 -04:00
'''
Symbolically links dotfiles.
'''
2014-04-24 15:41:34 -04:00
_directive = 'link'
2018-04-14 15:39:52 -04:00
_config_if_key = 'if'
2014-04-24 15:41:34 -04:00
2014-03-19 23:07:30 -04:00
def can_handle(self, directive):
2014-04-24 15:41:34 -04:00
return directive == self._directive
2014-03-19 23:07:30 -04:00
def handle(self, directive, data):
2014-04-24 15:41:34 -04:00
if directive != self._directive:
2016-01-16 22:00:15 -05:00
raise ValueError('Link cannot handle directive %s' % directive)
2018-04-14 15:39:52 -04:00
data_filtered = copy.deepcopy(data)
2018-05-24 10:34:48 -04:00
for d_key in list(data_filtered):
if isinstance(data_filtered[d_key], dict) and self._config_if_key in data_filtered[d_key]:
self._log.debug("testing conditional: %s" % data_filtered[d_key][self._config_if_key])
commandsuccess = self._test_success(data_filtered[d_key][self._config_if_key])
2018-04-14 15:39:52 -04:00
if commandsuccess:
# command successful, treat as normal link
2018-05-24 10:34:48 -04:00
del data_filtered[d_key][self._config_if_key]
2018-04-14 15:39:52 -04:00
else:
# remove the item from data, we aren't going to link it
2018-05-24 10:34:48 -04:00
del data_filtered[d_key]
2018-04-14 15:39:52 -04:00
return self._process_links(data_filtered)
2014-03-19 23:07:30 -04:00
def _process_links(self, links):
success = True
defaults = self._context.defaults().get('link', {})
2014-03-19 23:07:30 -04:00
for destination, source in links.items():
destination = os.path.expandvars(destination)
relative = defaults.get('relative', False)
force = defaults.get('force', False)
relink = defaults.get('relink', False)
create = defaults.get('create', False)
2018-04-06 12:29:28 -04:00
use_glob = defaults.get('glob', False)
if isinstance(source, dict):
# extended config
relative = source.get('relative', relative)
force = source.get('force', force)
relink = source.get('relink', relink)
create = source.get('create', create)
2018-04-06 12:29:28 -04:00
use_glob = source.get('glob', use_glob)
path = self._default_source(destination, source.get('path'))
else:
path = self._default_source(destination, source)
path = os.path.expandvars(os.path.expanduser(path))
2018-01-27 02:27:44 -05:00
if use_glob:
self._log.debug("Globbing with path: " + str(path))
glob_results = glob.glob(path)
if len(glob_results) is 0:
self._log.warning("Globbing couldn't find anything matching " + str(path))
success = False
continue
glob_star_loc = path.find('*')
if glob_star_loc is -1 and destination[-1] is '/':
self._log.error("Ambiguous action requested.")
self._log.error("No wildcard in glob, directory use undefined: " +
destination + " -> " + str(glob_results))
self._log.warning("Did you want to link the directory or into it?")
success = False
continue
elif glob_star_loc is -1 and len(glob_results) is 1:
# perform a normal link operation
if create:
success &= self._create(destination)
if force or relink:
success &= self._delete(path, destination, relative, force)
success &= self._link(path, destination, relative)
else:
2018-01-27 04:11:11 -05:00
self._log.lowinfo("Globs from '" + path + "': " + str(glob_results))
2018-01-27 02:27:44 -05:00
glob_base = path[:glob_star_loc]
for glob_full_item in glob_results:
glob_item = glob_full_item[len(glob_base):]
2018-04-06 12:56:16 -04:00
glob_link_destination = os.path.join(destination, glob_item)
2018-01-27 02:27:44 -05:00
if create:
success &= self._create(glob_link_destination)
2018-01-27 04:11:11 -05:00
if force or relink:
success &= self._delete(glob_full_item, glob_link_destination, relative, force)
2018-01-27 02:27:44 -05:00
success &= self._link(glob_full_item, glob_link_destination, relative)
else:
if create:
success &= self._create(destination)
if not self._exists(os.path.join(self._context.base_directory(), path)):
success = False
self._log.warning('Nonexistent target %s -> %s' %
(destination, path))
continue
if force or relink:
success &= self._delete(path, destination, relative, force)
success &= self._link(path, destination, relative)
2014-03-19 23:07:30 -04:00
if success:
self._log.info('All links have been set up')
else:
self._log.error('Some links were not successfully set up')
return success
2018-04-14 15:39:52 -04:00
def _test_success(self, testcommand):
try:
osstdout = subprocess.check_output(
testcommand,
stderr=subprocess.STDOUT,
shell=True,
executable=os.environ.get("SHELL", "/bin/sh")
)
except subprocess.CalledProcessError as e:
# self._log.warning("Command failed: " + str(testcommand))
self._log.debug("Command returned "+str(e.returncode)+": \n" + str(testcommand))
return False
return True
def _default_source(self, destination, source):
if source is None:
basename = os.path.basename(destination)
if basename.startswith('.'):
return basename[1:]
else:
return basename
else:
return source
2014-03-19 23:07:30 -04:00
def _is_link(self, path):
'''
Returns true if the path is a symbolic link.
'''
return os.path.islink(os.path.expanduser(path))
def _link_destination(self, path):
'''
Returns the destination of the symbolic link.
2014-03-19 23:07:30 -04:00
'''
path = os.path.expanduser(path)
return os.readlink(path)
2014-03-19 23:07:30 -04:00
def _exists(self, path):
'''
Returns true if the path exists.
'''
path = os.path.expanduser(path)
return os.path.exists(path)
def _create(self, path):
success = True
parent = os.path.abspath(os.path.join(os.path.expanduser(path), os.pardir))
if not self._exists(parent):
2018-01-27 02:27:44 -05:00
self._log.debug("Try to create parent: " + str(parent))
try:
os.makedirs(parent)
except OSError:
self._log.warning('Failed to create directory %s' % parent)
success = False
else:
self._log.lowinfo('Creating directory %s' % parent)
return success
def _delete(self, source, path, relative, force):
success = True
source = os.path.join(self._context.base_directory(), source)
fullpath = os.path.expanduser(path)
if relative:
source = self._relative_path(source, fullpath)
if ((self._is_link(path) and self._link_destination(path) != source) or
(self._exists(path) and not self._is_link(path))):
removed = False
try:
if os.path.islink(fullpath):
os.unlink(fullpath)
removed = True
elif force:
if os.path.isdir(fullpath):
shutil.rmtree(fullpath)
removed = True
else:
os.remove(fullpath)
removed = True
except OSError:
self._log.warning('Failed to remove %s' % path)
success = False
else:
if removed:
self._log.lowinfo('Removing %s' % path)
return success
def _relative_path(self, source, destination):
'''
Returns the relative path to get to the source file from the
destination file.
'''
destination_dir = os.path.dirname(destination)
return os.path.relpath(source, destination_dir)
def _link(self, source, link_name, relative):
2014-03-19 23:07:30 -04:00
'''
Links link_name to source.
Returns true if successfully linked files.
'''
success = False
destination = os.path.expanduser(link_name)
absolute_source = os.path.join(self._context.base_directory(), source)
if relative:
source = self._relative_path(absolute_source, destination)
else:
source = absolute_source
if (not self._exists(link_name) and self._is_link(link_name) and
self._link_destination(link_name) != source):
2014-03-19 23:07:30 -04:00
self._log.warning('Invalid link %s -> %s' %
(link_name, self._link_destination(link_name)))
# we need to use absolute_source below because our cwd is the dotfiles
# directory, and if source is relative, it will be relative to the
# destination directory
elif not self._exists(link_name) and self._exists(absolute_source):
try:
os.symlink(source, destination)
2014-08-20 16:58:42 -04:00
except OSError:
self._log.warning('Linking failed %s -> %s' % (link_name, source))
else:
self._log.lowinfo('Creating link %s -> %s' % (link_name, source))
success = True
2014-03-19 23:07:30 -04:00
elif self._exists(link_name) and not self._is_link(link_name):
self._log.warning(
'%s already exists but is a regular file or directory' %
link_name)
elif self._is_link(link_name) and self._link_destination(link_name) != source:
2014-03-19 23:07:30 -04:00
self._log.warning('Incorrect link %s -> %s' %
(link_name, self._link_destination(link_name)))
# again, we use absolute_source to check for existence
elif not self._exists(absolute_source):
if self._is_link(link_name):
self._log.warning('Nonexistent target %s -> %s' %
(link_name, source))
else:
self._log.warning('Nonexistent target for %s : %s' %
(link_name, source))
2014-03-19 23:07:30 -04:00
else:
self._log.lowinfo('Link exists %s -> %s' % (link_name, source))
success = True
return success