#!/usr/bin/env sh LOGDIR="$XDG_CONFIG_HOME"/sxmo/modem trap "gracefulexit" INT TERM fatalerr() { # E.g. hangup all calls, switch back to default audio, notify user, and die sxmo_vibratepine 1000 mmcli -m "$(mmcli -L | grep -qoE 'Modem\/([0-9]+)')" --voice-hangup-all alsactl --file /usr/share/sxmo/default_alsa_sound.conf restore notify-send "$1" kill -9 0 } gracefulexit() { fatalerr "Terminated via SIGTERM/SIGINT" } modem_n() { MODEMS="$(mmcli -L)" echo "$MODEMS" | grep -qoE 'Modem\/([0-9]+)' || fatalerr "Couldn't find modem - is your modem enabled?" echo "$MODEMS" | grep -oE 'Modem\/([0-9]+)' | cut -d'/' -f2 } contacts() { RES="$(cut -f3 "$LOGDIR/modemlog.tsv" | sort | uniq | awk NF)" echo "$RES" printf %b "$RES" | grep -q 8042221111 || echo "Test Number 8042221111" } modem_cmd_errcheck() { # shellcheck disable=SC2068 RES="$(mmcli $@ 2>&1)" OK="$?" echo "Command: mmcli $*" if [ "$OK" != 0 ]; then fatalerr "Problem executing mmcli command!\n$RES"; fi echo "$RES" } vid_to_number() { mmcli -m "$(modem_n)" -o "$1" -K | grep call.properties.number | cut -d ':' -f2 | tr -d ' ' | sed 's/^[+]//' | sed 's/^1//' } log_event() { EVT_HANDLE="$1" EVT_VID="$2" NUM="$(vid_to_number "$EVT_VID")" TIME="$(date --iso-8601=seconds)" mkdir -p "$LOGDIR" printf %b "$TIME\t$EVT_HANDLE\t$NUM\n" >> "$LOGDIR/modemlog.tsv" } toggleflag() { TOGGLEFLAG=$1 shift FLAGS="$*" echo -- "$FLAGS" | grep -- "$TOGGLEFLAG" >&2 && NEWFLAGS="$(echo -- "$FLAGS" | sed "s/$TOGGLEFLAG//g")" || NEWFLAGS="$(echo -- "$FLAGS $TOGGLEFLAG")" NEWFLAGS="$(echo -- "$NEWFLAGS" | sed "s/--//g; s/ / /g")" # shellcheck disable=SC2086 sxmo_megiaudioroute $NEWFLAGS echo -- "$NEWFLAGS" } toggleflagset() { FLAGS="$(toggleflag "$1" "$FLAGS")" } dialmenu() { CONTACTS="$(contacts)" NUMBER="$( printf %b "Close Menu\n$CONTACTS" | grep . | sxmo_dmenu_with_kb.sh -l 10 -p Number -c -fn Terminus-20 )" echo "$NUMBER" | grep "Close Menu" && kill 0 NUMBER="$(echo "$NUMBER" | awk -F' ' '{print $NF}' | tr -d -)" echo "$NUMBER" | grep -qE '^[+0-9]+$' || fatalerr "$NUMBER is not a number" echo "Attempting to dial: $NUMBER" >&2 CALLID="$( mmcli -m "$(modem_n)" --voice-create-call "number=$NUMBER" | grep -Eo "Call/[0-9]+" | grep -oE "[0-9]+" )" echo "Starting call with CALLID: $CALLID" >&2 startcall "$CALLID" >&2 echo "$CALLID" } startcall() { CALLID="$1" #modem_cmd_errcheck --voice-status -o $CALLID modem_cmd_errcheck -m "$(modem_n)" -o "$CALLID" --start log_event "call_start" "$CALLID" } acceptcall() { CALLID="$1" echo "Attempting to pickup CALLID $CALLID" #mmcli --voice-status -o $CALLID modem_cmd_errcheck -m "$(modem_n)" -o "$CALLID" --accept log_event "call_pickup" "$CALLID" } hangup() { CALLID="$1" modem_cmd_errcheck -m "$(modem_n)" -o "$CALLID" --hangup log_event "call_hangup" "$CALLID" fatalerr "Call hungup ok" exit 1 } togglewindowify() { if [ "$WINDOWIFIED" = "0" ]; then WINDOWIFIED=1 else WINDOWIFIED=0 fi } incallsetup() { DMENUIDX=0 CALLID="$1" NUMBER="$(vid_to_number "$CALLID")" WINDOWIFIED=0 # E.g. There's some bug with the modem that' requires us to toggle the # DAI a few times before starting the call for it to kick in FLAGS=" " toggleflagset "-e" toggleflagset "-m" toggleflagset "-2" toggleflagset "-2" toggleflagset "-2" } incallmonitor() { CALLID="$1" while true; do echo 1 > /tmp/sxmo_bar if mmcli -m "$(modem_n)" -K -o "$CALLID" | grep -E "^call.properties.state.+terminated"; then fatalerr "$NUMBER hung up the call" fi echo "Call still in progress" sleep 3 done } incallmenuloop() { CHOICES=" Volume ↑ ^ sxmo_vol.sh up Volume ↓ ^ sxmo_vol.sh down Mic $(echo -- "$FLAGS" | grep -q -- -m && echo ✓) ^ toggleflagset -m Linemic $(echo -- "$FLAGS" | grep -q -- -l && echo ✓) ^ toggleflagset -l Echomic $(echo -- "$FLAGS" | grep -q -- -z && echo ✓) ^ toggleflagset -z Earpiece $(echo -- "$FLAGS" | grep -q -- -e && echo ✓) ^ toggleflagset -e Linejack $(echo -- "$FLAGS" | grep -q -- -h && echo ✓) ^ toggleflagset -h Speakerphone $(echo -- "$FLAGS" | grep -q -- -s && echo ✓) ^ toggleflagset -s DTMF Tones ^ dtmfmenu $CALLID Hangup ^ hangup $CALLID $([ "$WINDOWIFIED" = 0 ] && echo Windowify || echo Unwindowify) ^ togglewindowify " echo "$CHOICES" | xargs -0 echo | cut -d'^' -f1 | sed '/^[[:space:]]*$/d' | awk '{$1=$1};1' | dmenu -idx $DMENUIDX -l 14 "$([ "$WINDOWIFIED" = 0 ] && echo "-c" || echo "-wm")" -fn "Terminus-30" -p "$NUMBER" | ( PICKED="$(cat)"; echo "$PICKED" | grep -Ev "." && fatalerr "$NUMBER hung up the call" CMD=$(echo "$CHOICES" | grep "$PICKED" | cut -d '^' -f2) DMENUIDX=$(echo "$(echo "$CHOICES" | grep -n "$PICKED" | cut -d ':' -f1)" - 1 | bc) eval "$CMD" incallmenuloop ) & wait # E.g. bg & wait to allow for SIGINT propogation } dtmfmenu() { CALLID="$1" DTMFINDEX=0 NUMS="0123456789*#ABCD" while true; do PICKED="$( echo "$NUMS" | grep -o . | sed '1 iReturn to Call Menu' | dmenu -l 20 -fn Terminus-20 -c -idx $DTMFINDEX -p "DTMF Tone" )" echo "$PICKED" | grep "Return to Call Menu" && return DTMFINDEX=$(echo "$NUMS" | grep -bo "$PICKED" | cut -d: -f1 | xargs -IN echo 2+N | bc) modem_cmd_errcheck -m "$(modem_n)" -o "$CALLID" --send-dtmf="$PICKED" done } dial() { CALLID="$(dialmenu)" incallsetup "$CALLID" incallmonitor "$CALLID" & incallmenuloop "$CALLID" } pickup() { acceptcall "$1" incallsetup "$1" incallmonitor "$1" & incallmenuloop "$1" } modem_n || fatalerr "Couldn't determine modem number - is modem online?" "$1" "$2"