-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathregen-examples.sh
executable file
·160 lines (138 loc) · 4.66 KB
/
regen-examples.sh
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
#!/bin/bash
# shellcheck disable=SC1090,SC1091
set -o errexit
set -o pipefail
set -o nounset
set -o noclobber
RED=$(tput setaf 1)
GREEN=$(tput setaf 2)
YELLOW=$(tput setaf 3)
BLUE=$(tput setaf 4)
RESET=$(tput sgr0)
debug() {
echo "${BLUE}DEBUG:${RESET} $*"
}
info() {
echo "${GREEN}INFO:${RESET} $*"
}
warn() {
echo "${YELLOW}WARN:${RESET} $*"
}
error() {
echo "${RED}ERROR:${RESET} $*" >&2
}
usage() {
echo "Usage: $0 [--help] [[EXAMPLE_STEP] ...]"
echo
echo "Generate the example data, and the project README.md"
echo
echo " --help, -h Show this help and exit"
echo
echo " EXAMPLE_STEP One of the examples/*/*.sh scripts"
}
setup_environment() {
info "Setting up environment ..."
local this_script_source="${BASH_SOURCE[0]}"
local repo_root
repo_root="$(cd -P "$(dirname "$this_script_source")" >/dev/null 2>&1 && pwd)/"
repo_root="$(readlink --canonicalize --no-newline "${repo_root}")"
cd "$repo_root"
if [[ ! -f .venv/bin/activate ]]; then
info "Python virtualenv not found. Attempting to create"
python -m venv --prompt generative .venv
source .venv/bin/activate
python -m pip install -r requirements.txt
else
source .venv/bin/activate
fi
cargo build --release
export PATH="$repo_root/target/release/:$PATH"
}
extract_snippet() {
local snippet="$1"
local gobble_leading_whitespace="${2:-0}"
local source_file="${3:-${BASH_SOURCE[1]}}"
# Extract the contents between '# BEGIN $snippet' and '# END $snippet'
local snippet_contents
snippet_contents="$(sed -En "/^[[:space:]]*# BEGIN $snippet\\b/,/^[[:space:]]*# END $snippet\\b/p" "$source_file" | sed '1d;$d')"
snippet_contents="$(echo -e "$snippet_contents" | sed -E "s/^[[:space:]]{$gobble_leading_whitespace}//")"
# Populate the variable with name '$snippet' with '$snippet_contents'
printf -v "$snippet" '%s' "$snippet_contents"
}
configure() {
local template="$1"
local final_output="$2"
local dry_run="${3:-false}"
local temp_output
temp_output="$(mktemp "$template.$final_output.XXXXXXXXX")"
# shellcheck disable=SC2064
trap "rm '$temp_output'" EXIT
info "Generating '$final_output' from template '$template'"
cp "$template" "$temp_output"
local placeholders
placeholders=$(grep --only-matching --extended-regexp '@[a-zA-Z_]+[a-zA-Z0-9_]*@' "$temp_output" || true)
for placeholder in $placeholders; do
local placeholder_var="${placeholder//@/}"
if [[ -n ${!placeholder_var+x} ]]; then
local placeholder_value="${!placeholder_var}"
# There are a few symbols that are special in the context of a sed replacement
# expression.
placeholder_value="${placeholder_value//\\/\\\\}"
# Actual newlines should be replaced with the two characters '\' and 'n'
placeholder_value="${placeholder_value//$'\n'/'\n'}"
# Ampersands are placeholders for the matched pattern, so they need to be escaped
placeholder_value="${placeholder_value//&/\\&}"
# debug "Found placeholder '$placeholder' with value '$placeholder_value'"
debug "Processing placeholder '$placeholder' ..."
sed -i "s~$placeholder~${placeholder_value}~" "$temp_output"
else
warn "Could not find substitution for '$placeholder'"
fi
done
if [[ "$dry_run" = "false" ]]; then
cp "$temp_output" "$final_output"
fi
}
main() {
local examples=()
while [[ $# -gt 0 ]]; do
case "$1" in
--help | -h)
usage
exit 0
;;
-*)
error "Unexpected option: $1"
usage >&2
exit 1
;;
*)
examples+=("$1")
;;
esac
shift
done
setup_environment
# If example scripts have been passed manually, run just them, otherwise run everything.
if [[ "${#examples[@]}" -gt 0 ]]; then
for example in "${examples[@]}"; do
if [[ ! -f "$example" ]]; then
error "Example '$example' doesn't exist"
exit 1
fi
info "Generating example: $example ..."
source "$example"
done
else
shopt -s globstar
for example in ./examples/**/*.sh; do
info "Generating example: $example ..."
source "$example"
done
fi
debug "Finished generating examples"
# Generate the README at the end, so that we don't leave it in a broken state, and so that we
# can optionally run just a single example.
configure README.md.in README.md
}
main "$@"