]> git.proxmox.com Git - mirror_frr.git/blame - tools/frrcommon.sh.in
lib: add frr-isisd to the native models
[mirror_frr.git] / tools / frrcommon.sh.in
CommitLineData
ea4d91bf
DL
1#!/bin/sh
2#
3# This is a "library" of sorts for use by the other FRR shell scripts. It
4# has most of the daemon start/stop logic, but expects the following shell
5# functions/commands to be provided by the "calling" script:
6#
7# log_success_msg
8# log_warning_msg
9# log_failure_msg
10#
11# (coincidentally, these are LSB standard functions.)
12#
13# Sourcing this file in a shell script will load FRR config variables but
14# not perform any action. Note there is an "exit 1" if the main config
15# file does not exist.
16#
17# This script should be installed in @CFG_SBIN@/frrcommon.sh
18
19PATH=/bin:/usr/bin:/sbin:/usr/sbin
20D_PATH="@CFG_SBIN@" # /usr/lib/frr
21C_PATH="@CFG_SYSCONF@" # /etc/frr
22V_PATH="@CFG_STATE@" # /var/run/frr
23VTYSH="@vtysh_bin@" # /usr/bin/vtysh
24FRR_USER="@enable_user@" # frr
25FRR_GROUP="@enable_group@" # frr
26FRR_VTY_GROUP="@enable_vty_group@" # frrvty
27
28# ORDER MATTERS FOR $DAEMONS!
29# - keep zebra first
30# - watchfrr does NOT belong in this list
31
32DAEMONS="zebra bgpd ripd ripngd ospfd ospf6d isisd babeld pimd ldpd nhrpd eigrpd sharpd pbrd staticd bfdd fabricd"
33RELOAD_SCRIPT="$D_PATH/frr-reload.py"
34
35#
36# general helpers
37#
38
39debug() {
40 [ -n "$watchfrr_debug" ] || return 0
41
42 printf '%s %s(%s):' "`date +%Y-%m-%dT%H:%M:%S.%N`" "$0" $$ >&2
43 # this is to show how arguments are split regarding whitespace & co.
44 # (e.g. for use with `debug "message" "$@"`)
45 while [ $# -gt 0 ]; do
46 printf ' "%s"' "$1" >&2
47 shift
48 done
49 printf '\n' >&2
50}
51
52chownfrr() {
53 [ -n "$FRR_USER" ] && chown "$FRR_USER" "$1"
54 [ -n "$FRR_GROUP" ] && chgrp "$FRR_GROUP" "$1"
55}
56
57vtysh_b () {
58 [ "$1" = "watchfrr" ] && return 0
59 [ -r "$C_PATH/frr.conf" ] || return 0
60 if [ -n "$1" ]; then
61 "$VTYSH" -b -n -d "$1"
62 else
63 "$VTYSH" -b -n
64 fi
65}
66
67daemon_inst() {
68 # note this sets global variables ($dmninst, $daemon, $inst)
69 dmninst="$1"
70 daemon="${dmninst%:*}"
71 inst=""
72 [ "$daemon" != "$dmninst" ] && inst="${dmninst#*:}"
73}
74
75daemon_list() {
76 # note $1 and $2 specify names for global variables to be set
77 local enabled disabled evar dvar
78 enabled=""
79 disabled=""
80 evar="$1"
81 dvar="$2"
82
83 for daemon in $DAEMONS; do
84 eval cfg=\$$daemon
85 eval inst=\$${daemon}_instances
86 [ "$daemon" = zebra ] && cfg=yes
87 if [ -n "$cfg" -a "$cfg" != "no" -a "$cfg" != "0" ]; then
88 debug "$daemon enabled"
89 enabled="$enabled $daemon"
90 if [ -n "$inst" ]; then
91 debug "$daemon multi-instance $inst"
92 for i in $inst; do
93 enabled="$enabled $daemon:$inst"
94 done
95 fi
96 else
97 debug "$daemon disabled"
98 disabled="$disabled $daemon"
99 fi
100 done
101
102 enabled="${enabled# }"
103 disabled="${disabled# }"
104 [ -z "$evar" ] && echo "$enabled"
105 [ -n "$evar" ] && eval $evar="\"$enabled\""
106 [ -n "$dvar" ] && eval $dvar="\"$disabled\""
107}
108
109#
110# individual daemon management
111#
112
113daemon_prep() {
114 local daemon inst cfg
115 daemon="$1"
116 inst="$2"
117 [ "$daemon" = "watchfrr" ] && return 0
118 [ -x "$D_PATH/$daemon" ] || {
119 log_failure_msg "cannot start $daemon${inst:+ (instance $inst)}: daemon binary not installed\n"
120 return 1
121 }
122 [ -r "$C_PATH/frr.conf" ] && return 0
123
124 cfg="$C_PATH/$daemon${inst:+-$inst}.conf"
125 if [ ! -r "$cfg" ]; then
126 touch "$cfg"
127 chownfrr "$cfg"
128 fi
129 return 0
130}
131
132daemon_start() {
133 local dmninst daemon inst args instopt wrap bin
134 daemon_inst "$1"
135
136 ulimit -n $MAX_FDS > /dev/null 2> /dev/null
137 daemon_prep "$daemon" "$inst" || return 1
138
139 eval wrap="\$${daemon}_wrap"
140 bin="$D_PATH/$daemon"
141 instopt="${inst:+-n $inst}"
142 eval args="\$${daemon}_options"
143
144 if eval "$all_wrap $wrap $bin -d $instopt $args"; then
145 log_success_msg "Started $dmninst"
146 vtysh_b "$daemon"
147 else
148 log_failure_msg "Failed to start $dmninst!"
149 fi
150}
151
152daemon_stop() {
153 local dmninst daemon inst pidfile vtyfile pid cnt fail
154 daemon_inst "$1"
155
156 pidfile="$V_PATH/$daemon${inst:+-$inst}.pid"
157 vtyfile="$V_PATH/$daemon${inst:+-$inst}.vty"
158
159 [ -r "$pidfile" ] || fail="pid file not found"
160 [ -z "$fail" ] && pid="`cat \"$pidfile\"`"
161 [ -z "$fail" -a -z "$pid" ] && fail="pid file is empty"
162 [ -n "$fail" ] || kill -0 "$pid" 2>/dev/null || fail="pid $pid not running"
163
164 if [ -n "$fail" ]; then
165 log_failure_msg "Cannot stop $dmninst: $fail"
166 return 1
167 fi
168
169 debug "kill -2 $pid"
170 kill -2 "$pid"
171 cnt=1200
172 while kill -0 "$pid" 2>/dev/null; do
173 sleep .1
174 [ $(( cnt -= 1 )) -gt 0 ] || break
175 done
176 if kill -0 "$pid" 2>/dev/null; then
177 log_failure_msg "Failed to stop $dmninst, pid $pid still running"
178 still_running=1
179 return 1
180 else
181 log_success_msg "Stopped $dmninst"
182 rm -f "$pidfile"
183 return 0
184 fi
185}
186
187daemon_status() {
188 local dmninst daemon inst pidfile pid fail
189 daemon_inst "$1"
190
191 pidfile="$V_PATH/$daemon${inst:+-$inst}.pid"
192
193 [ -r "$pidfile" ] || return 3
194 pid="`cat \"$pidfile\"`"
195 [ -z "$pid" ] && return 1
196 kill -0 "$pid" 2>/dev/null || return 1
197 return 0
198}
199
200print_status() {
201 daemon_status "$1"
202 rv=$?
203 if [ "$rv" -eq 0 ]; then
204 log_success_msg "Status of $1: running"
205 else
206 log_failure_msg "Status of $1: FAILED"
207 fi
208 return $rv
209}
210
211#
212# all-daemon commands
213#
214
215all_start() {
216 daemon_list daemons
217 for dmninst in $daemons; do
218 daemon_start "$dmninst"
219 done
220}
221
222all_stop() {
223 local pids reversed
224
225 daemon_list daemons disabled
226 [ "$1" = "--reallyall" ] && daemons="$daemons $disabled"
227
228 reversed=""
229 for dmninst in $daemons; do
230 reversed="$dmninst $reversed"
231 done
232
233 for dmninst in $reversed; do
234 daemon_stop "$dmninst" &
235 pids="$pids $!"
236 done
237 for pid in $pids; do
238 wait $pid
239 done
240}
241
242all_status() {
243 local fail
244
245 daemon_list daemons
246 fail=0
247 for dmninst in $daemons; do
248 print_status "$dmninst" || fail=1
249 done
250 return $fail
251}
252
253#
254# config sourcing
255#
256
257load_old_config() {
258 oldcfg="$1"
259 [ -r "$oldcfg" ] || return 0
260 [ -s "$oldcfg" ] || return 0
261 grep -v '^[[:blank:]]*\(#\|$\)' "$oldcfg" > /dev/null || return 0
262
263 log_warning_msg "Reading deprecated $oldcfg. Please move its settings to $C_PATH/daemons and remove it."
264
265 # save off settings from daemons for the OR below
266 for dmn in $DAEMONS; do eval "_new_$dmn=\${$dmn:-no}"; done
267
268 . "$oldcfg"
269
270 # OR together the daemon enabling options between config files
271 for dmn in $DAEMONS; do eval "test \$_new_$dmn != no && $dmn=\$_new_$dmn; unset _new_$dmn"; done
272}
273
274[ -r "$C_PATH/daemons" ] || {
275 log_failure_msg "cannot run $@: $C_PATH/daemons does not exist\n"
276 exit 1
277}
278. "$C_PATH/daemons"
279
280load_old_config "$C_PATH/daemons.conf"
281load_old_config "/etc/default/frr"
282load_old_config "/etc/sysconfig/frr"
283
284#
285# other defaults and dispatch
286#
287
288frrcommon_main() {
289 local cmd
290
291 debug "frrcommon_main" "$@"
292
293 cmd="$1"
294 shift
295
296 if [ "$1" = "all" -o -z "$1" ]; then
297 case "$cmd" in
298 start) all_start;;
299 stop) all_stop;;
300 restart)
301 all_stop
302 all_start
303 ;;
304 *) $cmd "$@";;
305 esac
306 else
307 case "$cmd" in
308 start) daemon_start "$@";;
309 stop) daemon_stop "$@";;
310 restart)
311 daemon_stop "$@"
312 daemon_start "$@"
313 ;;
314 *) $cmd "$@";;
315 esac
316 fi
317}