-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathtui-modules-logic.sh
More file actions
executable file
·334 lines (290 loc) · 9.94 KB
/
tui-modules-logic.sh
File metadata and controls
executable file
·334 lines (290 loc) · 9.94 KB
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
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
#!/bin/bash
# ═══════════════════════════════════════════════════════════════
# Modules Logic - Module definitions, state, toggle & POM ops
# Sourced by: modules
# Requires: SCRIPT_DIR set by caller, tui-common.sh sourced
#
# Module definitions are read from modules-conf.txt in the
# project root (SCRIPT_DIR). See modules-conf.txt.template
# for the expected format.
# ═══════════════════════════════════════════════════════════════
PLAINTEXT_CONFIG_DIR="${PLAINTEXT_CONFIG_DIR:-$HOME/codeplain/plaintext-config}"
_project_name=$(basename "$SCRIPT_DIR")
# Resolve modules-conf.txt: plaintext-config first, then project-local
if [[ -f "$PLAINTEXT_CONFIG_DIR/$_project_name/modules-conf.txt" ]]; then
MODULES_CONF="$PLAINTEXT_CONFIG_DIR/$_project_name/modules-conf.txt"
elif [[ -f "$SCRIPT_DIR/modules-conf.txt" ]]; then
MODULES_CONF="$SCRIPT_DIR/modules-conf.txt"
else
echo "ERROR: modules-conf.txt not found (checked plaintext-config and $SCRIPT_DIR)" >&2
exit 1
fi
ROOT_POM="$SCRIPT_DIR/pom.xml"
STATE_FILE="$SCRIPT_DIR/.modules-state"
# Webapp POM path: read WEBAPP_MODULE from build-conf.txt or plaintext-build.cfg
_WEBAPP_MODULE=""
_build_conf=""
if [[ -f "$PLAINTEXT_CONFIG_DIR/$_project_name/build-conf.txt" ]]; then
_build_conf="$PLAINTEXT_CONFIG_DIR/$_project_name/build-conf.txt"
elif [[ -f "$SCRIPT_DIR/build-conf.txt" ]]; then
_build_conf="$SCRIPT_DIR/build-conf.txt"
fi
if [[ -n "$_build_conf" ]]; then
while IFS='=' read -r key value; do
[[ "$key" =~ ^#.*$ ]] && continue
[[ -z "$key" ]] && continue
value="${value%%#*}"
value="${value%"${value##*[![:space:]]}"}"
if [[ "$key" == "WEBAPP_MODULE" ]]; then
_WEBAPP_MODULE="$value"
fi
done < "$_build_conf"
WEBAPP_POM="$SCRIPT_DIR/${_WEBAPP_MODULE}/pom.xml"
else
# Fallback: try to find from modules-conf.txt WEBAPP_POM setting
WEBAPP_POM=""
fi
# ── Module definitions (loaded from modules-conf.txt) ────────
ROOT_MODULES=()
# Toggleable modules - parallel arrays for bash 3.2
TOGGLE_NAMES=()
TOGGLE_STATE=()
TOGGLE_GROUP=() # group name (e.g. "root", "admin", "feature")
TOGGLE_DISPLAY=() # short display name
add_module() {
TOGGLE_NAMES+=("$1")
TOGGLE_STATE+=(1)
TOGGLE_GROUP+=("$2")
TOGGLE_DISPLAY+=("$3")
}
# Cross-dependencies between toggleable modules
# Format: "module|dependency" (module needs dependency to build)
DEPS=()
# Modules that cannot be toggled off
LOCKED_TOGGLE=()
# ── Parse modules-conf.txt ────────────────────────────────────
# Format:
# ROOT_MODULE=<name>
# TOGGLE=<name>|<group>|<display>
# DEP=<module>|<dependency>
# LOCKED=<name>
# WEBAPP_POM=<relative-path> (optional override)
while IFS= read -r line; do
# Skip comments and empty lines
[[ "$line" =~ ^[[:space:]]*# ]] && continue
[[ -z "${line// }" ]] && continue
# Strip inline comments
line="${line%%#*}"
# Trim trailing whitespace
line="${line%"${line##*[![:space:]]}"}"
local_key="${line%%=*}"
local_value="${line#*=}"
case "$local_key" in
ROOT_MODULE)
ROOT_MODULES+=("$local_value")
;;
TOGGLE)
# Format: name|group|display
IFS='|' read -r t_name t_group t_display <<< "$local_value"
add_module "$t_name" "$t_group" "$t_display"
;;
DEP)
DEPS+=("$local_value")
;;
LOCKED)
LOCKED_TOGGLE+=("$local_value")
;;
WEBAPP_POM)
WEBAPP_POM="$SCRIPT_DIR/$local_value"
;;
esac
done < "$MODULES_CONF"
TOTAL=${#TOGGLE_NAMES[@]}
# Validate WEBAPP_POM is set
if [[ -z "$WEBAPP_POM" || ! -f "$WEBAPP_POM" ]]; then
echo "WARNING: WEBAPP_POM not found at '$WEBAPP_POM'. POM manipulation may fail." >&2
fi
# ── Index / Lock helpers ─────────────────────────────────────
# Find index of a module by name
idx_of() {
local name="$1" i
for ((i=0; i<TOTAL; i++)); do
[[ "${TOGGLE_NAMES[$i]}" == "$name" ]] && echo $i && return
done
echo -1
}
is_locked() {
local name="$1" l
for l in "${LOCKED_TOGGLE[@]}"; do
[[ "$l" == "$name" ]] && return 0
done
return 1
}
# ── Dependency resolution ────────────────────────────────────
# When enabling a module, auto-enable its dependencies
resolve_enable() {
local name="$1" dep_entry dep mod
for dep_entry in "${DEPS[@]}"; do
mod="${dep_entry%%|*}"
dep="${dep_entry##*|}"
if [[ "$mod" == "$name" ]]; then
local di
di=$(idx_of "$dep")
if [[ $di -ge 0 && "${TOGGLE_STATE[$di]}" == "0" ]]; then
TOGGLE_STATE[$di]=1
resolve_enable "$dep" # recursive
fi
fi
done
}
# When disabling a module, auto-disable modules that depend on it
resolve_disable() {
local name="$1" dep_entry dep mod
for dep_entry in "${DEPS[@]}"; do
mod="${dep_entry%%|*}"
dep="${dep_entry##*|}"
if [[ "$dep" == "$name" ]]; then
local mi
mi=$(idx_of "$mod")
if [[ $mi -ge 0 && "${TOGGLE_STATE[$mi]}" == "1" ]]; then
TOGGLE_STATE[$mi]=0
resolve_disable "$mod" # recursive
fi
fi
done
}
# Toggle with dependency resolution
do_toggle() {
local idx="$1"
local name="${TOGGLE_NAMES[$idx]}"
# Locked modules can't be toggled
if is_locked "$name"; then
return
fi
if [[ "${TOGGLE_STATE[$idx]}" == "1" ]]; then
TOGGLE_STATE[$idx]=0
resolve_disable "$name"
else
TOGGLE_STATE[$idx]=1
resolve_enable "$name"
fi
}
# ── State persistence ────────────────────────────────────────
load_state() {
if [[ -f "$STATE_FILE" ]]; then
local i
for ((i=0; i<TOTAL; i++)); do
local mod="${TOGGLE_NAMES[$i]}"
local val
val=$(grep "^${mod}=" "$STATE_FILE" 2>/dev/null | cut -d= -f2)
if [[ "$val" == "0" ]]; then
TOGGLE_STATE[$i]=0
fi
done
fi
# Enforce locked modules are always on
local i
for ((i=0; i<TOTAL; i++)); do
if is_locked "${TOGGLE_NAMES[$i]}"; then
TOGGLE_STATE[$i]=1
fi
done
}
save_state() {
local i
> "$STATE_FILE"
for ((i=0; i<TOTAL; i++)); do
echo "${TOGGLE_NAMES[$i]}=${TOGGLE_STATE[$i]}" >> "$STATE_FILE"
done
}
has_disabled() {
local i
for ((i=0; i<TOTAL; i++)); do
[[ "${TOGGLE_STATE[$i]}" == "0" ]] && return 0
done
return 1
}
count_enabled() {
local count=0 i
for ((i=0; i<TOTAL; i++)); do
[[ "${TOGGLE_STATE[$i]}" == "1" ]] && count=$((count + 1))
done
echo $count
}
# ── POM manipulation ─────────────────────────────────────────
apply_changes() {
# Restore everything first, then disable selected modules
# Collect disabled module names
local disabled_mods=""
local i
for ((i=0; i<TOTAL; i++)); do
if [[ "${TOGGLE_STATE[$i]}" == "0" ]]; then
disabled_mods="${disabled_mods} ${TOGGLE_NAMES[$i]}"
fi
done
# Root pom.xml: restore all then disable
sed -i.bak 's|<!-- <module>\(.*\)</module> -->|<module>\1</module>|g' "$ROOT_POM"
rm -f "${ROOT_POM}.bak"
for mod in $disabled_mods; do
sed -i.bak "s|<module>${mod}</module>|<!-- <module>${mod}</module> -->|g" "$ROOT_POM"
rm -f "${ROOT_POM}.bak"
done
# Webapp pom.xml: use Python with sys.argv to avoid escaping issues
if [[ -n "$WEBAPP_POM" && -f "$WEBAPP_POM" ]]; then
python3 - "$WEBAPP_POM" $disabled_mods <<'PYEOF'
import re, sys
pom_path = sys.argv[1]
disabled = sys.argv[2:]
with open(pom_path, 'r') as f:
content = f.read()
# First restore any previously disabled
content = re.sub(r'<!-- DISABLED\n(.*?)\n-->', r'\1', content, flags=re.DOTALL)
# Now disable each module
for mod in disabled:
pattern = (
r'( <dependency>\n'
r' <groupId>ch\.plaintext</groupId>\n'
r' <artifactId>' + re.escape(mod) + r'</artifactId>'
r'(?:\n <version>[^<]*</version>)?'
r'\n </dependency>)'
)
content = re.sub(pattern, r'<!-- DISABLED' + '\n' + r'\1' + '\n' + '-->', content)
with open(pom_path, 'w') as f:
f.write(content)
PYEOF
fi
}
restore_all() {
sed -i.bak 's|<!-- <module>\(.*\)</module> -->|<module>\1</module>|g' "$ROOT_POM"
rm -f "${ROOT_POM}.bak"
if [[ -n "$WEBAPP_POM" && -f "$WEBAPP_POM" ]]; then
python3 - "$WEBAPP_POM" <<'PYEOF'
import re, sys
pom_path = sys.argv[1]
with open(pom_path, 'r') as f:
content = f.read()
content = re.sub(r'<!-- DISABLED\n(.*?)\n-->', r'\1', content, flags=re.DOTALL)
with open(pom_path, 'w') as f:
f.write(content)
PYEOF
fi
rm -f "$STATE_FILE"
}
save_and_apply() {
echo -e "${FG_BLUE}${BOLD}Applying module changes...${RESET}"
if ! has_disabled; then
restore_all
echo -e "${FG_GREEN}All modules enabled (full build)${RESET}"
else
save_state
apply_changes
enabled=$(count_enabled)
echo -e "${FG_GREEN}${enabled}/${TOTAL} modules enabled${RESET}"
echo -e "${FG_DIM}Disabled:${RESET}"
for ((i=0; i<TOTAL; i++)); do
if [[ "${TOGGLE_STATE[$i]}" == "0" ]]; then
echo -e " ${FG_DIM} ${TOGGLE_NAMES[$i]}${RESET}"
fi
done
fi
}