Project

General

Profile

1
#!/bin/bash -e
2
# runscripts: a bash-based replacement for make
3
# unlike make, supports full bash functionality including multiline commands
4
# usage: .../run function args
5
# this usage also applies to all files that include this file
6

    
7
if false; then #### runscript template:
8
#!/bin/bash -e
9
auto_fwd=1 # optional
10
. "$(dirname "${BASH_SOURCE[0]}")"/path/to/util_or_file_including_util.run "$@"
11
.rel other_includes
12

    
13
target()
14
{
15
	begin_target
16
	#...
17
}
18

    
19
all()
20
{
21
	begin_target
22
	target # defined above
23
	"$(dirname "${BASH_SOURCE[0]}")"/path_relative_to_self
24
	"$(dirname "${BASH_SOURCE[1]}")"/path_relative_to_caller
25
	"$top_dir"/path_relative_to_outermost_script
26
}
27

    
28
on_exit # needed if should be runnable as shell-include (with leading ".")
29
fi ####
30

    
31
. "$(dirname "${BASH_SOURCE[0]}")"/../sh/util.sh
32
.rel ../sh/make.sh
33

    
34
if self_not_included; then
35

    
36

    
37
#### setup
38

    
39
if is_dot_script; then
40
	if test "$1" = .; then set --; fi # $@ wrong: no args->contains script name
41
	include_args=("$@") # from `. .../util.run "$@"`
42
fi
43

    
44
fallback() # overridable handler for targets w/o function
45
{ echo_func; "$@"; } # by default, generate error that it doesn't exist
46

    
47
gateway() # overridable handler for *all* targets
48
{ echo_func; if is_callable "$1"; then "$@"; else fallback "$@"; fi; }
49

    
50
run_args_cmd() # runs the command line args command
51
{
52
	if is_dot_script; then set -- "${include_args[@]}"
53
	else              eval set -- "$(reverse "${BASH_ARGV[@]}")"
54
	fi
55
	test $# -ge 1 || set -- all
56
	echo_cmd "$top_script" "$@"
57
	indent; time gateway "$@"
58
}
59

    
60
# users can override run_args_cmd to perform other commands (or no commands)
61
# after the script is read
62
on_exit() { test $? -eq 0 || return; run_args_cmd; }
63
if ! is_dot_script; then trap on_exit EXIT; fi
64

    
65
func_override set_paths__util_sh
66
set_paths()
67
{
68
	set_paths__util_sh "$@"
69
	top_file="${top_script%[./]run}"; echo_vars top_file
70
	top_filename="$(basename "$top_file")"; echo_vars top_filename
71
}
72
set_paths
73

    
74

    
75
#### utils
76

    
77
wrap_fn="$top_script" # usage: sudo "$wrap_fn" fn ...
78

    
79
# usage: to_top_file cmd...
80
function to_top_file() { echo_func; stdout="$top_file" to_file "$@"; }
81
alias to_top_file='"to_top_file" ' # last space alias-expands next word
82

    
83
,() # usage: .../run , cmd1 cmd2 ... (like `make cmd1 cmd2 ...`)
84
# treats each of the command-line args as commands the way make does
85
# (instead of as args to the same command, the way runscripts do)
86
{ echo_func; for cmd in "$@"; do time echo_run "$cmd"; done; }
87

    
88
: "${auto_ignore=}" #usage: auto_ignore=1 .../__.run target_that_might_not_exist
89
unexport auto_ignore # don't pass this to invoked scripts except through fwd()
90
if test "$auto_ignore"; then
91
fallback() # don't generate error that it doesn't exist
92
{ echo_func; log_info "ignoring non-existant target: $*"; }
93
fi
94

    
95
# by default, use all subdirs, including .*
96
if ! isset subdirs; then subdirs=($(enter_top_dir; wildcard. {.,}*/)); fi
97
log-- echo_vars subdirs
98

    
99
fwd() # usage: [subdirs=(...);] fwd target args... # or use fwd_self
100
{
101
	echo_func
102
	: "${subdirs?}"
103
	
104
	local_export auto_ignore=1
105
	for subdir in "${subdirs[@]}"; do
106
		local runscript="$top_dir"/"$subdir"/run
107
		if test -f "$runscript"; then "$runscript" "$@"; fi
108
	done
109
}
110
alias fwd_self='fwd "$FUNCNAME" "$@"' # usage: target() { ...; fwd_self; }
111

    
112
: "${auto_fwd=}" # usage: auto_fwd=1; . .../file_including_util.run
113
# must be set *after* auto_ignore's fallback() to overwrite it
114
if test "$auto_fwd"; then
115
fallback() { echo_func; fwd "$@"; }
116
fi
117

    
118
fi
(14-14/16)