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 #### run script 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
all()
14
{
15
	echo_func
16
	"$(dirname "${BASH_SOURCE[0]}")"/path_relative_to_self
17
	"$(dirname "${BASH_SOURCE[1]}")"/path_relative_to_caller
18
	"$top_dir"/path_relative_to_outermost_script
19
}
20

    
21
on_exit # needed if should be runnable as shell-include (with leading ".")
22
fi ####
23

    
24
. "$(dirname "${BASH_SOURCE[0]}")"/../sh/util.sh
25
.rel ../sh/make.sh
26

    
27
if self_not_included; then
28

    
29

    
30
#### setup
31

    
32
if is_dot_script; then
33
	if test "$1" = .; then set --; fi # $@ wrong: no args->contains script name
34
	include_args=("$@") # from `. .../util.run "$@"`
35
fi
36

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

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

    
43
run_args_cmd() # runs the command line args command
44
{
45
	if is_dot_script; then set -- "${include_args[@]}"
46
	else              eval set -- "$(reverse "${BASH_ARGV[@]}")"
47
	fi
48
	test $# -ge 1 || set -- all
49
	echo_cmd "$top_script" "$@"; time gateway "$@"
50
}
51

    
52
# users can override run_args_cmd to perform other commands (or no commands)
53
# after the script is read
54
on_exit() { test $? -eq 0 || return; run_args_cmd; }
55
if ! is_dot_script; then trap on_exit EXIT; fi
56

    
57
func_override set_paths__util_sh
58
set_paths()
59
{
60
	set_paths__util_sh "$@"
61
	top_file="${top_script%[./]run}"; echo_vars top_file
62
	top_filename="$(basename "$top_file")"; echo_vars top_filename
63
}
64
set_paths
65

    
66

    
67
#### utils
68

    
69
wrap_fn="$top_script" # usage: sudo "$wrap_fn" fn ...
70

    
71
# usage: to_top_file cmd...
72
function to_top_file() { echo_func; stdout="$top_file" to_file "$@"; }
73
alias to_top_file='"to_top_file" ' # last space alias-expands next word
74

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

    
80
: "${auto_ignore=}" #usage: auto_ignore=1 .../__.run target_that_might_not_exist
81
if test "$auto_ignore"; then
82
fallback() # don't generate error that it doesn't exist
83
{ echo_func; log_info "ignoring non-existant target: $*"; }
84
fi
85

    
86
# by default, use all subdirs, including .*
87
if ! isset subdirs; then subdirs=($(enter_top_dir; wildcard. {.,}*/)); fi
88
clog-- echo_vars subdirs
89

    
90
fwd() # usage: [subdirs=(...);] fwd target args... # or use fwd_self
91
{
92
	echo_func
93
	: "${subdirs?}"
94
	
95
	local_export auto_ignore=1
96
	for subdir in "${subdirs[@]}"; do
97
		local runscript="$top_dir"/"$subdir"/run
98
		if test -f "$runscript"; then "$runscript" "$@"; fi
99
	done
100
}
101
alias fwd_self='fwd "$FUNCNAME" "$@"' # usage: target() { ...; fwd_self; }
102

    
103
: "${auto_fwd=}" # usage: auto_fwd=1; . .../file_including_util.run
104
# must be set *after* auto_ignore's fallback() to overwrite it
105
if test "$auto_fwd"; then
106
fallback() { echo_func; fwd "$@"; }
107
fi
108

    
109
fi
(14-14/16)