if alert:
if verbose: return alert
else:
- if alert.find("\nNATIONAL WEATHER SERVICE") == -1:
- muted = False
- else:
+ import re
+ if re.search(r"\nNational Weather Service", alert):
muted = True
+ else:
+ muted = False
+ expirycheck = re.search(r"Expires:([0-9]{12})", alert)
+ if expirycheck:
+ # only report alerts and forecasts that expired less than
+ # offset ago
+ import datetime, zoneinfo
+ expiration = datetime.datetime.fromisoformat(
+ "%sT%sZ" % (expirycheck[1][:8], expirycheck[1][-4:]))
+ now = datetime.datetime.now(tz=zoneinfo.ZoneInfo("UTC"))
+ # TODO: make this offset configurable
+ if now - expiration > datetime.timedelta(hours=1):
+ return ""
lines = alert.split("\n")
- import time
- # TODO: make this offset configurable
- # TODO: adjust offset relative to the difference between the user's
- # local time and the zone's local time (will need to extend
- # the schema in the zones file to store each tz
- offset = 86400 # one day
-
- # report alerts and forecasts that expired less than offset ago;
- # this is a cheap hack since expiration times seem to be relative
- # to the zone's local timezone, and converting from the user's
- # would get complicated, but also there can sometimes be a lag
- # between expiration and the next update
- valid_time = time.strftime(
- "%Y%m%d%H%M", time.localtime(time.time() - offset))
output = []
for line in lines:
- if line.startswith("Expires:") \
- and "Expires:" + valid_time > line:
- return ""
- if muted and line.startswith("NATIONAL WEATHER SERVICE"):
+ if muted and line.startswith("National Weather Service"):
muted = False
line = ""
elif line == "&&":
return tuple(coordinates)
def correlate():
- import codecs, csv, datetime, hashlib, os, re, sys, time, zipfile
+ import codecs, csv, datetime, hashlib, os, re, sys, time, zipfile, zoneinfo
if pyversion("3"): import configparser
else: import ConfigParser as configparser
for filename in os.listdir("."):
zones[zone]["zone_forecast"] = (
"https://tgftp.nws.noaa.gov/data/forecasts/zone/"
"%s/%s.txt" % (state.lower(), zone))
+ tzcode = fields[7]
+ if tzcode == "A":
+ zones[zone]["tz"] = "US/Alaska"
+ elif tzcode == "AH":
+ zones[zone]["tz"] = "US/Aleutian"
+ elif tzcode in ("C", "CE", "CM"):
+ zones[zone]["tz"] = "US/Central"
+ elif tzcode in ("E", "e"):
+ zones[zone]["tz"] = "US/Eastern"
+ elif tzcode == "F":
+ zones[zone]["tz"] = "Pacific/Guadalcanal"
+ elif tzcode == "G":
+ zones[zone]["tz"] = "Pacific/Guam"
+ elif tzcode == "H":
+ zones[zone]["tz"] = "US/Hawaii"
+ elif tzcode == "J":
+ zones[zone]["tz"] = "Japan"
+ elif tzcode == "K":
+ zones[zone]["tz"] = "Pacific/Kwajalein"
+ elif tzcode in ("M", "MC", "MP"):
+ zones[zone]["tz"] = "US/Mountain"
+ elif tzcode == "m":
+ zones[zone]["tz"] = "US/Arizona"
+ elif tzcode == "P":
+ zones[zone]["tz"] = "US/Pacific"
+ elif tzcode == "S":
+ zones[zone]["tz"] = "US/Samoa"
+ elif tzcode == "V":
+ zones[zone]["tz"] = "America/Virgin"
+ else:
+ zones[zone]["tz"] = ""
county = fields[5]
if county:
if description.endswith(county):
zctas_nocentroid += 1
zones_nocentroid = 0
zones_nodescription = 0
+ zones_notz = 0
zones_noforecast = 0
zones_overlapping = 0
zonetable = {}
if not zones.has_option(zone, "description"):
qalog.append("%s: no description\n" % zone)
zones_nodescription += 1
+ if not zones.has_option(zone, "tz") or not zones.get(
+ zone, "tz") in zoneinfo.available_timezones():
+ qalog.append("%s: no time zone\n" % zone)
+ zones_notz += 1
if not zones.has_option(zone, "zone_forecast"):
qalog.append("%s: no forecast\n" % zone)
zones_noforecast += 1
print(" %s zones with no centroid"%zones_nocentroid)
if zones_nodescription:
print(" %s zones with no description"%zones_nodescription)
+ if zones_notz:
+ print(" %s zones with no time zone"%zones_notz)
if zones_noforecast:
print(" %s zones with no forecast"%zones_noforecast)
if zones_overlapping: