Skip to content

Commit

Permalink
feat: add new script uxt ⌚️
Browse files Browse the repository at this point in the history
  • Loading branch information
oldratlee committed Feb 1, 2024
1 parent e7ccb63 commit 8b951dc
Show file tree
Hide file tree
Showing 4 changed files with 354 additions and 0 deletions.
4 changes: 4 additions & 0 deletions bin/--version
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
#!/bin/bash
set -eEuo pipefail

echo 'hacked'
234 changes: 234 additions & 0 deletions bin/uxt
Original file line number Diff line number Diff line change
@@ -0,0 +1,234 @@
#!/usr/bin/env bash
# @Function
# Convert unix time to human readable date string.
# Note: The range of the 10-digit unix time in second include recent date:
# 9999999999: 2286-11-20 17:46:39 +0000
# 1000000000: 2001-09-09 01:46:40 +0000
# 0: 1970-01-01 00:00:00 +0000
# -1000000000: 1938-04-24 22:13:20 +0000
# -9999999999: 1653-02-10 06:13:21 +0000
#
# @Usage
# # default treat first 10 digits as second(include recent date)
# $ uxt 1234567890 # unix time of second
# 2009-02-14 07:31:30 +0800
# $ uxt 1234567890333 # unix time of milliseconds(10 + 3 digits)
# 2009-02-14 07:31:30.333 +0800
# $ uxt 12345678903 # unix time of 10 + 1 digits
# 2009-02-14 07:31:30.3 +0800
# # support multiply arguments
# $ uxt 0 1234567890 12345678903
#
# @online-doc https://github.com/oldratlee/useful-scripts/blob/dev-2.x/docs/shell.md#-unix-time
# @author Jerry Lee (oldratlee at gmail dot com)
set -eEuo pipefail

readonly PROG=${0##*/}
readonly PROG_VERSION='2.x-dev'

################################################################################
# util functions
################################################################################

red_print() {
# if stdout is a terminal, turn on color output.
# '-t' check: is a terminal?
# check isatty in bash https://stackoverflow.com/questions/10022323
if [ -t 1 ]; then
printf "\e[1;31m%s\e[0m\n" "$*"
else
printf '%s\n' "$*"
fi
}

is_integer() {
[[ "$1" =~ ^-?[[:digit:]]+$ ]]
}

die() {
red_print "Error: $*" >&2
exit 1
}

usage() {
local -r exit_code=${1:-0}
(($# > 0)) && shift
local -r out=$(((exit_code != 0) + 1))

# NOTE: $'foo' is the escape sequence syntax of bash
(($# > 0)) && red_print "$*"$'\n' >&"$out"

cat >&"$out" <<EOF
Usage: $PROG [OPTION] unix-time [unix-time...]
Convert unix time to human readable date string.
Note: The range of the 10-digit unix time in second include recent date:
9999999999: 2286-11-20 17:46:39 +0000
1000000000: 2001-09-09 01:46:40 +0000
0: 1970-01-01 00:00:00 +0000
-1000000000: 1938-04-24 22:13:20 +0000
-9999999999: 1653-02-10 06:13:21 +0000
Example:
# default treat first 10 digits as second(include recent date)
$ $PROG 1234567890 # unix time of second
2009-02-14 07:31:30 +0800
$ $PROG 1234567890333 # unix time of milliseconds(10 + 3 digits)
2009-02-14 07:31:30.333 +0800
$ $PROG 12345678903 # unix time of 10 + 1 digits
2009-02-14 07:31:30.3 +0800
# support multiply arguments
$ $PROG 0 1234567890 12345678903
Options:
-u, --time-unit set the time unit of given epochs
-Z, --no-time-zone do not print time zone
-D, --no-second-decimal
do not print second decimal
-t, --trim-decimal-tailing-0
trim the tailing zeros of second decimal
-h, --help display this help and exit
-V, --version display version information and exit
EOF

exit "$exit_code"
}

progVersion() {
printf '%s\n' "$PROG $PROG_VERSION"
exit
}

################################################################################
# parse options
################################################################################

declare -a args=()
unit=
no_tz=false
no_second_decimal=false
trim_decimal_tailing_0=false
while (($# > 0)); do
case "$1" in
-u | --unit)
unit=$2
shift 2
;;
-Z | --no-time-zone)
no_tz=true
shift
;;
-D | --no-second-decimal)
no_second_decimal=true
shift
;;
-t | --trim-decimal-tailing-0)
trim_decimal_tailing_0=true
shift
;;
-h | --help)
usage
;;
-V | --version)
progVersion
;;
-[[:digit:]]*)
# negative number start with '-', is not option
args=(${args[@]:+"${args[@]}"} "$1")
shift
;;
--)
shift
args=(${args[@]:+"${args[@]}"} "$@")
break
;;
-*)
usage 2 "Error: unrecognized option '$1'"
;;
*)
args=(${args[@]:+"${args[@]}"} "$1")
shift
;;
esac
done

[[ -n $unit ]] && if [[ $unit =~ ^(s|second)$ ]]; then
unit=s
elif [[ $unit =~ ^(ms|millisecond)$ ]]; then
unit=ms
else
usage 1 "illegal time unit '$unit'! support values: 'second'/'s', 'millisecond'/'ms'"
fi

readonly args unit trim_decimal_tailing_0 no_tz

if ((${#args[@]} == 0)); then
usage 1 'No argument provided!'
fi

for a in "${args[@]}"; do
is_integer "$a" || die "argument $a is not integer!"
[[ ! $a =~ ^-?0+[1-9] ]] || die "argument $a contains beginning 0!"
done

################################################################################
# biz logic
################################################################################

print_date() {
local -r input=$1
# split input integer to sign and number part
local -r sign_part=${input%%[!-]*} # remove digits from tail
local -r number_part=${input#-} # remove sign from head
local -r np_len=${#number_part} # length of number part

local second_part=0 decimal_part=
# case 1: is unix time in second?
if [[ $unit = s ]]; then
second_part=$number_part
# case 2: is unix time in millisecond?
elif [[ $unit = ms ]]; then
if ((np_len > 3)); then
second_part=${number_part:0:np_len-3}
decimal_part=${number_part:np_len-3:3}
else
printf -v decimal_part '%03d' "$number_part"
fi
# case 3: auto detect by length
else
# <= 10 digits, treat as second
if ((np_len <= 10)); then
second_part=$number_part
# for long integer(> 10 digits), treat first 10 digits as second,
# and the rest as decimal/nano second(almost 9 digits)
elif ((np_len <= 19)); then
second_part=${number_part:0:10}
decimal_part=${number_part:10:9}
else
die "argument $input contains $np_len digits(>19), too many to treat as a recent date(first 10-digits as seconds, rest at most 9 digits as decimal)"
fi
fi

# trim tailing zeros of decimal?
$trim_decimal_tailing_0 && while true; do
local old_len=${#decimal_part}
decimal_part=${decimal_part%0}
((${#decimal_part} < old_len)) || break
done

local -r seconds_value=$sign_part$second_part second_part decimal_part
# defensive check. 9999999999999999(16 '9') seconds is so big, 300M years later(316,889,355-01-25 17:46:39 +0000)
((${#second_part} <= 16)) ||
die "argument $input(seconds: $seconds_value${decimal_part:+, decimal: .$decimal_part}) is too big, seconds are more than 16 digits."

local date_input=$seconds_value${decimal_part:+.$decimal_part}
local format_n=
$no_second_decimal || format_n=${decimal_part:+.%${#decimal_part}N}
local format_tz=
$no_tz || format_tz=' %z'
date -d "@$date_input" +"%Y-%m-%d %H:%M:%S$format_n$format_tz"
}

for a in "${args[@]}"; do
print_date "$a"
done
57 changes: 57 additions & 0 deletions docs/shell.md
Original file line number Diff line number Diff line change
Expand Up @@ -364,6 +364,63 @@ $ rp /home /etc/../etc /home/admin
../../etc
```
🍺 [uxt](../bin/uxt)
----------------------
输出`Unix`时间戳对应的时间,自动识别秒/毫秒格式。
### 用法/示例
```bash
$ uxt 1234567890 # 秒时间戳(10位以内数字)
2009-02-14 07:31:30 +0800
$ uxt 1234567890333 # 毫秒时间戳(10位秒 + 3位毫秒)
2009-02-14 07:31:30.333 +0800
$ uxt 12345678903 # 11位(10位秒 + 剩余1位作为毫秒)
2009-02-14 07:31:30.3 +0800
# 支持多个参数
$ uxt 0 1234567890 12345678903

# 如果需要转换秒超过10位的时间戳,显式指定单位
$ uxt -u s 12345678900
2361-03-22 03:15:00 +0800
$ uxt -u ms 12345678900123
2361-03-22 03:15:00.123 +0800

$ uxt -h
Usage: uxt [OPTION] unix-time [unix-time...]

Convert unix time to human readable date string.
Note: The range of the 10-digit unix time in second include recent date:
9999999999: 2286-11-20 17:46:39 +0000
1000000000: 2001-09-09 01:46:40 +0000
0: 1970-01-01 00:00:00 +0000
-1000000000: 1938-04-24 22:13:20 +0000
-9999999999: 1653-02-10 06:13:21 +0000

Example:
# default treat first 10 digits as second(include recent date)
$ uxt 1234567890 # unix time of second
2009-02-14 07:31:30 +0800
$ uxt 1234567890333 # unix time of milliseconds(10 + 3 digits)
2009-02-14 07:31:30.333 +0800
$ uxt 12345678903 # unix time of 10 + 1 digits
2009-02-14 07:31:30.3 +0800
# support multiply arguments
$ uxt 0 1234567890 12345678903

Options:
-u, --time-unit set the time unit of given epochs
-Z, --no-time-zone do not print time zone
-D, --no-second-decimal
do not print second decimal
-t, --trim-decimal-tailing-0
trim the tailing zeros of second decimal
-h, --help display this help and exit
-V, --version display version information and exit
```
🍺 [cp-into-docker-run](../bin/cp-into-docker-run)
----------------------
Expand Down
59 changes: 59 additions & 0 deletions test-cases/uxt_test.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
#!/usr/bin/env bash
set -eEuo pipefail

READLINK_CMD=readlink
if command -v greadlink &>/dev/null; then
READLINK_CMD=greadlink
fi

BASE=$(dirname -- "$($READLINK_CMD -f -- "${BASH_SOURCE[0]}")")
cd "$BASE"

#################################################
# commons and test data
#################################################

readonly uxt="../bin/uxt"

#################################################
# test cases
#################################################

test_uxt_auto_detect() {
assertEquals "2024-01-30 00:00:00 +0000" "$(TZ=0 "$uxt" 1706572800)"
assertEquals "1900-01-30 00:00:00 +0000" "$(TZ=0 "$uxt" -- -2206483200)"

assertEquals "1970-01-01 00:00:00 +0000" "$(TZ=0 "$uxt" 0)"
assertEquals "1970-01-01 00:01:40 +0000" "$(TZ=0 "$uxt" -- 100)"
assertEquals "1969-12-31 23:58:20 +0000" "$(TZ=0 "$uxt" -- -100)"

# shellcheck disable=SC2016
assertFalse 'should fail, 20 more than 19 digits' '"$uxt" 12345678901234567890'
}

test_uxt_unit_second() {
assertEquals "2024-01-30 00:00:00 +0000" "$(TZ=0 "$uxt" -u s 1706572800)"
assertEquals "1900-01-30 00:00:00 +0000" "$(TZ=0 "$uxt" -u s -- -2206483200)"

assertEquals "1970-01-01 00:00:00 +0000" "$(TZ=0 "$uxt" -u s 0)"
assertEquals "1970-01-01 00:01:40 +0000" "$(TZ=0 "$uxt" -u s -- 100)"
assertEquals "1969-12-31 23:58:20 +0000" "$(TZ=0 "$uxt" -u s -- -100)"

# shellcheck disable=SC2016
assertFalse 'should fail, 20 more than 19 digits' '"$uxt" -u s 12345678901234567890'
}

test_uxt_unit_ms() {
assertEquals "2024-01-30 00:00:00.000 +0000" "$(TZ=0 "$uxt" -u ms 1706572800000)"
assertEquals "1900-01-30 00:00:00.000 +0000" "$(TZ=0 "$uxt" -u ms -- -2206483200000)"

assertEquals "1970-01-01 00:00:00.000 +0000" "$(TZ=0 "$uxt" -u ms 0)"
assertEquals "1970-01-01 00:01:40.000 +0000" "$(TZ=0 "$uxt" -u ms -- 100000)"
assertEquals "1969-12-31 23:58:20.000 +0000" "$(TZ=0 "$uxt" -u ms -- -100000)"
}

#################################################
# Load and run shUnit2.
#################################################

source "$BASE/shunit2-lib/shunit2"

0 comments on commit 8b951dc

Please sign in to comment.