1
0
mirror of https://github.com/deadc0de6/dotdrop.git synced 2026-02-09 12:54:18 +00:00

inherit actions from profile include (for #152)

This commit is contained in:
deadc0de6
2019-06-07 09:55:08 +02:00
parent 1015672a40
commit a89fae29cf
2 changed files with 239 additions and 27 deletions

View File

@@ -36,7 +36,6 @@ class CfgYaml:
action_post = 'post' action_post = 'post'
# profiles/dotfiles entries # profiles/dotfiles entries
key_profiles_dotfiles = 'dotfiles'
key_dotfile_src = 'src' key_dotfile_src = 'src'
key_dotfile_dst = 'dst' key_dotfile_dst = 'dst'
key_dotfile_link = 'link' key_dotfile_link = 'link'
@@ -44,9 +43,11 @@ class CfgYaml:
key_dotfile_link_children = 'link_children' key_dotfile_link_children = 'link_children'
# profile # profile
key_profile_dotfiles = 'dotfiles'
key_profile_include = 'include' key_profile_include = 'include'
key_profile_variables = 'variables' key_profile_variables = 'variables'
key_profile_dvariables = 'dynvariables' key_profile_dvariables = 'dynvariables'
key_profile_actions = 'actions'
key_all = 'ALL' key_all = 'ALL'
# import entries # import entries
@@ -470,44 +471,72 @@ class CfgYaml:
"""resolve some other parts of the config""" """resolve some other parts of the config"""
# profile -> ALL # profile -> ALL
for k, v in self.profiles.items(): for k, v in self.profiles.items():
dfs = v.get(self.key_profiles_dotfiles, None) dfs = v.get(self.key_profile_dotfiles, None)
if not dfs: if not dfs:
continue continue
if self.debug: if self.debug:
self.log.dbg('add ALL to profile {}'.format(k)) self.log.dbg('add ALL to profile {}'.format(k))
if self.key_all in dfs: if self.key_all in dfs:
v[self.key_profiles_dotfiles] = self.dotfiles.keys() v[self.key_profile_dotfiles] = self.dotfiles.keys()
# profiles -> include other profile # profiles -> include other profile
for k, v in self.profiles.items(): for k, v in self.profiles.items():
self._rec_resolve_profile_include(k) self._rec_resolve_profile_include(k)
def _rec_resolve_profile_include(self, profile): def _rec_resolve_profile_include(self, profile):
"""recursively resolve include of other profiles's dotfiles""" """
values = self.profiles[profile] recursively resolve include of other profiles's:
current = values.get(self.key_profiles_dotfiles, []) * dotfiles
inc = values.get(self.key_profile_include, None) * actions
if not inc: """
return current this_profile = self.profiles[profile]
# include
dotfiles = this_profile.get(self.key_profile_dotfiles, [])
actions = this_profile.get(self.key_profile_actions, [])
includes = this_profile.get(self.key_profile_include, None)
if not includes:
# nothing to include
return dotfiles, actions
if self.debug:
self.log.dbg('{} includes: {}'.format(profile, ','.join(includes)))
self.log.dbg('{} dotfiles before include: {}'.format(profile,
dotfiles))
self.log.dbg('{} actions before include: {}'.format(profile,
actions))
seen = [] seen = []
for i in inc: for i in uniq_list(includes):
# ensure no include loop occurs
if i in seen: if i in seen:
raise YamlException('\"include loop\"') raise YamlException('\"include loop\"')
seen.append(i) seen.append(i)
# included profile even exists
if i not in self.profiles.keys(): if i not in self.profiles.keys():
self.log.warn('include unknown profile: {}'.format(i)) self.log.warn('include unknown profile: {}'.format(i))
continue continue
p = self.profiles[i] # recursive resolve
others = p.get(self.key_profiles_dotfiles, []) o_dfs, o_actions = self._rec_resolve_profile_include(i)
if self.key_profile_include in p.keys(): # merge dotfile keys
others.extend(self._rec_resolve_profile_include(i)) dotfiles.extend(o_dfs)
current.extend(others) this_profile[self.key_profile_dotfiles] = uniq_list(dotfiles)
# unique them # merge actions keys
values[self.key_profiles_dotfiles] = uniq_list(current) actions.extend(o_actions)
this_profile[self.key_profile_actions] = uniq_list(actions)
dotfiles = this_profile.get(self.key_profile_dotfiles, [])
actions = this_profile.get(self.key_profile_actions, [])
if self.debug: if self.debug:
dfs = values[self.key_profiles_dotfiles] self.log.dbg('{} dotfiles after include: {}'.format(profile,
self.log.dbg('{} dfs after include: {}'.format(profile, dfs)) dotfiles))
return values.get(self.key_profiles_dotfiles, []) self.log.dbg('{} actions after include: {}'.format(profile,
actions))
# since dotfiles and actions are resolved here
# and variables have been already done at the beginning
# of the parsing, we can clear these include
self.profiles[profile][self.key_profile_include] = None
return dotfiles, actions
def _resolve_path(self, path): def _resolve_path(self, path):
"""resolve a path either absolute or relative to config path""" """resolve a path either absolute or relative to config path"""
@@ -580,7 +609,7 @@ class CfgYaml:
if key not in self.profiles.keys(): if key not in self.profiles.keys():
# update yaml_dict # update yaml_dict
self.yaml_dict[self.key_profiles][key] = { self.yaml_dict[self.key_profiles][key] = {
self.key_profiles_dotfiles: [] self.key_profile_dotfiles: []
} }
if self.debug: if self.debug:
self.log.dbg('adding new profile: {}'.format(key)) self.log.dbg('adding new profile: {}'.format(key))
@@ -590,8 +619,8 @@ class CfgYaml:
"""add an existing dotfile key to a profile_key""" """add an existing dotfile key to a profile_key"""
self._new_profile(profile_key) self._new_profile(profile_key)
profile = self.yaml_dict[self.key_profiles][profile_key] profile = self.yaml_dict[self.key_profiles][profile_key]
if dotfile_key not in profile[self.key_profiles_dotfiles]: if dotfile_key not in profile[self.key_profile_dotfiles]:
profile[self.key_profiles_dotfiles].append(dotfile_key) profile[self.key_profile_dotfiles].append(dotfile_key)
if self.debug: if self.debug:
msg = 'add \"{}\" to profile \"{}\"'.format(dotfile_key, msg = 'add \"{}\" to profile \"{}\"'.format(dotfile_key,
profile_key) profile_key)
@@ -641,15 +670,15 @@ class CfgYaml:
return False return False
# get the profile dictionary # get the profile dictionary
profile = self.yaml_dict[self.key_profiles][pro_key] profile = self.yaml_dict[self.key_profiles][pro_key]
if df_key not in profile[self.key_profiles_dotfiles]: if df_key not in profile[self.key_profile_dotfiles]:
return True return True
if self.debug: if self.debug:
dfs = profile[self.key_profiles_dotfiles] dfs = profile[self.key_profile_dotfiles]
self.log.dbg('{} profile dotfiles: {}'.format(pro_key, dfs)) self.log.dbg('{} profile dotfiles: {}'.format(pro_key, dfs))
self.log.dbg('remove {} from profile {}'.format(df_key, pro_key)) self.log.dbg('remove {} from profile {}'.format(df_key, pro_key))
profile[self.key_profiles_dotfiles].remove(df_key) profile[self.key_profile_dotfiles].remove(df_key)
if self.debug: if self.debug:
dfs = profile[self.key_profiles_dotfiles] dfs = profile[self.key_profile_dotfiles]
self.log.dbg('{} profile dotfiles: {}'.format(pro_key, dfs)) self.log.dbg('{} profile dotfiles: {}'.format(pro_key, dfs))
self.dirty = True self.dirty = True
return True return True

183
tests-ng/include-actions.sh Executable file
View File

@@ -0,0 +1,183 @@
#!/usr/bin/env bash
# author: deadc0de6 (https://github.com/deadc0de6)
# Copyright (c) 2019, deadc0de6
#
# test the use of the keyword "include"
# with action inheritance
# returns 1 in case of error
#
# exit on first error
set -e
# all this crap to get current path
rl="readlink -f"
if ! ${rl} "${0}" >/dev/null 2>&1; then
rl="realpath"
if ! hash ${rl}; then
echo "\"${rl}\" not found !" && exit 1
fi
fi
cur=$(dirname "$(${rl} "${0}")")
#hash dotdrop >/dev/null 2>&1
#[ "$?" != "0" ] && echo "install dotdrop to run tests" && exit 1
#echo "called with ${1}"
# dotdrop path can be pass as argument
ddpath="${cur}/../"
[ "${1}" != "" ] && ddpath="${1}"
[ ! -d ${ddpath} ] && echo "ddpath \"${ddpath}\" is not a directory" && exit 1
export PYTHONPATH="${ddpath}:${PYTHONPATH}"
bin="python3 -m dotdrop.dotdrop"
echo "dotdrop path: ${ddpath}"
echo "pythonpath: ${PYTHONPATH}"
# get the helpers
source ${cur}/helpers
echo -e "\e[96m\e[1m==> RUNNING $(basename $BASH_SOURCE) <==\e[0m"
################################################################
# this is the test
################################################################
# the dotfile source
tmps=`mktemp -d --suffix='-dotdrop-tests'`
mkdir -p ${tmps}/dotfiles
# the dotfile destination
tmpd=`mktemp -d --suffix='-dotdrop-tests'`
# the action temp
tmpa=`mktemp -d --suffix='-dotdrop-tests'`
# create the config file
cfg="${tmps}/config.yaml"
cat > ${cfg} << _EOF
actions:
pre:
preaction: echo 'pre' >> ${tmpa}/pre
preaction2: echo 'pre2' >> ${tmpa}/pre2
post:
postaction: echo 'post' >> ${tmpa}/post
postaction2: echo 'post2' >> ${tmpa}/post2
nakedaction: echo 'naked' >> ${tmpa}/naked
config:
backup: true
create: true
dotpath: dotfiles
dotfiles:
f_abc:
dst: ${tmpd}/abc
src: abc
profiles:
p0:
include:
- p3
p1:
dotfiles:
- f_abc
actions:
- preaction
- postaction
p2:
include:
- p1
actions:
- preaction2
- postaction2
p3:
include:
- p2
actions:
- nakedaction
_EOF
# create the source
mkdir -p ${tmps}/dotfiles/
echo "test" > ${tmps}/dotfiles/abc
# install
echo "PROFILE p2"
cd ${ddpath} | ${bin} install -f -c ${cfg} -p p2 -V
# checks
[ ! -e ${tmpa}/pre ] && echo "pre not found" && exit 1
nb=`wc -l ${tmpa}/pre | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "pre executed multiple times" && exit 1
[ ! -e ${tmpa}/pre2 ] && echo "pre2 not found" && exit 1
nb=`wc -l ${tmpa}/pre2 | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "pre2 executed multiple times" && exit 1
[ ! -e ${tmpa}/post ] && echo "post not found" && exit 1
nb=`wc -l ${tmpa}/post | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "post executed multiple times" && exit 1
[ ! -e ${tmpa}/post2 ] && echo "post2 not found" && exit 1
nb=`wc -l ${tmpa}/post2 | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "post2 executed multiple times" && exit 1
# install
rm -f ${tmpa}/pre ${tmpa}/pre2 ${tmpa}/post ${tmpa}/post2 ${tmpa}/naked
rm -f ${tmpd}/abc
echo "PROFILE p3"
cd ${ddpath} | ${bin} install -f -c ${cfg} -p p3 -V
# checks
[ ! -e ${tmpa}/pre ] && echo "pre not found" && exit 1
nb=`wc -l ${tmpa}/pre | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "pre executed multiple times" && exit 1
[ ! -e ${tmpa}/pre2 ] && echo "pre2 not found" && exit 1
nb=`wc -l ${tmpa}/pre2 | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "pre2 executed multiple times" && exit 1
[ ! -e ${tmpa}/post ] && echo "post not found" && exit 1
nb=`wc -l ${tmpa}/post | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "post executed multiple times" && exit 1
[ ! -e ${tmpa}/post2 ] && echo "post2 not found" && exit 1
nb=`wc -l ${tmpa}/post2 | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "post2 executed multiple times" && exit 1
[ ! -e ${tmpa}/naked ] && echo "naked not found" && exit 1
nb=`wc -l ${tmpa}/naked | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "naked executed multiple times" && exit 1
# install
rm -f ${tmpa}/pre ${tmpa}/pre2 ${tmpa}/post ${tmpa}/post2 ${tmpa}/naked
rm -f ${tmpd}/abc
echo "PROFILE p0"
cd ${ddpath} | ${bin} install -f -c ${cfg} -p p0 -V
# checks
[ ! -e ${tmpa}/pre ] && echo "pre not found" && exit 1
nb=`wc -l ${tmpa}/pre | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "pre executed multiple times" && exit 1
[ ! -e ${tmpa}/pre2 ] && echo "pre2 not found" && exit 1
nb=`wc -l ${tmpa}/pre2 | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "pre2 executed multiple times" && exit 1
[ ! -e ${tmpa}/post ] && echo "post not found" && exit 1
nb=`wc -l ${tmpa}/post | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "post executed multiple times" && exit 1
[ ! -e ${tmpa}/post2 ] && echo "post2 not found" && exit 1
nb=`wc -l ${tmpa}/post2 | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "post2 executed multiple times" && exit 1
[ ! -e ${tmpa}/naked ] && echo "naked not found" && exit 1
nb=`wc -l ${tmpa}/naked | awk '{print $1}'`
[ "${nb}" != "1" ] && echo "naked executed multiple times" && exit 1
## CLEANING
rm -rf ${tmps} ${tmpd} ${tmpa}
echo "OK"
exit 0