#!/usr/bin/env zsh # To use it, just place it in your $fpath # say # To use this, either source it from your script, or place it in your $fpath # and autoload it. Then, use the `say`-function. # RFC 5424 # https://www.rfc-editor.org/rfc/rfc5424 # # Numerical Severity # Code # # 0 Emergency: system is unusable # 1 Alert: action must be taken immediately # 2 Critical: critical conditions # 3 Error: error conditions # 4 Warning: warning conditions # 5 Notice: normal but significant condition # 6 Informational: informational messages # 7 Debug: debug-level messages # # # Variables # # Standardized ${0:t} handling, according to the Zsh Plugin Standard # https://zplugin.readthedocs.io/en/latest/zsh-plugin-standard/ 0="${${ZERO:-${0:#${ZSH_ARGZERO:-}}}:-${(%):-%N}}" 0="${${(M)0:#/*}:-$PWD/$0}" # Default severity for messages typeset -gi _SAY_DEFAULT_SEV=${_SAY_DEFAULT_SEV:-6} # Print severity typeset -gi _SAY_PRINT_SEV=${_SAY_PRINT_SEV:-0} # Print date typeset -gi _SAY_PRINT_DATE=${_SAY_PRINT_DATES:-0} # Date format. typeset -g _SAY_DATE_FMT=${_SAY_DATE_FMT:-'%Y-%m-%dT%T %Z'} # Default print flags typeset -g _SAY_PRINTFLAGS=(${=_SAY_PRINTFLAGS:-}) # 0: no color # 1: only color severity # 2: color severity and date # 3: color everything typeset -gi _SAY_COLOR=${_SAY_COLOR:-3} # Tags for log-messages : ${_SAY_SEV_MSG_0='[EMERGENCY]'} \ ${_SAY_SEV_MSG_1='[ALERT]'} \ ${_SAY_SEV_MSG_2='[CRITICAL]'} \ ${_SAY_SEV_MSG_3='[ERROR]'} \ ${_SAY_SEV_MSG_4='[WARNING]'} \ ${_SAY_SEV_MSG_5='[NOTICE]'} \ ${_SAY_SEV_MSG_6='[INFO]'} \ ${_SAY_SEV_MSG_7='[DEBUG]'} typeset -gm '_SAY_SEV_MSG_*' # I really want to stick to known colors that work in most terminals, but I # also really want grey debug-messages. So let's check if the terminal supports # 256 colors before we use them. # https://zsh.sourceforge.io/Doc/Release/Zsh-Modules.html#The-zsh_002ftermcap-Module zmodload -F zsh/termcap +p:termcap 2>/dev/null && [[ ${termcap[Co]:-0} == 256 ]] && : ${_SAY_COLORCODE_s7='%F{008}'} ${_SAY_COLORCODE_e7='%f'} # Set colorcodes : ${_SAY_COLORCODE_s0='%B%K{white}%F{red}'} ${_SAY_COLORCODE_e0='%f%k%b'} \ ${_SAY_COLORCODE_s1='%B%F{red}'} ${_SAY_COLORCODE_e1='%f%b'} \ ${_SAY_COLORCODE_s2='%B%F{red}'} ${_SAY_COLORCODE_e2='%f%b'} \ ${_SAY_COLORCODE_s3='%F{red}'} ${_SAY_COLORCODE_e3='%f'} \ ${_SAY_COLORCODE_s4='%F{yellow}'} ${_SAY_COLORCODE_e4='%f'} \ ${_SAY_COLORCODE_s5='%F{cyan}'} ${_SAY_COLORCODE_e5='%f'} \ ${_SAY_COLORCODE_s6=''} ${_SAY_COLORCODE_e6=''} \ ${_SAY_COLORCODE_s7=''} ${_SAY_COLORCODE_e7=''} typeset -gm '_SAY_COLORCODE_s*' '_SAY_COLORCODE_e*' # Helptext : ${_SAY_HELPSTRING='NAME say - a function you can use to log messages in a zsh-script. SYNOPSIS say [OPTIONS] MSG Alternative syntax: say [OPTIONS] SEVERITY "MSG" [EXITCODE] This allows you to use \`say debug this is a debug msg\`. But, if you set severity in an option as well, it will override the severity in the message-part. I.e. \`say -4 debug this is a debug msg\` will print as an ERROR, not a DEBUG. Exitcode set as option also overrides an exitcode at the end of the line. Also, if multiple severity-options are given, i.e. \`-2 -4\` or \`-54\`, the last one will take precedence. DESCTIPTION say has the following options: -0 Log with severity EMERGENCY -1 Log with severity ALERT -2 Log with severity CRITICAL -3 Log with severity ERROR -4 Log with severity WARNING -5 Log with severity NOTICE -6 Log with severity INFO -7 Log with severity DEBUG -C No color. This is the default when run from a non-interactive shell (actually, when $TTY is unset) -c NUM NUM can be 0, 1, 2, or 3. They each mean the following: 0: no color 1: only color severity-tag 2: color severity-tag and date 3: color everything (default) Colors are ignored if in a non-interactive shell (when $TTY is unset) -D Do not print date and time (default) -d Print date and time -e NUM Exit, with NUM as exitcode, after logging message. NUM must be in the range {0..255}. -f FLAGS Flags for the zsh print() function, which is used to print messages. Takes flags without the dash ("-"). Also, wordsplitting is done on the flags, so if you want to provide a flag with an argument, then drop the space between the flag and the argument. I.e. you want \`print -C 2 -u 2\` to print two columns to stderr, do \`say -f"C2 u2"\`. -h Print this help-text -i NUM File descriptor to read messages from. For example, use \`-i 0\` to read messages from STDIN. If an MSG is still provided, it will be treated as a prefix for all messages read from the file descriptor. -n No color, no date, no severity-tag. Same as \`-C -D -S\`. -r NUM return with NUM as return-code. NUM must be in the range {0..255}. \`-e\` (exit) takes precedence over \`-r\` (return). -S Do not print severity-tag (default) -s Print the severity-tag, for example "[DEBUG] message" -u NUM Print to this filedescriptor. 1 = stdout, 2 = stderr. By default, LOGLEVEL 0-4 prints to stderr, and 5-7 prints to stdout. -v Be a bit more verbose. Currently only works in conjunction with -@ -x Trace say(). For debugging say(). -@ Treat message an as array of messages, and print each item on a new line AUTHOR Dennis Eriksen '} typeset -gr _SAY_HELPSTRING zmodload -F zsh/datetime +b:strftime 2>/dev/null || true # Just use `date` if we can't load datetime # say [OPTIONS] MSG say() { emulate -LR zsh unsetopt unset # Make sure LOGLEVEL is an integer # (L) is an expansion-flag that converts letters to lower case # Do this inside the function because LOGLEVEL can easily be set and user by something else. integer loglevel case ${(L)LOGLEVEL:-6} in 0|emerg*) loglevel=0 ;; 1|alert) loglevel=1 ;; 2|crit*) loglevel=2 ;; 3|err*) loglevel=3 ;; 4|warn*) loglevel=4 ;; 5|notice) loglevel=5 ;; 6|info*) loglevel=6 ;; 7|debug) loglevel=7 ;; *) loglevel=6 ;; esac; local sev msg local -a pflags=( $_SAY_PRINTFLAGS ) # printflags integer exitbool exitcode returnbool returncode arrbool inputfd inputfdbool verbool outputfd # # This is where the magic happens # # handle options local OPTIND OPTARG opt while getopts ":01234567Cc:Dde:f:hi:nr:Ssu:vx@" opt; do case $opt in 0) sev=0 ;; 1) sev=1 ;; 2) sev=2 ;; 3) sev=3 ;; 4) sev=4 ;; 5) sev=5 ;; 6) sev=6 ;; 7) sev=7 ;; C) _SAY_COLOR=0 ;; c) _say_checkint $OPTARG 0 3 || return 100 _SAY_COLOR=$OPTARG ;; D) _SAY_PRINT_DATE=0 ;; d) _SAY_PRINT_DATE=1 ;; e) _say_checkint $OPTARG 0 255 || return 103 exitcode=$OPTARG exitbool=1 ;; f) [[ $OPTARG == *-* ]] && print -ru2 -- "\`${0:t} -f\` takes flags for the print() function, but without the '-'." && return 107 pflags+=( $=OPTARG ) ;; h) print -r -- $_SAY_HELPSTRING && return 0 ;; i) _say_checkint $OPTARG 0 || return 106 inputfdbool=1 inputfd=$OPTARG ;; n) _SAY_COLOR=0 _SAY_PRINT_DATE=0 _SAY_PRINT_SEV=0 ;; r) [[ ! $OPTARG == <0-255> ]] && print -ru2 -- "\`${0:t} -r\` takes exactly one argument - an int between 0 and 255. You provided: '$OPTARG'" && return 105 returncode=$OPTARG returnbool=1 ;; S) _SAY_PRINT_SEV=0 ;; s) _SAY_PRINT_SEV=1 ;; u) [[ ! $OPTARG == <0-> ]] && print -ru2 -- "\`${0:t} -u\` takes one argument - a positive integer. \`${0:t} -u n\` will print to filedescriptor n." && return 104 pflags+=( u$OPTARG ) ;; v) verbool=1 ;; x) set -x ;; @) arrbool=1 ;; :) [[ $OPTARG == [i] ]] && inputfdbool=1 && continue print -ru2 -- "${0:t}: option '$OPTARG' is missing an argument" print -ru2 -- "Try \`${0:t} -h\` for more info" && return 101 ;; ?) print -ru2 -- "${0:t}: invalid option -- '$OPTARG'" print -ru2 -- "Try \`${0:t} -h\` for more info" && return 102 ;; esac done; (( OPTIND > 1 )) && shift $(( OPTIND - 1 )) # If not set by option, try to determine severity from $1, but only if there # is more than one argument remaining, and there are no spaces in $1. # This enables us to use `say error "MESSAGE"`, but won't catch `say "error message"` if [[ -z $sev && $ARGC -gt 1 && $1 != *' '* ]] || [[ -z $sev && $ARGC -eq 1 && $inputfdbool -eq 1 ]]; then case ${(L)1} in 0|emerg*) sev=0 ;; 1|alert*) sev=1 ;; 2|crit*) sev=2 ;; 3|err*) sev=3 ;; 4|warn*) sev=4 ;; 5|notice*) sev=5 ;; 6|info*) sev=6 ;; 7|debug*) sev=7 ;; esac; # Do the same thing with exitcode, but only if severity is 3 (error) or # lower. It does not make sense to exit on 4 (warn) or higher. if (( ! exitbool && ARGC == 3 && ${sev:-$_SAY_DEFAULT_SEV} <= 3 )) && [[ $3 == <0-255> ]]; then exitcode=$3 exitbool=1 shift -p # Remove the last positional argument ($3), as it was just used fi [[ $sev == <0-7> ]] && shift 1 # Shift arguments if we managed to set sev fi; # Set sev if we failed to set it aleady. Also, force integer. integer sev=${sev:-$_SAY_DEFAULT_SEV} # If severity is below loglevel if (( sev > loglevel )); then # If inputbool is not set, we can just return here (( inputfdbool )) || return 0 exec {outputfd}>/dev/null fi [[ -n $TTY ]] || _SAY_COLOR=0 # No color unless we have a TTY to print to (( _SAY_COLOR )) && pflags+=( P ) # We don't need to use -P unless we have colors. # Set filedescriptor to print to, if -u is not already set in pflags if (( outputfd )); then pflags+=( u$outputfd ) elif (( ${pflags[(i)*u*]} > $#pflags )); then # if $sev is 4 (warn) or below, print to stderr, else print to stdout pflags+=( u$(( 4 >= sev ? 2 : 1 )) ) fi # Set message and reset args msg=$@ # If we're reading from a file descriptor if (( inputfdbool )); then while read line; do _say_print $msg $line; done <&$inputfd # Treat message as array? elif (( arrbool && ARGC )); then (( verbool )) && _say_print " ARGC=$ARGC" for i in {1..$ARGC}; do msg=$argv[$i] (( verbool )) && msg=" \$$i=$msg" _say_print "$msg" done # If not, just print the message else _say_print $msg fi # Close outputfd if set (( outputfd )) && exec {outputfd}>&- # Exit say() (( ! exitbool )) || exit $exitcode (( ! returnbool )) || return $returncode return 0 } # _say_print MSG _say_print() { emulate -LR zsh unsetopt unset # If no arguments, just add newline and return (( ARGC )) || { print -- && return } local msg msgsev msgdate string cs ce msg=$@ # resolve msg severity and colorcodes msgsev=_SAY_SEV_MSG_$sev cs=_SAY_COLORCODE_s$sev # cs = color start ce=_SAY_COLORCODE_e$sev # ce = color end msgsev=${(P)msgsev} cs=${(P)cs} ce=${(P)ce} # Begin logic if (( _SAY_PRINT_DATE )); then (( $+builtins[strftime] )) && msgdate="$(strftime "$_SAY_DATE_FMT")" || msgdate="$(date "+$_SAY_DATE_FMT")" (( _SAY_COLOR >= 2 )) && msgdate="${cs}${msgdate}${ce}" string="$msgdate " fi if (( _SAY_PRINT_SEV )); then (( _SAY_COLOR >= 1 )) && msgsev="${cs}${msgsev}${ce}" string+="$msgsev " fi (( _SAY_COLOR == 3 )) && msg="${cs}${msg}${ce}" string+="$msg" print ${pflags/#/-} -- "$string" } # _say_checkint NUM [MIN] [MAX] _say_checkint() { emulate -LR zsh unsetopt unset local num=$1 min=${2:-} max=${3:-} if [[ $num != <-> ]] || [[ -n $min && $num -lt $min ]] || [[ -n $max && $num -gt $max ]]; then print -u2 -- "-$opt takes an int in the range <$min-$max> (inclusive). You provided: \'$num\'" return 1 fi } # saytest saytest() { emulate -LR zsh unsetopt unset local LOGLEVEL=7 msgsev for i in {0..7}; do msgsev=_SAY_SEV_MSG_$i msgsev=${(P)msgsev:1:-1} print -n "${msgsev}:" repeat $(( $#_SAY_SEV_MSG_0 - 1 - $#msgsev )); print -n " " say -$i -- "say -$i -- message" done } say debug "function \`${0:t}\` loaded"