-
Notifications
You must be signed in to change notification settings - Fork 89
/
munge.bashrc
executable file
·307 lines (284 loc) · 8.01 KB
/
munge.bashrc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
# ===========================================================================
## Paths list concatenations that avoid duplicates
# Script initially found in the /etc of a linux distribution, then
# patched by Luc Hermitte.
# Licence, likely GPL v2. To be confirmed.
#
# Examples:
# $ munge PATH "$HOME/bin"
# $ munge MANPATH "$HOME/man"
unset verbose_munge
# verbose_munge=1
## Helper functions {{{1
# _is_unset {{{2
function _is_unset()
{
[[ -z ${!1+x} ]]
}
# _is_set {{{2
function _is_set()
{
# [[ -v $1 ]] with bash 4.2+
[[ -n ${!1+x} ]]
}
# _die {{{2
function _die()
{
local msg=$1
[ -z "${msg}" ] && msg="Died"
# printf "${BASH_SOURCE[1]}:${BASH_LINENO[0]}: ${FUNCNAME[1]}: ${msg}" >&2
printf "${msg}" >&2
return 0
}
# _split_path {{{2
function _split_path()
{
local IFS=:
local res=( $1 )
echo "${res[@]}"
}
# _filter_array {{{2
# remove all occurrences of pattern from array elements
# $1: pattern
# $2+: array to filter
function _filter_array()
{
declare -a res
local pat=$1
shift
for e in "$@"; do
[[ "${e}" != "${pat}" ]] && res+=("$e")
done
echo "${res[@]}"
}
## Path list manipulation functions {{{1
# munge {{{2
function munge()
{
if [ $# -lt 2 ] ; then
echo "munge <env-variable> <new-path> [after]"
echo ""
echo "See also: change_or_munge, and remove_path"
return 1
fi
local var=$1
# local val=$(eval echo \$$1)
local val=${!1}
_is_unset "${var}" && [[ "${verbose_munge}" = "1" ]] && _die "munge <env-variable> <new-path> [after]\n<env-variable> \$$1 does not exist." && return 1
# linux -> egrep -q,
# solaris -> egrep
if [ -d "$2" ] ; then
if ! echo $val | /bin/egrep -q "(^|:)$2($|:)" ; then
if [ "$3" = "after" ] ; then
eval $var=$(echo -ne \""$val:$2"\")
else
# echo $var=$2:$val
eval $var=$(echo -ne \""$2:$val"\")
# echo $var=$(eval echo \$$var)
fi
# else
elif [ "${verbose_munge+set}" = "set" ] ; then
echo "path already in $var ($2)"
fi
else
if [[ "${verbose_munge+set}" = "set" ]] ; then
echo "inexistent path not added to $var ($2)"
fi
return 2
fi
# Add an alias over variable name to simplify its display
alias $var="echo \$$var | tr ':' '\n'"
return 0
}
# change_or_munge {{{2
# replaces $2 by $3 in $1 unless it is not present. In that case, $3 is
# munged into $1
function change_or_munge()
{
if [ $# -lt 2 ] ; then
echo "change_or_munge <env-variable> <old-path> <new-path>"
echo ""
echo "See also: munge, and remove_path"
return 1
fi
local var=$1
# local val=$(eval echo \$$1)
local val=${!1}
_is_unset "${var}" && _die "change_or_munge <env-variable> <old-path> <new-path>\n<env-variable> \$$1 does not exist." && return 1
local old=$2
local new=$3
# linux -> egrep -q,
# solaris -> egrep
if ! echo $val | /bin/egrep -q "(^|:)$2($|:)" ; then
munge "$1" "$3" "$4"
else
# echo "replace $old with $new"
val=$(echo $val | perl -pe "s#(^|:)$old(\$|:)#\\1$new\\2#")
eval $var=$(echo -ne \""$val"\")
fi
echo $val | tr ':' '\n'
return 0
}
# remove path {{{2
function remove_path()
{
if [ $# -lt 2 ] ; then
echo "remove_path <env-variable> <paths-to-remove>..."
echo ""
echo "See also: munge, and change_or_munge"
return 1
fi
local var=$1
# local val=$(eval echo \$$1)
local val=${!1}
_is_unset "${var}" && _die "remove_path <env-variable> <paths-to-remove>...\n<env-variable> \$$1 does not exist." && return 1
shift
while [ $# -ge 1 ] ; do
local old=$1
val=$(echo $val | perl -pe "s#(^|:)$old(\$|:)#\\1#;s#::#:#;s#^:|:\$##")
eval $var=$(echo -ne \""$val"\")
val=$(eval echo \$$var)
shift
done
echo $val | tr ':' '\n'
return 0
}
# lh_test() {
# echo "VAR=$1 VALUE=$(eval echo \$$1)"
# }
# ===========================================================================
# Paths cleaning {{{2
# apply a kind a O(n^2) uniq that clean duplicates
function clean_path()
{
var=$1
val=$(eval printf \$$1)
declare -a dirs
# dirs=(${val//:/ })
# split at each element, supports pathnames containing whitepaces
OLD=$IFS
local IFS=':'
dirs=(${val})
local IFS=$OLD
declare -a res
declare -a end=(${dirs[@]})
while [ ${#end[@]} -gt 0 ]
do
crt=${end[0]} # head
end=("${end[@]:1}") # tail
# remove all occurrences of $crt in the tail ($end)
duplicate=0
i=0
while [ $i -ne ${#end[@]} ] ; do
if [ "${end[$i]}" = "$crt" ] ; then
unset end[$i]
duplicate=1
else
i=$(($i+1))
fi
done
if [ $duplicate -eq 0 ] ; then
# echo "$crt"
res=( "${res[@]}" "${crt}")
fi
done
local IFS=':'
str="${res[*]}"
local IFS=$OLD
# echo $str
eval $var=$(echo -ne \""$str"\")
return 0
}
# munge_std_variables {{{2
# Given the path received, tries to fill PATH, MANPATH, LD_LIBRARY_PATH,
# PKG_CONFIG_PATH...
function munge_std_variables()
{
if [ $# -lt 1 ] ; then
echo "munge_std_variables <new-path> [after]"
return 1
fi
local path=$1
if [ ! -d "${path}" ] ; then
if [[ "${verbose_munge+set}" = "set" ]] ; then
echo "inexistent path not added to std variables (${PATH})"
fi
return 1
fi
munge PATH "${path}/bin" $2
munge LD_LIBRARY_PATH "${path}/lib" $2
munge LD_LIBRARY_PATH "${path}/lib64" $2
munge MANPATH "${path}/man" $2
munge MANPATH "${path}/share/man" $2
munge INFOPATH "${path}/info" $2
munge INFOPATH "${path}/share/info" $2
munge PKG_CONFIG_PATH "${path}/lib/pkgconfig" $2
return 0
}
## Completion {{{1
# munge {{{2
function _munge()
{
local cur=${COMP_WORDS[COMP_CWORD]}
case $COMP_CWORD in
1) COMPREPLY=( $(compgen -v ${cur}) )
# Add a space in variable names case
;;
*) COMPREPLY=( $(compgen -d ${cur}) )
# Add a trailling '/' to the result found
if [[ ${#COMPREPLY[@]} -eq 1 ]]; then
i=${COMPREPLY[0]}
if [[ "$i" == "$cur" && $i != "*/" ]]; then
COMPREPLY[0]="${i}/"
fi
fi
# Don't add a space in directories case
compopt -o nospace
;;
esac
}
complete -F _munge munge
# remove_path {{{2
function _remove_path()
{
export cas=0
local cur=${COMP_WORDS[COMP_CWORD]}
case $COMP_CWORD in
1) COMPREPLY=( $(compgen -v ${cur}) ) ;;
*)
declare -a used=( "${COMP_WORDS[@]:2}" )
unset used[${#used[@]}-1]
;& # fallthrough
2)
local envname=${COMP_WORDS[1]}
local paths=( $(_split_path ${!envname}) )
COMPREPLY=( $(compgen -W "$(printf "%s\n" "${paths[@]}")" -- ${cur}) )
if [ -n "${used}" ] ; then
cas+=*
for p in "${used[@]}" ; do
COMPREPLY=( $(_filter_array "${p}" "${COMPREPLY[@]}"))
done
fi
;;
esac
}
complete -F _remove_path remove_path
# change_or_munge {{{2
function _change_or_munge()
{
local cur=${COMP_WORDS[COMP_CWORD]}
case $COMP_CWORD in
1) COMPREPLY=( $(compgen -v ${cur}) ) ;;
2) local envname=${COMP_WORDS[1]}
local paths=( $(_split_path ${!envname}) )
COMPREPLY=( $(compgen -W "$(printf "%s\n" "${paths[@]}")" -- ${cur}) ) ;;
*) COMPREPLY=( $(compgen -d ${cur}) ) ;;
esac
}
complete -F _change_or_munge change_or_munge
# clean_path {{{2
complete -v clean_path
# munge_std_variables {{{2
complete -d munge_std_variables
# }}}1
# vim:ft=sh:fdm=marker