#!/bin/bash # /lib/systemd/system-sleep/rtcwake # chmod a+rx AWAKE_TIME="01:23:45" # hour:minute:second GAP="10 seconds" # precision of awake time in date(1) format WAIT_TRIGGER='true' # test to go back to sleep when action is finished # typically pgrep on user or for specific process WAIT_TRIGGER_DELAY=60 # sleep duration in seconds between two tests with WAIT_TRIGGER MAX_TRIGGER=60 # don't go back to sleep if WAIT_TRIGGER isn't true # after WAIT_TRIGGER_DELAY * MAX_TRIGGER seconds LOG=/dev/null # same format than AWAKE_TIME but as an array of times in increase order #AWAKE_TIMES=("01:23:45" "02:46:00" "13:57:00" "20:00:00") #WAIT_TRIGGER_DELAYS=(60 60 60 60) #WAIT_TRIGGERS=("true" "true" "true" "true") #MAX_TRIGGERS=(60 60 60 60) #ON_ERRORS=() unset RTCWAKE_TIME AWAKE_TIME test -r $0.conf && source $0.conf # all previous variables can be set here echo -e "\n$(date) $1/$2" >> $LOG case "$1/$2" in pre/suspend) NOW=$(date +%_H%M%S) NOW_s=$(date +%s) if test -z "$AWAKE_TIME"; then for ((i=0; i<${#AWAKE_TIMES[@]}; i++)); do if test $NOW -lt ${AWAKE_TIMES[i]//:}; then RTCWAKE_TIME=$(date +%s -d ${AWAKE_TIMES[i]}) suspend_duration=$(( $RTCWAKE_TIME - $NOW_s )) if test $suspend_duration -gt ${MIN_TIME_BEFORE_NEXT_AWAKE:-120}; then break fi elif test $i -eq $((${#AWAKE_TIMES[@]} - 1)); then RTCWAKE_TIME=$(date +%s -d "tomorrow ${AWAKE_TIMES[0]}") fi done else RTCWAKE_TIME=$([ $NOW -lt ${AWAKE_TIME//:} ] && date +%s -d $AWAKE_TIME || date +%s -d "tomorrow $AWAKE_TIME") fi suspend_duration=$(( $RTCWAKE_TIME - $NOW_s )) date_awake=$(date -d "$suspend_duration seconds") echo "$NOW: will be awaken in $suspend_duration seconds ($date_awake) (mode=$1)" >> $LOG rtcwake -m no -s $suspend_duration ;; post/suspend) at -M now <<< "$0 post wait-for-trigger" > /dev/null 2>&1 ;; post/wait-for-trigger) NOW=$(date +%_H%M%S) if test -z "$AWAKE_TIME"; then for ((i=0; i<${#AWAKE_TIMES[@]}; i++)); do if test $NOW -ge ${AWAKE_TIMES[i]//:} -a $NOW -le $(date +%_H%M%S -d "${AWAKE_TIMES[i]} $GAP"); then AWAKE_TIME=${AWAKE_TIMES[i]} WAIT_TRIGGER_DELAY=${WAIT_TRIGGER_DELAYS[i]:-60} MAX_TRIGGER=${MAX_TRIGGERS[i]:-60} WAIT_TRIGGER=${WAIT_TRIGGERS[i]:-true} test -n "${ON_ERRORS[i]}" && ON_ERROR=${ON_ERRORS[i]} test -n "${AWAKE_ACTIONS[i]}" && AWAKE_ACTION=${AWAKE_ACTIONS[i]} break fi done fi test -n "$AWAKE_ACTION" && at -M now <<< $AWAKE_ACTION >> $LOG 2>&1 if test -n "$AWAKE_TIME" && test $NOW -ge ${AWAKE_TIME//:} -a $NOW -le $(date +%_H%M%S -d "$AWAKE_TIME $GAP"); then cat >> $LOG << EOL NOW: $NOW AWAKE_TIME: $AWAKE_TIME WAIT_TRIGGER_DELAY: $WAIT_TRIGGER_DELAY WAIT_TRIGGER: $WAIT_TRIGGER MAX_TRIGGER: $MAX_TRIGGER EOL COUNT=1 while sleep $WAIT_TRIGGER_DELAY && eval "$WAIT_TRIGGER"; do let COUNT++ if test $COUNT -gt $MAX_TRIGGER; then if test -n "$ON_ERROR"; then eval "$ON_ERROR" fi break elif test -n "$STAY_AWAKE" && test -e $STAY_AWAKE; then rm -f $STAY_AWAKE exit fi done systemctl suspend else echo "sortie de veille manuelle" >> $LOG pkill -fx "/bin/bash $0 post wait-for-trigger" rtcwake -m disable fi ;; esac