Home | History | Annotate | Line # | Download | only in time
tzselect.ksh revision 1.12
      1   1.8  christos #! /bin/bash
      2   1.2     perry #
      3  1.12  christos #	$NetBSD: tzselect.ksh,v 1.12 2014/10/07 21:51:03 christos Exp $
      4   1.2     perry #
      5   1.8  christos PKGVERSION='(tzcode) '
      6   1.8  christos TZVERSION=see_Makefile
      7   1.8  christos REPORT_BUGS_TO=tz@iana.org
      8   1.5    kleink 
      9   1.1       jtc # Ask the user about the time zone, and output the resulting TZ value to stdout.
     10   1.1       jtc # Interact with the user via stderr and stdin.
     11   1.1       jtc 
     12   1.6   mlelstv # Contributed by Paul Eggert.
     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.8  christos #	Bash <http://www.gnu.org/software/bash/bash.html>
     23   1.8  christos #	Korn Shell <http://www.kornshell.com/>
     24   1.8  christos #	Public Domain Korn Shell <http://www.cs.mun.ca/~michael/pdksh/>
     25   1.1       jtc #
     26  1.10  christos # For portability to Solaris 9 /bin/sh this script avoids some POSIX
     27  1.10  christos # features and common extensions, such as $(...) (which works sometimes
     28  1.10  christos # but not others), $((...)), and $10.
     29  1.10  christos #
     30   1.1       jtc # This script also uses several features of modern awk programs.
     31   1.8  christos # If your host lacks awk, or has an old awk that does not conform to Posix,
     32   1.1       jtc # you can use either of the following free programs instead:
     33   1.1       jtc #
     34   1.8  christos #	Gawk (GNU awk) <http://www.gnu.org/software/gawk/>
     35   1.8  christos #	mawk <http://invisible-island.net/mawk/>
     36   1.1       jtc 
     37   1.1       jtc 
     38   1.1       jtc # Specify default values for environment variables if they are unset.
     39   1.1       jtc : ${AWK=awk}
     40  1.10  christos : ${TZDIR=`pwd`}
     41   1.1       jtc 
     42   1.1       jtc # Check for awk Posix compliance.
     43   1.1       jtc ($AWK -v x=y 'BEGIN { exit 123 }') </dev/null >/dev/null 2>&1
     44   1.1       jtc [ $? = 123 ] || {
     45  1.11  christos 	echo >&2 "$0: Sorry, your '$AWK' program is not Posix compatible."
     46   1.1       jtc 	exit 1
     47   1.1       jtc }
     48   1.1       jtc 
     49   1.9  christos coord=
     50   1.9  christos location_limit=10
     51  1.11  christos zonetabtype=zone1970
     52   1.9  christos 
     53   1.9  christos usage="Usage: tzselect [--version] [--help] [-c COORD] [-n LIMIT]
     54   1.6   mlelstv Select a time zone interactively.
     55   1.6   mlelstv 
     56   1.9  christos Options:
     57   1.9  christos 
     58   1.9  christos   -c COORD
     59   1.9  christos     Instead of asking for continent and then country and then city,
     60   1.9  christos     ask for selection from time zones whose largest cities
     61   1.9  christos     are closest to the location with geographical coordinates COORD.
     62   1.9  christos     COORD should use ISO 6709 notation, for example, '-c +4852+00220'
     63   1.9  christos     for Paris (in degrees and minutes, North and East), or
     64   1.9  christos     '-c -35-058' for Buenos Aires (in degrees, South and West).
     65   1.9  christos 
     66   1.9  christos   -n LIMIT
     67   1.9  christos     Display at most LIMIT locations when -c is used (default $location_limit).
     68   1.9  christos 
     69   1.9  christos   --version
     70   1.9  christos     Output version information.
     71   1.9  christos 
     72   1.9  christos   --help
     73   1.9  christos     Output this help.
     74   1.9  christos 
     75   1.9  christos Report bugs to $REPORT_BUGS_TO."
     76   1.9  christos 
     77  1.10  christos # Ask the user to select from the function's arguments,
     78  1.10  christos # and assign the selected argument to the variable 'select_result'.
     79  1.10  christos # Exit on EOF or I/O error.  Use the shell's 'select' builtin if available,
     80  1.10  christos # falling back on a less-nice but portable substitute otherwise.
     81  1.10  christos if
     82  1.10  christos   case $BASH_VERSION in
     83  1.10  christos   ?*) : ;;
     84  1.10  christos   '')
     85  1.10  christos     # '; exit' should be redundant, but Dash doesn't properly fail without it.
     86  1.11  christos     (eval 'set --; select x; do break; done; exit') </dev/null 2>/dev/null
     87  1.10  christos   esac
     88  1.10  christos then
     89  1.10  christos   # Do this inside 'eval', as otherwise the shell might exit when parsing it
     90  1.10  christos   # even though it is never executed.
     91  1.10  christos   eval '
     92  1.10  christos     doselect() {
     93  1.10  christos       select select_result
     94  1.10  christos       do
     95  1.10  christos 	case $select_result in
     96  1.10  christos 	"") echo >&2 "Please enter a number in range." ;;
     97  1.10  christos 	?*) break
     98  1.10  christos 	esac
     99  1.10  christos       done || exit
    100  1.10  christos     }
    101  1.10  christos 
    102  1.10  christos     # Work around a bug in bash 1.14.7 and earlier, where $PS3 is sent to stdout.
    103  1.10  christos     case $BASH_VERSION in
    104  1.10  christos     [01].*)
    105  1.10  christos       case `echo 1 | (select x in x; do break; done) 2>/dev/null` in
    106  1.10  christos       ?*) PS3=
    107  1.10  christos       esac
    108  1.10  christos     esac
    109  1.10  christos   '
    110  1.10  christos else
    111  1.10  christos   doselect() {
    112  1.10  christos     # Field width of the prompt numbers.
    113  1.10  christos     select_width=`expr $# : '.*'`
    114  1.10  christos 
    115  1.10  christos     select_i=
    116  1.10  christos 
    117  1.10  christos     while :
    118  1.10  christos     do
    119  1.10  christos       case $select_i in
    120  1.10  christos       '')
    121  1.10  christos 	select_i=0
    122  1.10  christos 	for select_word
    123  1.10  christos 	do
    124  1.10  christos 	  select_i=`expr $select_i + 1`
    125  1.10  christos 	  printf >&2 "%${select_width}d) %s\\n" $select_i "$select_word"
    126  1.10  christos 	done ;;
    127  1.10  christos       *[!0-9]*)
    128  1.10  christos 	echo >&2 'Please enter a number in range.' ;;
    129  1.10  christos       *)
    130  1.10  christos 	if test 1 -le $select_i && test $select_i -le $#; then
    131  1.10  christos 	  shift `expr $select_i - 1`
    132  1.10  christos 	  select_result=$1
    133  1.10  christos 	  break
    134  1.10  christos 	fi
    135  1.10  christos 	echo >&2 'Please enter a number in range.'
    136  1.10  christos       esac
    137  1.10  christos 
    138  1.10  christos       # Prompt and read input.
    139  1.10  christos       printf >&2 %s "${PS3-#? }"
    140  1.10  christos       read select_i || exit
    141  1.10  christos     done
    142  1.10  christos   }
    143  1.10  christos fi
    144  1.10  christos 
    145  1.12  christos while getopts c:n:t:-: opt
    146   1.9  christos do
    147   1.9  christos     case $opt$OPTARG in
    148   1.9  christos     c*)
    149   1.9  christos 	coord=$OPTARG ;;
    150   1.9  christos     n*)
    151   1.9  christos 	location_limit=$OPTARG ;;
    152  1.11  christos     t*) # Undocumented option, used for developer testing.
    153  1.11  christos 	zonetabtype=$OPTARG ;;
    154   1.9  christos     -help)
    155   1.9  christos 	exec echo "$usage" ;;
    156   1.9  christos     -version)
    157   1.9  christos 	exec echo "tzselect $PKGVERSION$TZVERSION" ;;
    158   1.9  christos     -*)
    159   1.9  christos 	echo >&2 "$0: -$opt$OPTARG: unknown option; try '$0 --help'"; exit 1 ;;
    160   1.9  christos     *)
    161   1.9  christos 	echo >&2 "$0: try '$0 --help'"; exit 1 ;;
    162   1.9  christos     esac
    163   1.9  christos done
    164   1.9  christos 
    165  1.10  christos shift `expr $OPTIND - 1`
    166   1.9  christos case $# in
    167   1.9  christos 0) ;;
    168   1.9  christos *) echo >&2 "$0: $1: unknown argument"; exit 1 ;;
    169   1.9  christos esac
    170   1.6   mlelstv 
    171   1.1       jtc # Make sure the tables are readable.
    172   1.1       jtc TZ_COUNTRY_TABLE=$TZDIR/iso3166.tab
    173  1.11  christos TZ_ZONE_TABLE=$TZDIR/$zonetabtype.tab
    174   1.1       jtc for f in $TZ_COUNTRY_TABLE $TZ_ZONE_TABLE
    175   1.1       jtc do
    176  1.11  christos 	<"$f" || {
    177   1.1       jtc 		echo >&2 "$0: time zone files are not set up correctly"
    178   1.1       jtc 		exit 1
    179   1.1       jtc 	}
    180   1.1       jtc done
    181   1.1       jtc 
    182   1.1       jtc newline='
    183   1.1       jtc '
    184   1.1       jtc IFS=$newline
    185   1.1       jtc 
    186   1.1       jtc 
    187   1.9  christos # Awk script to read a time zone table and output the same table,
    188   1.9  christos # with each column preceded by its distance from 'here'.
    189   1.9  christos output_distances='
    190   1.9  christos   BEGIN {
    191   1.9  christos     FS = "\t"
    192   1.9  christos     while (getline <TZ_COUNTRY_TABLE)
    193   1.9  christos       if ($0 ~ /^[^#]/)
    194   1.9  christos         country[$1] = $2
    195   1.9  christos     country["US"] = "US" # Otherwise the strings get too long.
    196   1.9  christos   }
    197  1.12  christos   function abs(x) {
    198  1.12  christos     return x < 0 ? -x : x;
    199  1.12  christos   }
    200  1.12  christos   function min(x, y) {
    201  1.12  christos     return x < y ? x : y;
    202  1.12  christos   }
    203  1.12  christos   function convert_coord(coord, deg, minute, ilen, sign, sec) {
    204   1.9  christos     if (coord ~ /^[-+]?[0-9]?[0-9][0-9][0-9][0-9][0-9][0-9]([^0-9]|$)/) {
    205   1.9  christos       degminsec = coord
    206   1.9  christos       intdeg = degminsec < 0 ? -int(-degminsec / 10000) : int(degminsec / 10000)
    207   1.9  christos       minsec = degminsec - intdeg * 10000
    208   1.9  christos       intmin = minsec < 0 ? -int(-minsec / 100) : int(minsec / 100)
    209   1.9  christos       sec = minsec - intmin * 100
    210   1.9  christos       deg = (intdeg * 3600 + intmin * 60 + sec) / 3600
    211   1.9  christos     } else if (coord ~ /^[-+]?[0-9]?[0-9][0-9][0-9][0-9]([^0-9]|$)/) {
    212   1.9  christos       degmin = coord
    213   1.9  christos       intdeg = degmin < 0 ? -int(-degmin / 100) : int(degmin / 100)
    214  1.12  christos       minute = degmin - intdeg * 100
    215  1.12  christos       deg = (intdeg * 60 + minute) / 60
    216   1.9  christos     } else
    217   1.9  christos       deg = coord
    218   1.9  christos     return deg * 0.017453292519943296
    219   1.9  christos   }
    220   1.9  christos   function convert_latitude(coord) {
    221   1.9  christos     match(coord, /..*[-+]/)
    222   1.9  christos     return convert_coord(substr(coord, 1, RLENGTH - 1))
    223   1.9  christos   }
    224   1.9  christos   function convert_longitude(coord) {
    225   1.9  christos     match(coord, /..*[-+]/)
    226   1.9  christos     return convert_coord(substr(coord, RLENGTH))
    227   1.9  christos   }
    228   1.9  christos   # Great-circle distance between points with given latitude and longitude.
    229   1.9  christos   # Inputs and output are in radians.  This uses the great-circle special
    230   1.9  christos   # case of the Vicenty formula for distances on ellipsoids.
    231  1.12  christos   function gcdist(lat1, long1, lat2, long2, dlong, x, y, num, denom) {
    232   1.9  christos     dlong = long2 - long1
    233   1.9  christos     x = cos (lat2) * sin (dlong)
    234   1.9  christos     y = cos (lat1) * sin (lat2) - sin (lat1) * cos (lat2) * cos (dlong)
    235   1.9  christos     num = sqrt (x * x + y * y)
    236   1.9  christos     denom = sin (lat1) * sin (lat2) + cos (lat1) * cos (lat2) * cos (dlong)
    237   1.9  christos     return atan2(num, denom)
    238   1.9  christos   }
    239  1.12  christos   # Parallel distance between points with given latitude and longitude.
    240  1.12  christos   # This is the product of the longitude difference and the cosine
    241  1.12  christos   # of the latitude of the point that is further from the equator.
    242  1.12  christos   # I.e., it considers longitudes to be further apart if they are
    243  1.12  christos   # nearer the equator.
    244  1.12  christos   function pardist(lat1, long1, lat2, long2) {
    245  1.12  christos     return abs (long1 - long2) * min (cos (lat1), cos (lat2))
    246  1.12  christos   }
    247  1.12  christos   # The distance function is the sum of the great-circle distance and
    248  1.12  christos   # the parallel distance.  It could be weighted.
    249  1.12  christos   function dist(lat1, long1, lat2, long2) {
    250  1.12  christos     return gcdist (lat1, long1, lat2, long2) + pardist (lat1, long1, lat2, long2)
    251  1.12  christos   }
    252   1.9  christos   BEGIN {
    253   1.9  christos     coord_lat = convert_latitude(coord)
    254   1.9  christos     coord_long = convert_longitude(coord)
    255   1.9  christos   }
    256   1.9  christos   /^[^#]/ {
    257   1.9  christos     here_lat = convert_latitude($2)
    258   1.9  christos     here_long = convert_longitude($2)
    259  1.11  christos     line = $1 "\t" $2 "\t" $3
    260  1.11  christos     sep = "\t"
    261  1.11  christos     ncc = split($1, cc, /,/)
    262  1.11  christos     for (i = 1; i <= ncc; i++) {
    263  1.11  christos       line = line sep country[cc[i]]
    264  1.11  christos       sep = ", "
    265  1.11  christos     }
    266   1.9  christos     if (NF == 4)
    267   1.9  christos       line = line " - " $4
    268   1.9  christos     printf "%g\t%s\n", dist(coord_lat, coord_long, here_lat, here_long), line
    269   1.9  christos   }
    270   1.9  christos '
    271   1.1       jtc 
    272   1.1       jtc # Begin the main loop.  We come back here if the user wants to retry.
    273   1.1       jtc while
    274   1.1       jtc 
    275   1.1       jtc 	echo >&2 'Please identify a location' \
    276   1.1       jtc 		'so that time zone rules can be set correctly.'
    277   1.1       jtc 
    278   1.1       jtc 	continent=
    279   1.1       jtc 	country=
    280   1.1       jtc 	region=
    281   1.1       jtc 
    282   1.9  christos 	case $coord in
    283   1.9  christos 	?*)
    284   1.9  christos 		continent=coord;;
    285   1.9  christos 	'')
    286   1.1       jtc 
    287   1.1       jtc 	# Ask the user for continent or ocean.
    288   1.1       jtc 
    289   1.9  christos 	echo >&2 'Please select a continent, ocean, "coord", or "TZ".'
    290   1.1       jtc 
    291  1.10  christos         quoted_continents=`
    292  1.10  christos 	  $AWK '
    293  1.10  christos 	    BEGIN { FS = "\t" }
    294   1.9  christos 	    /^[^#]/ {
    295   1.9  christos               entry = substr($3, 1, index($3, "/") - 1)
    296   1.9  christos               if (entry == "America")
    297   1.9  christos 		entry = entry "s"
    298   1.9  christos               if (entry ~ /^(Arctic|Atlantic|Indian|Pacific)$/)
    299   1.9  christos 		entry = entry " Ocean"
    300   1.9  christos               printf "'\''%s'\''\n", entry
    301   1.9  christos             }
    302  1.11  christos           ' <"$TZ_ZONE_TABLE" |
    303   1.9  christos 	  sort -u |
    304   1.9  christos 	  tr '\n' ' '
    305   1.9  christos 	  echo ''
    306  1.10  christos 	`
    307   1.9  christos 
    308   1.9  christos 	eval '
    309  1.10  christos 	    doselect '"$quoted_continents"' \
    310   1.9  christos 		"coord - I want to use geographical coordinates." \
    311   1.9  christos 		"TZ - I want to specify the time zone using the Posix TZ format."
    312  1.10  christos 	    continent=$select_result
    313  1.10  christos 	    case $continent in
    314  1.10  christos 	    Americas) continent=America;;
    315  1.10  christos 	    *" "*) continent=`expr "$continent" : '\''\([^ ]*\)'\''`
    316  1.10  christos 	    esac
    317   1.9  christos 	'
    318   1.9  christos 	esac
    319   1.9  christos 
    320   1.1       jtc 	case $continent in
    321   1.9  christos 	TZ)
    322   1.1       jtc 		# Ask the user for a Posix TZ string.  Check that it conforms.
    323   1.1       jtc 		while
    324   1.1       jtc 			echo >&2 'Please enter the desired value' \
    325   1.1       jtc 				'of the TZ environment variable.'
    326   1.1       jtc 			echo >&2 'For example, GST-10 is a zone named GST' \
    327   1.1       jtc 				'that is 10 hours ahead (east) of UTC.'
    328   1.1       jtc 			read TZ
    329   1.1       jtc 			$AWK -v TZ="$TZ" 'BEGIN {
    330   1.1       jtc 				tzname = "[^-+,0-9][^-+,0-9][^-+,0-9]+"
    331   1.1       jtc 				time = "[0-2]?[0-9](:[0-5][0-9](:[0-5][0-9])?)?"
    332   1.1       jtc 				offset = "[-+]?" time
    333   1.1       jtc 				date = "(J?[0-9]+|M[0-9]+\.[0-9]+\.[0-9]+)"
    334   1.1       jtc 				datetime = "," date "(/" time ")?"
    335   1.1       jtc 				tzpattern = "^(:.*|" tzname offset "(" tzname \
    336   1.1       jtc 				  "(" offset ")?(" datetime datetime ")?)?)$"
    337   1.1       jtc 				if (TZ ~ tzpattern) exit 1
    338   1.1       jtc 				exit 0
    339   1.1       jtc 			}'
    340   1.1       jtc 		do
    341  1.11  christos 			echo >&2 "'$TZ' is not a conforming" \
    342   1.1       jtc 				'Posix time zone string.'
    343   1.1       jtc 		done
    344   1.1       jtc 		TZ_for_date=$TZ;;
    345   1.1       jtc 	*)
    346   1.9  christos 		case $continent in
    347   1.9  christos 		coord)
    348   1.9  christos 		    case $coord in
    349   1.9  christos 		    '')
    350   1.9  christos 			echo >&2 'Please enter coordinates' \
    351   1.9  christos 				'in ISO 6709 notation.'
    352   1.9  christos 			echo >&2 'For example, +4042-07403 stands for'
    353   1.9  christos 			echo >&2 '40 degrees 42 minutes north,' \
    354   1.9  christos 				'74 degrees 3 minutes west.'
    355   1.9  christos 			read coord;;
    356   1.9  christos 		    esac
    357  1.10  christos 		    distance_table=`$AWK \
    358   1.9  christos 			    -v coord="$coord" \
    359   1.9  christos 			    -v TZ_COUNTRY_TABLE="$TZ_COUNTRY_TABLE" \
    360  1.11  christos 			    "$output_distances" <"$TZ_ZONE_TABLE" |
    361   1.9  christos 		      sort -n |
    362   1.9  christos 		      sed "${location_limit}q"
    363  1.10  christos 		    `
    364  1.10  christos 		    regions=`echo "$distance_table" | $AWK '
    365   1.9  christos 		      BEGIN { FS = "\t" }
    366   1.9  christos 		      { print $NF }
    367  1.10  christos 		    '`
    368   1.9  christos 		    echo >&2 'Please select one of the following' \
    369   1.9  christos 			    'time zone regions,'
    370   1.9  christos 		    echo >&2 'listed roughly in increasing order' \
    371   1.9  christos 			    "of distance from $coord".
    372  1.10  christos 		    doselect $regions
    373  1.10  christos 		    region=$select_result
    374  1.10  christos 		    TZ=`echo "$distance_table" | $AWK -v region="$region" '
    375   1.9  christos 		      BEGIN { FS="\t" }
    376   1.9  christos 		      $NF == region { print $4 }
    377  1.10  christos 		    '`
    378   1.9  christos 		    ;;
    379   1.9  christos 		*)
    380   1.1       jtc 		# Get list of names of countries in the continent or ocean.
    381  1.10  christos 		countries=`$AWK \
    382   1.1       jtc 			-v continent="$continent" \
    383   1.1       jtc 			-v TZ_COUNTRY_TABLE="$TZ_COUNTRY_TABLE" \
    384   1.1       jtc 		'
    385  1.10  christos 			BEGIN { FS = "\t" }
    386   1.1       jtc 			/^#/ { next }
    387   1.1       jtc 			$3 ~ ("^" continent "/") {
    388  1.11  christos 			    ncc = split($1, cc, /,/)
    389  1.11  christos 			    for (i = 1; i <= ncc; i++)
    390  1.11  christos 				if (!cc_seen[cc[i]]++) cc_list[++ccs] = cc[i]
    391   1.1       jtc 			}
    392   1.1       jtc 			END {
    393   1.1       jtc 				while (getline <TZ_COUNTRY_TABLE) {
    394   1.1       jtc 					if ($0 !~ /^#/) cc_name[$1] = $2
    395   1.1       jtc 				}
    396   1.1       jtc 				for (i = 1; i <= ccs; i++) {
    397   1.1       jtc 					country = cc_list[i]
    398   1.1       jtc 					if (cc_name[country]) {
    399   1.1       jtc 					  country = cc_name[country]
    400   1.1       jtc 					}
    401   1.1       jtc 					print country
    402   1.1       jtc 				}
    403   1.1       jtc 			}
    404  1.11  christos 		' <"$TZ_ZONE_TABLE" | sort -f`
    405   1.1       jtc 
    406   1.1       jtc 
    407   1.1       jtc 		# If there's more than one country, ask the user which one.
    408   1.1       jtc 		case $countries in
    409   1.1       jtc 		*"$newline"*)
    410   1.9  christos 			echo >&2 'Please select a country' \
    411   1.9  christos 				'whose clocks agree with yours.'
    412  1.10  christos 			doselect $countries
    413  1.10  christos 			country=$select_result;;
    414   1.1       jtc 		*)
    415   1.1       jtc 			country=$countries
    416   1.1       jtc 		esac
    417   1.1       jtc 
    418   1.1       jtc 
    419   1.1       jtc 		# Get list of names of time zone rule regions in the country.
    420  1.10  christos 		regions=`$AWK \
    421   1.1       jtc 			-v country="$country" \
    422   1.1       jtc 			-v TZ_COUNTRY_TABLE="$TZ_COUNTRY_TABLE" \
    423   1.1       jtc 		'
    424   1.1       jtc 			BEGIN {
    425  1.10  christos 				FS = "\t"
    426   1.1       jtc 				cc = country
    427   1.1       jtc 				while (getline <TZ_COUNTRY_TABLE) {
    428   1.1       jtc 					if ($0 !~ /^#/  &&  country == $2) {
    429   1.1       jtc 						cc = $1
    430   1.1       jtc 						break
    431   1.1       jtc 					}
    432   1.1       jtc 				}
    433   1.1       jtc 			}
    434  1.11  christos 			$1 ~ cc { print $4 }
    435  1.11  christos 		' <"$TZ_ZONE_TABLE"`
    436   1.1       jtc 
    437   1.1       jtc 
    438   1.1       jtc 		# If there's more than one region, ask the user which one.
    439   1.1       jtc 		case $regions in
    440   1.1       jtc 		*"$newline"*)
    441   1.1       jtc 			echo >&2 'Please select one of the following' \
    442   1.1       jtc 				'time zone regions.'
    443  1.10  christos 			doselect $regions
    444  1.10  christos 			region=$select_result;;
    445   1.1       jtc 		*)
    446   1.1       jtc 			region=$regions
    447   1.1       jtc 		esac
    448   1.1       jtc 
    449   1.1       jtc 		# Determine TZ from country and region.
    450  1.10  christos 		TZ=`$AWK \
    451   1.1       jtc 			-v country="$country" \
    452   1.1       jtc 			-v region="$region" \
    453   1.1       jtc 			-v TZ_COUNTRY_TABLE="$TZ_COUNTRY_TABLE" \
    454   1.1       jtc 		'
    455   1.1       jtc 			BEGIN {
    456  1.10  christos 				FS = "\t"
    457   1.1       jtc 				cc = country
    458   1.1       jtc 				while (getline <TZ_COUNTRY_TABLE) {
    459   1.1       jtc 					if ($0 !~ /^#/  &&  country == $2) {
    460   1.1       jtc 						cc = $1
    461   1.1       jtc 						break
    462   1.1       jtc 					}
    463   1.1       jtc 				}
    464   1.1       jtc 			}
    465  1.11  christos 			$1 ~ cc && $4 == region { print $3 }
    466  1.11  christos 		' <"$TZ_ZONE_TABLE"`
    467   1.9  christos 		esac
    468   1.1       jtc 
    469   1.1       jtc 		# Make sure the corresponding zoneinfo file exists.
    470   1.1       jtc 		TZ_for_date=$TZDIR/$TZ
    471  1.11  christos 		<"$TZ_for_date" || {
    472   1.1       jtc 			echo >&2 "$0: time zone files are not set up correctly"
    473   1.1       jtc 			exit 1
    474   1.1       jtc 		}
    475   1.1       jtc 	esac
    476   1.1       jtc 
    477   1.1       jtc 
    478   1.1       jtc 	# Use the proposed TZ to output the current date relative to UTC.
    479   1.1       jtc 	# Loop until they agree in seconds.
    480   1.1       jtc 	# Give up after 8 unsuccessful tries.
    481   1.1       jtc 
    482   1.1       jtc 	extra_info=
    483   1.1       jtc 	for i in 1 2 3 4 5 6 7 8
    484   1.1       jtc 	do
    485  1.10  christos 		TZdate=`LANG=C TZ="$TZ_for_date" date`
    486  1.10  christos 		UTdate=`LANG=C TZ=UTC0 date`
    487  1.10  christos 		TZsec=`expr "$TZdate" : '.*:\([0-5][0-9]\)'`
    488  1.10  christos 		UTsec=`expr "$UTdate" : '.*:\([0-5][0-9]\)'`
    489   1.1       jtc 		case $TZsec in
    490   1.1       jtc 		$UTsec)
    491   1.1       jtc 			extra_info="
    492   1.1       jtc Local time is now:	$TZdate.
    493   1.1       jtc Universal Time is now:	$UTdate."
    494   1.1       jtc 			break
    495   1.1       jtc 		esac
    496   1.1       jtc 	done
    497   1.1       jtc 
    498   1.1       jtc 
    499   1.1       jtc 	# Output TZ info and ask the user to confirm.
    500   1.1       jtc 
    501   1.1       jtc 	echo >&2 ""
    502   1.1       jtc 	echo >&2 "The following information has been given:"
    503   1.1       jtc 	echo >&2 ""
    504   1.9  christos 	case $country%$region%$coord in
    505   1.9  christos 	?*%?*%)	echo >&2 "	$country$newline	$region";;
    506   1.9  christos 	?*%%)	echo >&2 "	$country";;
    507   1.9  christos 	%?*%?*) echo >&2 "	coord $coord$newline	$region";;
    508   1.9  christos 	%%?*)	echo >&2 "	coord $coord";;
    509   1.1       jtc 	+)	echo >&2 "	TZ='$TZ'"
    510   1.1       jtc 	esac
    511   1.1       jtc 	echo >&2 ""
    512   1.1       jtc 	echo >&2 "Therefore TZ='$TZ' will be used.$extra_info"
    513   1.1       jtc 	echo >&2 "Is the above information OK?"
    514   1.1       jtc 
    515  1.10  christos 	doselect Yes No
    516  1.10  christos 	ok=$select_result
    517   1.1       jtc 	case $ok in
    518   1.1       jtc 	Yes) break
    519   1.1       jtc 	esac
    520   1.9  christos do coord=
    521   1.1       jtc done
    522   1.1       jtc 
    523   1.5    kleink case $SHELL in
    524   1.5    kleink *csh) file=.login line="setenv TZ '$TZ'";;
    525   1.5    kleink *) file=.profile line="TZ='$TZ'; export TZ"
    526   1.5    kleink esac
    527   1.5    kleink 
    528   1.5    kleink echo >&2 "
    529   1.5    kleink You can make this change permanent for yourself by appending the line
    530   1.5    kleink 	$line
    531   1.5    kleink to the file '$file' in your home directory; then log out and log in again.
    532   1.5    kleink 
    533   1.5    kleink Here is that TZ value again, this time on standard output so that you
    534   1.5    kleink can use the $0 command in shell scripts:"
    535   1.5    kleink 
    536   1.1       jtc echo "$TZ"
    537