Skip to content

Commit

Permalink
Merge branch 'ypid/add/link_real_path_option'
Browse files Browse the repository at this point in the history
  • Loading branch information
anishathalye committed Jan 3, 2020
2 parents 3fcc13d + 320d5d0 commit ec8498f
Show file tree
Hide file tree
Showing 7 changed files with 67 additions and 16 deletions.
1 change: 1 addition & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -176,6 +176,7 @@ Available extended configuration parameters:
| `relink` | Removes the old target if it's a symlink (default:false) |
| `force` | Force removes the old target, file or folder, and forces a new link (default:false) |
| `relative` | Use a relative path to the source when creating the symlink (default:false, absolute links) |
| `canonicalize-path` | Resolve any symbolic links encountered in the source to symlink to the canonical path (default:true, real paths) |
| `glob` | Treat a `*` character as a wildcard, and perform link operations on all of those matches (default:false) |
| `if` | Execute this in your `$SHELL` and only link if it is successful. |
| `ignore-missing` | Do not fail if the source is missing and create the link anyway (default:false) |
Expand Down
4 changes: 2 additions & 2 deletions dotbot/cli.py
Original file line number Diff line number Diff line change
Expand Up @@ -73,10 +73,10 @@ def main():
if not isinstance(tasks, list):
raise ReadingError('Configuration file must be a list of tasks')
if options.base_directory:
base_directory = options.base_directory
base_directory = os.path.abspath(options.base_directory)
else:
# default to directory of config file
base_directory = os.path.dirname(os.path.realpath(options.config_file))
base_directory = os.path.dirname(os.path.abspath(options.config_file))
os.chdir(base_directory)
dispatcher = Dispatcher(base_directory)
success = dispatcher.dispatch(tasks)
Expand Down
8 changes: 6 additions & 2 deletions dotbot/context.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import copy
import os

class Context(object):
'''
Expand All @@ -13,8 +14,11 @@ def __init__(self, base_directory):
def set_base_directory(self, base_directory):
self._base_directory = base_directory

def base_directory(self):
return self._base_directory
def base_directory(self, canonical_path=True):
base_directory = self._base_directory
if canonical_path:
base_directory = os.path.realpath(base_directory)
return base_directory

def set_defaults(self, defaults):
self._defaults = defaults
Expand Down
4 changes: 2 additions & 2 deletions dotbot/dispatcher.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,8 @@ def __init__(self, base_directory):
self._load_plugins()

def _setup_context(self, base_directory):
path = os.path.abspath(os.path.realpath(
os.path.expanduser(base_directory)))
path = os.path.abspath(
os.path.expanduser(base_directory))
if not os.path.exists(path):
raise DispatchError('Nonexistent base directory')
self._context = Context(path)
Expand Down
23 changes: 13 additions & 10 deletions dotbot/plugins/link.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ def _process_links(self, links):
for destination, source in links.items():
destination = os.path.expandvars(destination)
relative = defaults.get('relative', False)
canonical_path = defaults.get('canonicalize-path', True)
force = defaults.get('force', False)
relink = defaults.get('relink', False)
create = defaults.get('create', False)
Expand All @@ -36,6 +37,7 @@ def _process_links(self, links):
# extended config
test = source.get('if', test)
relative = source.get('relative', relative)
canonical_path = source.get('canonicalize-path', canonical_path)
force = source.get('force', force)
relink = source.get('relink', relink)
create = source.get('create', create)
Expand Down Expand Up @@ -68,8 +70,8 @@ def _process_links(self, links):
if create:
success &= self._create(destination)
if force or relink:
success &= self._delete(path, destination, relative, force)
success &= self._link(path, destination, relative, ignore_missing)
success &= self._delete(path, destination, relative, canonical_path, force)
success &= self._link(path, destination, relative, canonical_path, ignore_missing)
else:
self._log.lowinfo("Globs from '" + path + "': " + str(glob_results))
glob_base = path[:glob_star_loc]
Expand All @@ -79,8 +81,8 @@ def _process_links(self, links):
if create:
success &= self._create(glob_link_destination)
if force or relink:
success &= self._delete(glob_full_item, glob_link_destination, relative, force)
success &= self._link(glob_full_item, glob_link_destination, relative, ignore_missing)
success &= self._delete(glob_full_item, glob_link_destination, relative, canonical_path, force)
success &= self._link(glob_full_item, glob_link_destination, relative, canonical_path, ignore_missing)
else:
if create:
success &= self._create(destination)
Expand All @@ -94,8 +96,8 @@ def _process_links(self, links):
(destination, path))
continue
if force or relink:
success &= self._delete(path, destination, relative, force)
success &= self._link(path, destination, relative, ignore_missing)
success &= self._delete(path, destination, relative, canonical_path, force)
success &= self._link(path, destination, relative, canonical_path, ignore_missing)
if success:
self._log.info('All links have been set up')
else:
Expand Down Expand Up @@ -159,9 +161,9 @@ def _create(self, path):
self._log.lowinfo('Creating directory %s' % parent)
return success

def _delete(self, source, path, relative, force):
def _delete(self, source, path, relative, canonical_path, force):
success = True
source = os.path.join(self._context.base_directory(), source)
source = os.path.join(self._context.base_directory(canonical_path=canonical_path), source)
fullpath = os.path.expanduser(path)
if relative:
source = self._relative_path(source, fullpath)
Expand Down Expand Up @@ -195,15 +197,16 @@ def _relative_path(self, source, destination):
destination_dir = os.path.dirname(destination)
return os.path.relpath(source, destination_dir)

def _link(self, source, link_name, relative, ignore_missing):
def _link(self, source, link_name, relative, canonical_path, ignore_missing):
'''
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)
base_directory = self._context.base_directory(canonical_path=canonical_path)
absolute_source = os.path.join(base_directory, source)
if relative:
source = self._relative_path(absolute_source, destination)
else:
Expand Down
20 changes: 20 additions & 0 deletions test/tests/link-canonicalize.bash
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
test_description='linking canonicalizes path by default'
. '../test-lib.bash'

test_expect_success 'setup' '
echo "apple" > ${DOTFILES}/f &&
ln -s dotfiles dotfiles-symlink
'

test_expect_success 'run' '
cat > "${DOTFILES}/${INSTALL_CONF}" <<EOF
- link:
~/.f:
path: f
EOF
${DOTBOT_EXEC} -c dotfiles-symlink/${INSTALL_CONF}
'

test_expect_success 'test' '
[ "$(readlink ~/.f | cut -d/ -f4-)" = "dotfiles/f" ]
'
23 changes: 23 additions & 0 deletions test/tests/link-no-canonicalize.bash
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
test_description='linking path canonicalization can be disabled'
. '../test-lib.bash'

test_expect_success 'setup' '
echo "apple" > ${DOTFILES}/f &&
ln -s dotfiles dotfiles-symlink
'

test_expect_success 'run' '
cat > "${DOTFILES}/${INSTALL_CONF}" <<EOF
- defaults:
link:
canonicalize-path: false
- link:
~/.f:
path: f
EOF
${DOTBOT_EXEC} -c ./dotfiles-symlink/${INSTALL_CONF}
'

test_expect_success 'test' '
[ "$(readlink ~/.f | cut -d/ -f4-)" = "dotfiles-symlink/f" ]
'

0 comments on commit ec8498f

Please sign in to comment.