summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAnish Athalye <me@anishathalye.com>2018-04-13 08:49:09 -0400
committerAnish Athalye <me@anishathalye.com>2018-04-13 08:49:09 -0400
commit2f4cc0d9cb70235e7c5820dc4e7dd89b1d375fb5 (patch)
treeb2545b0ca9eb6a7a141a898b0a9d15896210a90a
parenta517c4c5d0b6bdd8ab3a62f3c6c3f38837354bde (diff)
parent972e80f18850d0cd45f22d2a4c14320f3cd204c6 (diff)
Merge branch 'glob'
-rw-r--r--README.md34
-rw-r--r--dotbot/dispatcher.py3
-rw-r--r--plugins/link.py58
-rw-r--r--test/tests/link-glob-ambiguous.bash45
-rw-r--r--test/tests/link-glob-multi-star.bash31
-rw-r--r--test/tests/link-glob.bash47
6 files changed, 199 insertions, 19 deletions
diff --git a/README.md b/README.md
index 1d6e938..06d073b 100644
--- a/README.md
+++ b/README.md
@@ -165,17 +165,22 @@ files if necessary. Environment variables in paths are automatically expanded.
Link commands are specified as a dictionary mapping targets to source
locations. Source locations are specified relative to the base directory (that
-is specified when running the installer). Directory names should *not* contain
-a trailing "/" character.
+is specified when running the installer). If linking directories, *do not* include a trailing slash.
Link commands support an (optional) extended configuration. In this type of
configuration, instead of specifying source locations directly, targets are
-mapped to extended configuration dictionaries. These dictionaries map `path` to
-the source path, specify `create` as `true` if the parent directory should be
-created if necessary, specify `relink` as `true` if incorrect symbolic links
-should be automatically overwritten, specify `force` as `true` if the file or
-directory should be forcibly linked, and specify `relative` as `true` if the
-symbolic link should have a relative path.
+mapped to extended configuration dictionaries.
+
+Available extended configuration parameters:
+
+| Link Option | Explanation |
+| -- | -- |
+| `path` | The target for the symlink, the same as in the shortcut syntax (default:null, automatic (see below)) |
+| `create` | When true, create parent directories to the link as needed. (default:false) |
+| `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 when creating the symlink (default:false, absolute links) |
+| `glob` | Treat a `*` character as a wildcard, and perform link operations on all of those matches (default:false) |
#### Example
@@ -207,6 +212,10 @@ the following three config files equivalent:
~/.zshrc:
force: true
path: zshrc
+ ~/.config/:
+ glob: true
+ path: config/*
+ relink: true
```
```yaml
@@ -217,6 +226,10 @@ the following three config files equivalent:
relink: true
~/.zshrc:
force: true
+ ~/.config/:
+ glob: true
+ path: config/*
+ relink: true
```
```json
@@ -230,6 +243,11 @@ the following three config files equivalent:
},
"~/.zshrc": {
"force": true
+ },
+ "~/.config/": {
+ "glob": true,
+ "path": "config/*",
+ "relink": true
}
}
}
diff --git a/dotbot/dispatcher.py b/dotbot/dispatcher.py
index cc07435..d1a4f95 100644
--- a/dotbot/dispatcher.py
+++ b/dotbot/dispatcher.py
@@ -30,10 +30,11 @@ class Dispatcher(object):
try:
success &= plugin.handle(action, task[action])
handled = True
- except Exception:
+ except Exception as err:
self._log.error(
'An error was encountered while executing action %s' %
action)
+ self._log.debug(err)
if not handled:
success = False
self._log.error('Action %s not handled' % action)
diff --git a/plugins/link.py b/plugins/link.py
index 4b50320..5274e3b 100644
--- a/plugins/link.py
+++ b/plugins/link.py
@@ -1,4 +1,5 @@
import os
+import glob
import shutil
import dotbot
@@ -27,26 +28,62 @@ class Link(dotbot.Plugin):
force = defaults.get('force', False)
relink = defaults.get('relink', False)
create = defaults.get('create', False)
+ 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)
+ 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))
- 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 create:
- success &= self._create(destination)
- if force or relink:
- success &= self._delete(path, destination, relative, force)
- success &= self._link(path, destination, relative)
+ 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:
+ self._log.lowinfo("Globs from '" + path + "': " + str(glob_results))
+ glob_base = path[:glob_star_loc]
+ for glob_full_item in glob_results:
+ glob_item = glob_full_item[len(glob_base):]
+ glob_link_destination = os.path.join(destination, glob_item)
+ 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)
+ 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)
if success:
self._log.info('All links have been set up')
else:
@@ -87,6 +124,7 @@ class Link(dotbot.Plugin):
success = True
parent = os.path.abspath(os.path.join(os.path.expanduser(path), os.pardir))
if not self._exists(parent):
+ self._log.debug("Try to create parent: " + str(parent))
try:
os.makedirs(parent)
except OSError:
diff --git a/test/tests/link-glob-ambiguous.bash b/test/tests/link-glob-ambiguous.bash
new file mode 100644
index 0000000..7e23348
--- /dev/null
+++ b/test/tests/link-glob-ambiguous.bash
@@ -0,0 +1,45 @@
+test_description='link glob ambiguous'
+. '../test-lib.bash'
+
+test_expect_success 'setup' '
+mkdir ${DOTFILES}/foo
+'
+
+test_expect_failure 'run 1' '
+run_dotbot <<EOF
+- link:
+ ~/foo/:
+ path: foo
+ glob: true
+EOF
+'
+
+test_expect_failure 'test 1' '
+test -d ~/foo
+'
+
+test_expect_failure 'run 2' '
+run_dotbot <<EOF
+- link:
+ ~/foo/:
+ path: foo/
+ glob: true
+EOF
+'
+
+test_expect_failure 'test 2' '
+test -d ~/foo
+'
+
+test_expect_success 'run 3' '
+run_dotbot <<EOF
+- link:
+ ~/foo:
+ path: foo
+ glob: true
+EOF
+'
+
+test_expect_success 'test 3' '
+test -d ~/foo
+'
diff --git a/test/tests/link-glob-multi-star.bash b/test/tests/link-glob-multi-star.bash
new file mode 100644
index 0000000..11ae740
--- /dev/null
+++ b/test/tests/link-glob-multi-star.bash
@@ -0,0 +1,31 @@
+test_description='link glob'
+. '../test-lib.bash'
+
+test_expect_success 'setup' '
+mkdir ${DOTFILES}/config &&
+mkdir ${DOTFILES}/config/foo &&
+mkdir ${DOTFILES}/config/bar &&
+echo "apple" > ${DOTFILES}/config/foo/a &&
+echo "banana" > ${DOTFILES}/config/bar/b &&
+echo "cherry" > ${DOTFILES}/config/bar/c
+'
+
+test_expect_success 'run' '
+run_dotbot -v <<EOF
+- defaults:
+ link:
+ glob: true
+ create: true
+- link:
+ ~/.config/: config/*/*
+EOF
+'
+
+test_expect_success 'test' '
+! readlink ~/.config/ &&
+! readlink ~/.config/foo &&
+readlink ~/.config/foo/a &&
+grep "apple" ~/.config/foo/a &&
+grep "banana" ~/.config/bar/b &&
+grep "cherry" ~/.config/bar/c
+'
diff --git a/test/tests/link-glob.bash b/test/tests/link-glob.bash
new file mode 100644
index 0000000..f1c813d
--- /dev/null
+++ b/test/tests/link-glob.bash
@@ -0,0 +1,47 @@
+test_description='link glob'
+. '../test-lib.bash'
+
+test_expect_success 'setup 1' '
+mkdir ${DOTFILES}/bin &&
+echo "apple" > ${DOTFILES}/bin/a &&
+echo "banana" > ${DOTFILES}/bin/b &&
+echo "cherry" > ${DOTFILES}/bin/c
+'
+
+test_expect_success 'run 1' '
+run_dotbot -v <<EOF
+- defaults:
+ link:
+ glob: true
+ create: true
+- link:
+ ~/bin: bin/*
+EOF
+'
+
+test_expect_success 'test 1' '
+grep "apple" ~/bin/a &&
+grep "banana" ~/bin/b &&
+grep "cherry" ~/bin/c
+'
+
+test_expect_success 'setup 2' '
+rm -rf ~/bin
+'
+
+test_expect_success 'run 2' '
+run_dotbot -v <<EOF
+- defaults:
+ link:
+ glob: true
+ create: true
+- link:
+ ~/bin/: bin/*
+EOF
+'
+
+test_expect_success 'test 2' '
+grep "apple" ~/bin/a &&
+grep "banana" ~/bin/b &&
+grep "cherry" ~/bin/c
+'