Home | History | Annotate | Line # | Download | only in time
tzselect.ksh revision 1.22
      1   1.8  christos #! /bin/bash
      2   1.2     perry #
      3  1.18  christos # Ask the user about the time zone, and output the resulting TZ value to stdout.
      4  1.18  christos # Interact with the user via stderr and stdin.
      5  1.18  christos #
      6  1.22  christos #	$NetBSD: tzselect.ksh,v 1.22 2023/09/16 18:40:26 christos Exp $
      7   1.2     perry #
      8   1.8  christos PKGVERSION='(tzcode) '
      9   1.8  christos TZVERSION=see_Makefile
     10   1.8  christos REPORT_BUGS_TO=tz@iana.org
     11   1.5    kleink 
     12  1.15  christos # Contributed by Paul Eggert.  This file is in the public domain.
     13   1.1       jtc 
     14   1.1       jtc # Porting notes:
     15   1.1       jtc #
     16  1.10  christos # This script requires a Posix-like shell and prefers the extension of a
     17   1.8  christos # 'select' statement.  The 'select' statement was introduced in the
     18   1.8  christos # Korn shell and is available in Bash and other shell implementations.
     19   1.8  christos # If your host lacks both Bash and the Korn shell, you can get their
     20   1.8  christos # source from one of these locations:
     21   1.1       jtc #
     22  1.16  christos #	Bash <https://www.gnu.org/software/bash/>
     23   1.8  christos #	Korn Shell <http://www.kornshell.com/>
     24  1.19  christos #	MirBSD Korn Shell <http://www.mirbsd.org/mksh.htm>
     25   1.1       jtc #
     26  1.19  christos # For portability to Solaris 10 /bin/sh (supported by Oracle through
     27  1.19  christos # January 2024) this script avoids some POSIX features and common
     28  1.19  christos # extensions, such as $(...) (which works sometimes but not others),
     29  1.19  christos # $((...)), ! CMD, ${#ID}, ${ID##PAT}, ${ID%%PAT}, and $10.
     30  1.19  christos 
     31  1.10  christos #
     32   1.1       jtc # This script also uses several features of modern awk programs.
     33   1.8  christos # If your host lacks awk, or has an old awk that does not conform to Posix,
     34   1.1       jtc # you can use either of the following free programs instead:
     35   1.1       jtc #
     36  1.16  christos #	Gawk (GNU awk) <https://www.gnu.org/software/gawk/>
     37  1.16  christos #	mawk <https://invisible-island.net/mawk/>
     38  1.21  christos #	nawk <https://github.com/onetrueawk/awk>
     39   1.1       jtc 
     40   1.1       jtc 
     41   1.1       jtc # Specify default values for environment variables if they are unset.
     42   1.1       jtc : ${AWK=awk}
     43  1.10  christos : ${TZDIR=`pwd`}
     44   1.1       jtc 
     45  1.22  christos # Output one argument as-is to standard output, with trailing newline.
     46  1.14  christos # Safer than 'echo', which can mishandle '\' or leading '-'.
     47  1.14  christos say() {
     48  1.14  christos     printf '%s\n' "$1"
     49  1.14  christos }
     50  1.14  christos 
     51   1.1       jtc # Check for awk Posix compliance.
     52   1.1       jtc ($AWK -v x=y 'BEGIN { exit 123 }') </dev/null >/dev/null 2>&1
     53   1.1       jtc [ $? = 123 ] || {
     54  1.14  christos 	say >&2 "$0: Sorry, your '$AWK' program is not Posix compatible."
     55   1.1       jtc 	exit 1
     56   1.1       jtc }
     57   1.1       jtc 
     58   1.9  christos coord=
     59   1.9  christos location_limit=10
     60  1.11  christos zonetabtype=zone1970
     61   1.9  christos 
     62   1.9  christos usage="Usage: tzselect [--version] [--help] [-c COORD] [-n LIMIT]
     63  1.18  christos Select a timezone interactively.
     64   1.6   mlelstv 
     65   1.9  christos Options:
     66   1.9  christos 
     67   1.9  christos   -c COORD
     68   1.9  christos     Instead of asking for continent and then country and then city,
     69   1.9  christos     ask for selection from time zones whose largest cities
     70   1.9  christos     are closest to the location with geographical coordinates COORD.
     71   1.9  christos     COORD should use ISO 6709 notation, for example, '-c +4852+00220'
     72   1.9  christos     for Paris (in degrees and minutes, North and East), or
     73   1.9  christos     '-c -35-058' for Buenos Aires (in degrees, South and West).
     74   1.9  christos 
     75   1.9  christos   -n LIMIT
     76   1.9  christos     Display at most LIMIT locations when -c is used (default $location_limit).
     77   1.9  christos 
     78   1.9  christos   --version
     79   1.9  christos     Output version information.
     80   1.9  christos 
     81   1.9  christos   --help
     82   1.9  christos     Output this help.
     83   1.9  christos 
     84   1.9  christos Report bugs to $REPORT_BUGS_TO."
     85   1.9  christos 
     86  1.10  christos # Ask the user to select from the function's arguments,
     87  1.10  christos # and assign the selected argument to the variable 'select_result'.
     88  1.22  christos # Exit on EOF or I/O error.  Use the shell's nicer 'select' builtin if
     89  1.22  christos # available, falling back on a portable substitute otherwise.
     90  1.10  christos if
     91  1.10  christos   case $BASH_VERSION in
     92  1.10  christos   ?*) : ;;
     93  1.10  christos   '')
     94  1.10  christos     # '; exit' should be redundant, but Dash doesn't properly fail without it.
     95  1.11  christos     (eval 'set --; select x; do break; done; exit') </dev/null 2>/dev/null
     96  1.10  christos   esac
     97  1.10  christos then
     98  1.10  christos   # Do this inside 'eval', as otherwise the shell might exit when parsing it
     99  1.10  christos   # even though it is never executed.
    100  1.10  christos   eval '
    101  1.10  christos     doselect() {
    102  1.10  christos       select select_result
    103  1.10  christos       do
    104  1.10  christos 	case $select_result in
    105  1.10  christos 	"") echo >&2 "Please enter a number in range." ;;
    106  1.10  christos 	?*) break
    107  1.10  christos 	esac
    108  1.10  christos       done || exit
    109  1.10  christos     }
    110  1.10  christos   '
    111  1.10  christos else
    112  1.10  christos   doselect() {
    113  1.10  christos     # Field width of the prompt numbers.
    114  1.10  christos     select_width=`expr $# : '.*'`
    115  1.10  christos 
    116  1.10  christos     select_i=
    117  1.10  christos 
    118  1.10  christos     while :
    119  1.10  christos     do
    120  1.10  christos       case $select_i in
    121  1.10  christos       '')
    122  1.10  christos 	select_i=0
    123  1.10  christos 	for select_word
    124  1.10  christos 	do
    125  1.10  christos 	  select_i=`expr $select_i + 1`
    126  1.10  christos 	  printf >&2 "%${select_width}d) %s\\n" $select_i "$select_word"
    127  1.10  christos 	done ;;
    128  1.10  christos       *[!0-9]*)
    129  1.10  christos 	echo >&2 'Please enter a number in range.' ;;
    130  1.10  christos       *)
    131  1.10  christos 	if test 1 -le $select_i && test $select_i -le $#; then
    132  1.10  christos 	  shift `expr $select_i - 1`
    133  1.10  christos 	  select_result=$1
    134  1.10  christos 	  break
    135  1.10  christos 	fi
    136  1.10  christos 	echo >&2 'Please enter a number in range.'
    137  1.10  christos       esac
    138  1.10  christos 
    139  1.10  christos       # Prompt and read input.
    140  1.10  christos       printf >&2 %s "${PS3-#? }"
    141  1.10  christos       read select_i || exit
    142  1.10  christos     done
    143  1.10  christos   }
    144  1.10  christos fi
    145  1.10  christos 
    146  1.12  christos while getopts c:n:t:-: opt
    147   1.9  christos do
    148   1.9  christos     case $opt$OPTARG in
    149   1.9  christos     c*)
    150   1.9  christos 	coord=$OPTARG ;;
    151   1.9  christos     n*)
    152   1.9  christos 	location_limit=$OPTARG ;;
    153  1.11  christos     t*) # Undocumented option, used for developer testing.
    154  1.11  christos 	zonetabtype=$OPTARG ;;
    155   1.9  christos     -help)
    156   1.9  christos 	exec echo "$usage" ;;
    157   1.9  christos     -version)
    158   1.9  christos 	exec echo "tzselect $PKGVERSION$TZVERSION" ;;
    159   1.9  christos     -*)
    160  1.14  christos 	say >&2 "$0: -$opt$OPTARG: unknown option; try '$0 --help'"; exit 1 ;;
    161   1.9  christos     *)
    162  1.14  christos 	say >&2 "$0: try '$0 --help'"; exit 1 ;;
    163   1.9  christos     esac
    164   1.9  christos done
    165   1.9  christos 
    166  1.10  christos shift `expr $OPTIND - 1`
    167   1.9  christos case $# in
    168   1.9  christos 0) ;;
    169  1.14  christos *) say >&2 "$0: $1: unknown argument"; exit 1 ;;
    170   1.9  christos esac
    171   1.6   mlelstv 
    172   1.1       jtc # Make sure the tables are readable.
    173   1.1       jtc TZ_COUNTRY_TABLE=$TZDIR/iso3166.tab
    174  1.11  christos TZ_ZONE_TABLE=$TZDIR/$zonetabtype.tab
    175   1.1       jtc for f in $TZ_COUNTRY_TABLE $TZ_ZONE_TABLE
    176   1.1       jtc do
    177  1.11  christos 	<"$f" || {
    178  1.14  christos 		say >&2 "$0: time zone files are not set up correctly"
    179   1.1       jtc 		exit 1
    180   1.1       jtc 	}
    181   1.1       jtc done
    182   1.1       jtc 
    183  1.14  christos # If the current locale does not support UTF-8, convert data to current
    184  1.14  christos # locale's format if possible, as the shell aligns columns better that way.
    185  1.14  christos # Check the UTF-8 of U+12345 CUNEIFORM SIGN URU TIMES KI.
    186  1.19  christos $AWK 'BEGIN { u12345 = "\360\222\215\205"; exit length(u12345) != 1 }' || {
    187  1.14  christos     { tmp=`(mktemp -d) 2>/dev/null` || {
    188  1.14  christos 	tmp=${TMPDIR-/tmp}/tzselect.$$ &&
    189  1.14  christos 	(umask 77 && mkdir -- "$tmp")
    190  1.14  christos     };} &&
    191  1.14  christos     trap 'status=$?; rm -fr -- "$tmp"; exit $status' 0 HUP INT PIPE TERM &&
    192  1.14  christos     (iconv -f UTF-8 -t //TRANSLIT <"$TZ_COUNTRY_TABLE" >$tmp/iso3166.tab) \
    193  1.14  christos         2>/dev/null &&
    194  1.14  christos     TZ_COUNTRY_TABLE=$tmp/iso3166.tab &&
    195  1.14  christos     iconv -f UTF-8 -t //TRANSLIT <"$TZ_ZONE_TABLE" >$tmp/$zonetabtype.tab &&
    196  1.14  christos     TZ_ZONE_TABLE=$tmp/$zonetabtype.tab
    197  1.19  christos }
    198  1.14  christos 
    199   1.1       jtc newline='
    200   1.1       jtc '
    201   1.1       jtc IFS=$newline
    202   1.1       jtc 
    203  1.22  christos # Awk script to output a country list.
    204  1.22  christos output_country_list='
    205  1.22  christos   BEGIN { FS = "\t" }
    206  1.22  christos   /^#$/ { next }
    207  1.22  christos   /^#[^@]/ { next }
    208  1.22  christos   {
    209  1.22  christos     commentary = $0 ~ /^#@/
    210  1.22  christos     if (commentary) {
    211  1.22  christos       col1ccs = substr($1, 3)
    212  1.22  christos       conts = $2
    213  1.22  christos     } else {
    214  1.22  christos       col1ccs = $1
    215  1.22  christos       conts = $3
    216  1.22  christos     }
    217  1.22  christos     ncc = split(col1ccs, cc, /,/)
    218  1.22  christos     ncont = split(conts, cont, /,/)
    219  1.22  christos     for (i = 1; i <= ncc; i++) {
    220  1.22  christos       elsewhere = commentary
    221  1.22  christos       for (ci = 1; ci <= ncont; ci++) {
    222  1.22  christos 	if (cont[ci] ~ continent_re) {
    223  1.22  christos 	  if (!cc_seen[cc[i]]++) cc_list[++ccs] = cc[i]
    224  1.22  christos 	  elsewhere = 0
    225  1.22  christos 	}
    226  1.22  christos       }
    227  1.22  christos       if (elsewhere) {
    228  1.22  christos 	for (i = 1; i <= ncc; i++) {
    229  1.22  christos 	  cc_elsewhere[cc[i]] = 1
    230  1.22  christos 	}
    231  1.22  christos       }
    232  1.22  christos     }
    233  1.22  christos   }
    234  1.22  christos   END {
    235  1.22  christos 	  while (getline <TZ_COUNTRY_TABLE) {
    236  1.22  christos 		  if ($0 !~ /^#/) cc_name[$1] = $2
    237  1.22  christos 	  }
    238  1.22  christos 	  for (i = 1; i <= ccs; i++) {
    239  1.22  christos 		  country = cc_list[i]
    240  1.22  christos 		  if (cc_elsewhere[country]) continue
    241  1.22  christos 		  if (cc_name[country]) {
    242  1.22  christos 		    country = cc_name[country]
    243  1.22  christos 		  }
    244  1.22  christos 		  print country
    245  1.22  christos 	  }
    246  1.22  christos   }
    247  1.22  christos '
    248   1.1       jtc 
    249   1.9  christos # Awk script to read a time zone table and output the same table,
    250  1.22  christos # with each row preceded by its distance from 'here'.
    251  1.22  christos # If output_times is set, each row is instead preceded by its local time
    252  1.22  christos # and any apostrophes are escaped for the shell.
    253  1.22  christos output_distances_or_times='
    254   1.9  christos   BEGIN {
    255   1.9  christos     FS = "\t"
    256  1.22  christos     if (!output_times) {
    257  1.22  christos       while (getline <TZ_COUNTRY_TABLE)
    258  1.22  christos 	if ($0 ~ /^[^#]/)
    259  1.22  christos 	  country[$1] = $2
    260  1.22  christos       country["US"] = "US" # Otherwise the strings get too long.
    261  1.22  christos     }
    262   1.9  christos   }
    263  1.12  christos   function abs(x) {
    264  1.12  christos     return x < 0 ? -x : x;
    265  1.12  christos   }
    266  1.12  christos   function min(x, y) {
    267  1.12  christos     return x < y ? x : y;
    268  1.12  christos   }
    269  1.12  christos   function convert_coord(coord, deg, minute, ilen, sign, sec) {
    270   1.9  christos     if (coord ~ /^[-+]?[0-9]?[0-9][0-9][0-9][0-9][0-9][0-9]([^0-9]|$)/) {
    271   1.9  christos       degminsec = coord
    272   1.9  christos       intdeg = degminsec < 0 ? -int(-degminsec / 10000) : int(degminsec / 10000)
    273   1.9  christos       minsec = degminsec - intdeg * 10000
    274   1.9  christos       intmin = minsec < 0 ? -int(-minsec / 100) : int(minsec / 100)
    275   1.9  christos       sec = minsec - intmin * 100
    276   1.9  christos       deg = (intdeg * 3600 + intmin * 60 + sec) / 3600
    277   1.9  christos     } else if (coord ~ /^[-+]?[0-9]?[0-9][0-9][0-9][0-9]([^0-9]|$)/) {
    278   1.9  christos       degmin = coord
    279   1.9  christos       intdeg = degmin < 0 ? -int(-degmin / 100) : int(degmin / 100)
    280  1.12  christos       minute = degmin - intdeg * 100
    281  1.12  christos       deg = (intdeg * 60 + minute) / 60
    282   1.9  christos     } else
    283   1.9  christos       deg = coord
    284   1.9  christos     return deg * 0.017453292519943296
    285   1.9  christos   }
    286   1.9  christos   function convert_latitude(coord) {
    287   1.9  christos     match(coord, /..*[-+]/)
    288   1.9  christos     return convert_coord(substr(coord, 1, RLENGTH - 1))
    289   1.9  christos   }
    290   1.9  christos   function convert_longitude(coord) {
    291   1.9  christos     match(coord, /..*[-+]/)
    292   1.9  christos     return convert_coord(substr(coord, RLENGTH))
    293   1.9  christos   }
    294   1.9  christos   # Great-circle distance between points with given latitude and longitude.
    295   1.9  christos   # Inputs and output are in radians.  This uses the great-circle special
    296   1.9  christos   # case of the Vicenty formula for distances on ellipsoids.
    297  1.12  christos   function gcdist(lat1, long1, lat2, long2, dlong, x, y, num, denom) {
    298   1.9  christos     dlong = long2 - long1
    299  1.13  christos     x = cos(lat2) * sin(dlong)
    300  1.13  christos     y = cos(lat1) * sin(lat2) - sin(lat1) * cos(lat2) * cos(dlong)
    301  1.13  christos     num = sqrt(x * x + y * y)
    302  1.13  christos     denom = sin(lat1) * sin(lat2) + cos(lat1) * cos(lat2) * cos(dlong)
    303   1.9  christos     return atan2(num, denom)
    304   1.9  christos   }
    305  1.12  christos   # Parallel distance between points with given latitude and longitude.
    306  1.12  christos   # This is the product of the longitude difference and the cosine
    307  1.12  christos   # of the latitude of the point that is further from the equator.
    308  1.12  christos   # I.e., it considers longitudes to be further apart if they are
    309  1.12  christos   # nearer the equator.
    310  1.12  christos   function pardist(lat1, long1, lat2, long2) {
    311  1.13  christos     return abs(long1 - long2) * min(cos(lat1), cos(lat2))
    312  1.12  christos   }
    313  1.12  christos   # The distance function is the sum of the great-circle distance and
    314  1.12  christos   # the parallel distance.  It could be weighted.
    315  1.12  christos   function dist(lat1, long1, lat2, long2) {
    316  1.13  christos     return gcdist(lat1, long1, lat2, long2) + pardist(lat1, long1, lat2, long2)
    317  1.12  christos   }
    318   1.9  christos   BEGIN {
    319   1.9  christos     coord_lat = convert_latitude(coord)
    320   1.9  christos     coord_long = convert_longitude(coord)
    321   1.9  christos   }
    322   1.9  christos   /^[^#]/ {
    323  1.22  christos     inline[inlines++] = $0
    324  1.22  christos     ncc = split($1, cc, /,/)
    325  1.22  christos     for (i = 1; i <= ncc; i++)
    326  1.22  christos       cc_used[cc[i]]++
    327  1.22  christos   }
    328  1.22  christos   END {
    329  1.22  christos    for (h = 0; h < inlines; h++) {
    330  1.22  christos     $0 = inline[h]
    331  1.11  christos     line = $1 "\t" $2 "\t" $3
    332  1.11  christos     sep = "\t"
    333  1.11  christos     ncc = split($1, cc, /,/)
    334  1.22  christos     split("", item_seen)
    335  1.22  christos     item_seen[""] = 1
    336  1.11  christos     for (i = 1; i <= ncc; i++) {
    337  1.22  christos       item = cc_used[cc[i]] <= 1 ? country[cc[i]] : $4
    338  1.22  christos       if (item_seen[item]++) continue
    339  1.22  christos       line = line sep item
    340  1.22  christos       sep = "; "
    341  1.22  christos     }
    342  1.22  christos     if (output_times) {
    343  1.22  christos       fmt = "TZ='\''%s'\'' date +'\''%d %%Y %%m %%d %%H:%%M %%a %%b\t%s'\''\n"
    344  1.22  christos       gsub(/'\''/, "&\\\\&&", line)
    345  1.22  christos       printf fmt, $3, h, line
    346  1.22  christos     } else {
    347  1.22  christos       here_lat = convert_latitude($2)
    348  1.22  christos       here_long = convert_longitude($2)
    349  1.22  christos       printf "%g\t%s\n", dist(coord_lat, coord_long, here_lat, here_long), line
    350  1.11  christos     }
    351  1.22  christos    }
    352   1.9  christos   }
    353   1.9  christos '
    354   1.1       jtc 
    355   1.1       jtc # Begin the main loop.  We come back here if the user wants to retry.
    356   1.1       jtc while
    357   1.1       jtc 
    358   1.1       jtc 	echo >&2 'Please identify a location' \
    359   1.1       jtc 		'so that time zone rules can be set correctly.'
    360   1.1       jtc 
    361   1.1       jtc 	continent=
    362   1.1       jtc 	country=
    363   1.1       jtc 	region=
    364   1.1       jtc 
    365   1.9  christos 	case $coord in
    366   1.9  christos 	?*)
    367   1.9  christos 		continent=coord;;
    368   1.9  christos 	'')
    369   1.1       jtc 
    370   1.1       jtc 	# Ask the user for continent or ocean.
    371   1.1       jtc 
    372  1.22  christos 	echo >&2 'Please select a continent, ocean, "coord", "TZ", or "time".'
    373   1.1       jtc 
    374  1.10  christos         quoted_continents=`
    375  1.10  christos 	  $AWK '
    376  1.20  christos 	    function handle_entry(entry) {
    377  1.20  christos 	      entry = substr(entry, 1, index(entry, "/") - 1)
    378  1.20  christos 	      if (entry == "America")
    379  1.20  christos 	       entry = entry "s"
    380  1.20  christos 	      if (entry ~ /^(Arctic|Atlantic|Indian|Pacific)$/)
    381  1.20  christos 	       entry = entry " Ocean"
    382  1.20  christos 	      printf "'\''%s'\''\n", entry
    383  1.20  christos 	    }
    384  1.10  christos 	    BEGIN { FS = "\t" }
    385   1.9  christos 	    /^[^#]/ {
    386  1.20  christos               handle_entry($3)
    387   1.9  christos             }
    388  1.20  christos 	    /^#@/ {
    389  1.20  christos 	      ncont = split($2, cont, /,/)
    390  1.20  christos 	      for (ci = 1; ci <= ncont; ci++) {
    391  1.20  christos 	        handle_entry(cont[ci])
    392  1.20  christos 	      }
    393  1.20  christos 	    }
    394  1.11  christos           ' <"$TZ_ZONE_TABLE" |
    395   1.9  christos 	  sort -u |
    396   1.9  christos 	  tr '\n' ' '
    397   1.9  christos 	  echo ''
    398  1.10  christos 	`
    399   1.9  christos 
    400   1.9  christos 	eval '
    401  1.10  christos 	    doselect '"$quoted_continents"' \
    402   1.9  christos 		"coord - I want to use geographical coordinates." \
    403  1.22  christos 		"TZ - I want to specify the timezone using the Posix TZ format." \
    404  1.22  christos 		"time - I know local time already."
    405  1.10  christos 	    continent=$select_result
    406  1.10  christos 	    case $continent in
    407  1.10  christos 	    Americas) continent=America;;
    408  1.10  christos 	    *" "*) continent=`expr "$continent" : '\''\([^ ]*\)'\''`
    409  1.10  christos 	    esac
    410   1.9  christos 	'
    411   1.9  christos 	esac
    412   1.9  christos 
    413   1.1       jtc 	case $continent in
    414   1.9  christos 	TZ)
    415   1.1       jtc 		# Ask the user for a Posix TZ string.  Check that it conforms.
    416   1.1       jtc 		while
    417   1.1       jtc 			echo >&2 'Please enter the desired value' \
    418   1.1       jtc 				'of the TZ environment variable.'
    419  1.18  christos 			echo >&2 'For example, AEST-10 is abbreviated' \
    420  1.18  christos 				'AEST and is 10 hours'
    421  1.17  christos 			echo >&2 'ahead (east) of Greenwich,' \
    422  1.17  christos 				'with no daylight saving time.'
    423   1.1       jtc 			read TZ
    424   1.1       jtc 			$AWK -v TZ="$TZ" 'BEGIN {
    425  1.15  christos 				tzname = "(<[[:alnum:]+-]{3,}>|[[:alpha:]]{3,})"
    426  1.15  christos 				time = "(2[0-4]|[0-1]?[0-9])" \
    427  1.15  christos 				  "(:[0-5][0-9](:[0-5][0-9])?)?"
    428   1.1       jtc 				offset = "[-+]?" time
    429  1.15  christos 				mdate = "M([1-9]|1[0-2])\\.[1-5]\\.[0-6]"
    430  1.15  christos 				jdate = "((J[1-9]|[0-9]|J?[1-9][0-9]" \
    431  1.15  christos 				  "|J?[1-2][0-9][0-9])|J?3[0-5][0-9]|J?36[0-5])"
    432  1.15  christos 				datetime = ",(" mdate "|" jdate ")(/" time ")?"
    433   1.1       jtc 				tzpattern = "^(:.*|" tzname offset "(" tzname \
    434   1.1       jtc 				  "(" offset ")?(" datetime datetime ")?)?)$"
    435   1.1       jtc 				if (TZ ~ tzpattern) exit 1
    436   1.1       jtc 				exit 0
    437   1.1       jtc 			}'
    438   1.1       jtc 		do
    439  1.18  christos 		    say >&2 "'$TZ' is not a conforming Posix timezone string."
    440   1.1       jtc 		done
    441   1.1       jtc 		TZ_for_date=$TZ;;
    442   1.1       jtc 	*)
    443   1.9  christos 		case $continent in
    444   1.9  christos 		coord)
    445   1.9  christos 		    case $coord in
    446   1.9  christos 		    '')
    447   1.9  christos 			echo >&2 'Please enter coordinates' \
    448   1.9  christos 				'in ISO 6709 notation.'
    449   1.9  christos 			echo >&2 'For example, +4042-07403 stands for'
    450   1.9  christos 			echo >&2 '40 degrees 42 minutes north,' \
    451   1.9  christos 				'74 degrees 3 minutes west.'
    452   1.9  christos 			read coord;;
    453   1.9  christos 		    esac
    454  1.10  christos 		    distance_table=`$AWK \
    455   1.9  christos 			    -v coord="$coord" \
    456   1.9  christos 			    -v TZ_COUNTRY_TABLE="$TZ_COUNTRY_TABLE" \
    457  1.22  christos 			    "$output_distances_or_times" <"$TZ_ZONE_TABLE" |
    458   1.9  christos 		      sort -n |
    459   1.9  christos 		      sed "${location_limit}q"
    460  1.10  christos 		    `
    461  1.22  christos 		    regions=`$AWK \
    462  1.22  christos 		      -v distance_table="$distance_table" '
    463  1.22  christos 		      BEGIN {
    464  1.22  christos 		        nlines = split(distance_table, line, /\n/)
    465  1.22  christos 			for (nr = 1; nr <= nlines; nr++) {
    466  1.22  christos 			  nf = split(line[nr], f, /\t/)
    467  1.22  christos 			  print f[nf]
    468  1.22  christos 			}
    469  1.22  christos 		      }
    470  1.10  christos 		    '`
    471  1.22  christos 		    echo >&2 'Please select one of the following timezones,'
    472   1.9  christos 		    echo >&2 'listed roughly in increasing order' \
    473   1.9  christos 			    "of distance from $coord".
    474  1.10  christos 		    doselect $regions
    475  1.10  christos 		    region=$select_result
    476  1.22  christos 		    TZ=`$AWK \
    477  1.22  christos 		      -v distance_table="$distance_table" \
    478  1.22  christos 		      -v region="$region" '
    479  1.22  christos 		      BEGIN {
    480  1.22  christos 		        nlines = split(distance_table, line, /\n/)
    481  1.22  christos 			for (nr = 1; nr <= nlines; nr++) {
    482  1.22  christos 			  nf = split(line[nr], f, /\t/)
    483  1.22  christos 			  if (f[nf] == region) {
    484  1.22  christos 			    print f[4]
    485  1.22  christos 			  }
    486  1.22  christos 			}
    487  1.22  christos 		      }
    488  1.10  christos 		    '`
    489   1.9  christos 		    ;;
    490   1.9  christos 		*)
    491  1.22  christos 		case $continent in
    492  1.22  christos 		time)
    493  1.22  christos 		  minute_format='%a %b %d %H:%M'
    494  1.22  christos 		  old_minute=`TZ=UTC0 date +"$minute_format"`
    495  1.22  christos 		  for i in 1 2 3
    496  1.22  christos 		  do
    497  1.22  christos 		    time_table_command=`
    498  1.22  christos 		      $AWK -v output_times=1 \
    499  1.22  christos 			  "$output_distances_or_times" <"$TZ_ZONE_TABLE"
    500  1.22  christos 		    `
    501  1.22  christos 		    time_table=`eval "$time_table_command"`
    502  1.22  christos 		    new_minute=`TZ=UTC0 date +"$minute_format"`
    503  1.22  christos 		    case $old_minute in
    504  1.22  christos 		    "$new_minute") break;;
    505  1.22  christos 		    esac
    506  1.22  christos 		    old_minute=$new_minute
    507  1.22  christos 		  done
    508  1.22  christos 		  echo >&2 "The system says Universal Time is $new_minute."
    509  1.22  christos 		  echo >&2 "Assuming that's correct, what is the local time?"
    510  1.22  christos 		  eval doselect `
    511  1.22  christos 		    say "$time_table" |
    512  1.22  christos 		    sort -k2n -k2,5 -k1n |
    513  1.22  christos 		    $AWK '{
    514  1.22  christos 		      line = $6 " " $7 " " $4 " " $5
    515  1.22  christos 		      if (line == oldline) next
    516  1.22  christos 		      oldline = line
    517  1.22  christos 		      gsub(/'\''/, "&\\\\&&", line)
    518  1.22  christos 		      printf "'\''%s'\''\n", line
    519  1.22  christos 		    }'
    520  1.22  christos 		  `
    521  1.22  christos 		  time=$select_result
    522  1.22  christos 		  zone_table=`
    523  1.22  christos 		    say "$time_table" |
    524  1.22  christos 		    $AWK -v time="$time" '{
    525  1.22  christos 		      if ($6 " " $7 " " $4 " " $5 == time) {
    526  1.22  christos 			sub(/[^\t]*\t/, "")
    527  1.22  christos 			print
    528  1.22  christos 		      }
    529  1.22  christos 		    }'
    530  1.22  christos 		  `
    531  1.22  christos 		  countries=`
    532  1.22  christos 		     say "$zone_table" |
    533  1.22  christos 		     $AWK \
    534  1.22  christos 			-v continent_re='' \
    535  1.22  christos 			-v TZ_COUNTRY_TABLE="$TZ_COUNTRY_TABLE" \
    536  1.22  christos 			"$output_country_list" |
    537  1.22  christos 		     sort -f
    538  1.22  christos 		  `
    539  1.22  christos 		  ;;
    540  1.22  christos 		*)
    541  1.22  christos 		  zone_table=file
    542  1.22  christos 		  # Get list of names of countries in the continent or ocean.
    543  1.22  christos 		  countries=`$AWK \
    544  1.20  christos 			-v continent_re="^$continent/" \
    545   1.1       jtc 			-v TZ_COUNTRY_TABLE="$TZ_COUNTRY_TABLE" \
    546  1.22  christos 			"$output_country_list" \
    547  1.22  christos 			<"$TZ_ZONE_TABLE" | sort -f
    548  1.22  christos 		  `;;
    549  1.22  christos 		esac
    550   1.1       jtc 
    551   1.1       jtc 		# If there's more than one country, ask the user which one.
    552   1.1       jtc 		case $countries in
    553   1.1       jtc 		*"$newline"*)
    554   1.9  christos 			echo >&2 'Please select a country' \
    555   1.9  christos 				'whose clocks agree with yours.'
    556  1.10  christos 			doselect $countries
    557  1.22  christos 			country_result=$select_result
    558  1.10  christos 			country=$select_result;;
    559   1.1       jtc 		*)
    560   1.1       jtc 			country=$countries
    561   1.1       jtc 		esac
    562   1.1       jtc 
    563   1.1       jtc 
    564  1.18  christos 		# Get list of timezones in the country.
    565  1.22  christos 		regions=`
    566  1.22  christos 		  case $zone_table in
    567  1.22  christos 		  file) cat -- "$TZ_ZONE_TABLE";;
    568  1.22  christos 		  *) say "$zone_table";;
    569  1.22  christos 		  esac |
    570  1.22  christos 		  $AWK \
    571   1.1       jtc 			-v country="$country" \
    572   1.1       jtc 			-v TZ_COUNTRY_TABLE="$TZ_COUNTRY_TABLE" \
    573  1.22  christos 		    '
    574   1.1       jtc 			BEGIN {
    575  1.10  christos 				FS = "\t"
    576   1.1       jtc 				cc = country
    577   1.1       jtc 				while (getline <TZ_COUNTRY_TABLE) {
    578   1.1       jtc 					if ($0 !~ /^#/  &&  country == $2) {
    579   1.1       jtc 						cc = $1
    580   1.1       jtc 						break
    581   1.1       jtc 					}
    582   1.1       jtc 				}
    583   1.1       jtc 			}
    584  1.14  christos 			/^#/ { next }
    585  1.11  christos 			$1 ~ cc { print $4 }
    586  1.22  christos 		    '
    587  1.22  christos 		`
    588   1.1       jtc 
    589   1.1       jtc 
    590   1.1       jtc 		# If there's more than one region, ask the user which one.
    591   1.1       jtc 		case $regions in
    592   1.1       jtc 		*"$newline"*)
    593  1.18  christos 			echo >&2 'Please select one of the following timezones.'
    594  1.10  christos 			doselect $regions
    595  1.22  christos 			region=$select_result
    596   1.1       jtc 		esac
    597   1.1       jtc 
    598   1.1       jtc 		# Determine TZ from country and region.
    599  1.22  christos 		TZ=`
    600  1.22  christos 		  case $zone_table in
    601  1.22  christos 		  file) cat -- "$TZ_ZONE_TABLE";;
    602  1.22  christos 		  *) say "$zone_table";;
    603  1.22  christos 		  esac |
    604  1.22  christos 		  $AWK \
    605   1.1       jtc 			-v country="$country" \
    606   1.1       jtc 			-v region="$region" \
    607   1.1       jtc 			-v TZ_COUNTRY_TABLE="$TZ_COUNTRY_TABLE" \
    608  1.22  christos 		    '
    609   1.1       jtc 			BEGIN {
    610  1.10  christos 				FS = "\t"
    611   1.1       jtc 				cc = country
    612   1.1       jtc 				while (getline <TZ_COUNTRY_TABLE) {
    613   1.1       jtc 					if ($0 !~ /^#/  &&  country == $2) {
    614   1.1       jtc 						cc = $1
    615   1.1       jtc 						break
    616   1.1       jtc 					}
    617   1.1       jtc 				}
    618   1.1       jtc 			}
    619  1.14  christos 			/^#/ { next }
    620  1.22  christos 			$1 ~ cc && ($4 == region || !region) { print $3 }
    621  1.22  christos 		    '
    622  1.22  christos 		`;;
    623   1.9  christos 		esac
    624   1.1       jtc 
    625   1.1       jtc 		# Make sure the corresponding zoneinfo file exists.
    626   1.1       jtc 		TZ_for_date=$TZDIR/$TZ
    627  1.11  christos 		<"$TZ_for_date" || {
    628  1.14  christos 			say >&2 "$0: time zone files are not set up correctly"
    629   1.1       jtc 			exit 1
    630   1.1       jtc 		}
    631   1.1       jtc 	esac
    632   1.1       jtc 
    633   1.1       jtc 
    634   1.1       jtc 	# Use the proposed TZ to output the current date relative to UTC.
    635   1.1       jtc 	# Loop until they agree in seconds.
    636   1.1       jtc 	# Give up after 8 unsuccessful tries.
    637   1.1       jtc 
    638   1.1       jtc 	extra_info=
    639   1.1       jtc 	for i in 1 2 3 4 5 6 7 8
    640   1.1       jtc 	do
    641  1.10  christos 		TZdate=`LANG=C TZ="$TZ_for_date" date`
    642  1.10  christos 		UTdate=`LANG=C TZ=UTC0 date`
    643  1.10  christos 		TZsec=`expr "$TZdate" : '.*:\([0-5][0-9]\)'`
    644  1.10  christos 		UTsec=`expr "$UTdate" : '.*:\([0-5][0-9]\)'`
    645   1.1       jtc 		case $TZsec in
    646   1.1       jtc 		$UTsec)
    647   1.1       jtc 			extra_info="
    648  1.15  christos Selected time is now:	$TZdate.
    649   1.1       jtc Universal Time is now:	$UTdate."
    650   1.1       jtc 			break
    651   1.1       jtc 		esac
    652   1.1       jtc 	done
    653   1.1       jtc 
    654   1.1       jtc 
    655   1.1       jtc 	# Output TZ info and ask the user to confirm.
    656   1.1       jtc 
    657   1.1       jtc 	echo >&2 ""
    658  1.22  christos 	echo >&2 "Based on the following information:"
    659   1.1       jtc 	echo >&2 ""
    660  1.22  christos 	case $time%$country_result%$region%$coord in
    661  1.22  christos 	?*%?*%?*%)
    662  1.22  christos 	  say >&2 "	$time$newline	$country_result$newline	$region";;
    663  1.22  christos 	?*%?*%%|?*%%?*%) say >&2 "	$time$newline	$country_result$region";;
    664  1.22  christos 	?*%%%) say >&2 "	$time";;
    665  1.22  christos 	%?*%?*%) say >&2 "	$country_result$newline	$region";;
    666  1.22  christos 	%?*%%)	say >&2 "	$country_result";;
    667  1.22  christos 	%%?*%?*) say >&2 "	coord $coord$newline	$region";;
    668  1.22  christos 	%%%?*)	say >&2 "	coord $coord";;
    669  1.14  christos 	*)	say >&2 "	TZ='$TZ'"
    670   1.1       jtc 	esac
    671  1.14  christos 	say >&2 ""
    672  1.22  christos 	say >&2 "TZ='$TZ' will be used.$extra_info"
    673  1.14  christos 	say >&2 "Is the above information OK?"
    674   1.1       jtc 
    675  1.10  christos 	doselect Yes No
    676  1.10  christos 	ok=$select_result
    677   1.1       jtc 	case $ok in
    678   1.1       jtc 	Yes) break
    679   1.1       jtc 	esac
    680   1.9  christos do coord=
    681   1.1       jtc done
    682   1.1       jtc 
    683   1.5    kleink case $SHELL in
    684   1.5    kleink *csh) file=.login line="setenv TZ '$TZ'";;
    685   1.5    kleink *) file=.profile line="TZ='$TZ'; export TZ"
    686   1.5    kleink esac
    687   1.5    kleink 
    688  1.15  christos test -t 1 && say >&2 "
    689   1.5    kleink You can make this change permanent for yourself by appending the line
    690   1.5    kleink 	$line
    691   1.5    kleink to the file '$file' in your home directory; then log out and log in again.
    692   1.5    kleink 
    693   1.5    kleink Here is that TZ value again, this time on standard output so that you
    694   1.5    kleink can use the $0 command in shell scripts:"
    695   1.5    kleink 
    696  1.14  christos say "$TZ"
    697