aboutsummaryrefslogtreecommitdiff
path: root/modules
diff options
context:
space:
mode:
authorGravatar Evelyn2016-12-16 10:05:54 +0000
committerGravatar Evelyn2016-12-16 10:05:54 +0000
commitba16f84b1519711020d67c6d5f40de184b34d10b (patch)
tree4738dc44a1de5d32527f7fd00953b7a8eada413f /modules
parentAdd quote to signals module (diff)
signature
Convert NR module to use SV for more detail. Abridge some service info, add primitive service filtering, command for headcode/uid/rsid service summaries. If you have a NR API key, you'll need to switch it!
Diffstat (limited to 'modules')
-rw-r--r--modules/nr.py156
1 files changed, 107 insertions, 49 deletions
diff --git a/modules/nr.py b/modules/nr.py
index 19ced31e..d43c74f3 100644
--- a/modules/nr.py
+++ b/modules/nr.py
@@ -1,8 +1,15 @@
-import collections, datetime, re, time
+import collections, re, time
+from datetime import datetime
import Utils
from suds.client import Client
+from collections import Counter
-URL = 'https://lite.realtime.nationalrail.co.uk/OpenLDBWS/wsdl.aspx?2016-02-16'
+# Note that this module requires the open *Staff Version* of the Darwin API
+# You can register for an API key here: http://openldbsv.nationalrail.co.uk/
+# We use this instead of the 'regular' version because it offers a *lot* more
+# information.
+
+URL = 'https://lite.realtime.nationalrail.co.uk/OpenLDBSVWS/wsdl.aspx?ver=2016-02-16'
class Module(object):
@@ -16,8 +23,12 @@ class Module(object):
usage="<crs_id>")
bot.events.on("received").on("command").on("nrservice"
).hook(self.service, min_args=1,
- help="Get train service information for a Darwin ID (Powered by NRE)",
+ help="Get train service information for a headcode or RID (Powered by NRE)",
usage="<service_id>")
+ bot.events.on("received").on("command").on("nrhead"
+ ).hook(self.head, min_args=1,
+ help="Get information for a given headcode (Powered by NRE)",
+ usage="<headcode>")
def time_compare(self, one, two):
return (one.hour - two.hour) * 60 + (one.minute - two.minute)
@@ -40,34 +51,48 @@ class Module(object):
def arrivals(self, event):
token = self.bot.config["nre-api-key"]
- crs = event["args_split"][0].upper()
+ location_code = event["args_split"][0].upper()
client = Client(URL)
header_token = client.factory.create('ns2:AccessToken')
header_token.TokenValue = token
client.set_options(soapheaders=header_token)
- query = client.service.GetDepartureBoard(50, crs)
-
+ method = client.service.GetDepartureBoardByCRS if len(location_code) == 3 else client.service.GetDepartureBoardByTIPLOC
+ query = method(50, location_code, datetime.now().isoformat().split(".")[0], 120,
+ client.factory.create("filterList"), "to", '', "PBS", False)
trains = []
for t in query["trainServices"][0]:
trains.append({
- "estimated" : t["etd"],
- "scheduled" : t["std"],
- "time" : t["std"] if t["etd"] == "On time" else t["etd"],
- "on_time" : t["etd"] == "On time",
+ "scheduled" : datetime.strptime(t["std"], "%Y-%m-%dT%H:%M:%S"),
+ "called" : "atd" in t,
"dest_name": t["destination"][0][0]["locationName"],
- "dest_id": t["destination"][0][0]["crs"],
- "service_id" : t["serviceID"],
+ "dest_id": t["destination"][0][0]["crs"] if "crs" in t["destination"][0][0] else "---",
+ "rid" : t["rid"],
+ "uid" : t["uid"],
+ "head" : t["trainid"],
"via": '' if not "via" in t["destination"][0][0] else t["destination"][0][0]["via"],
"platform": "?" if not "platform" in t else t["platform"]
})
+ if "etd" in t or "atd" in t:
+ trains[-1]["departure"] = datetime.strptime(t["etd"] if "etd" in t else t["atd"], "%Y-%m-%dT%H:%M:%S")
+ trains[-1]["time"] = trains[-1]["departure"].strftime("%H%M")
+ elif "isCancelled" in t and t["isCancelled"]:
+ trains[-1]["departure"] = "Cancelled"
+ trains[-1]["time"] = "Cancelled"
+ else:
+ trains[-1]["departure"] = t["departureType"]
+ trains[-1]["time"] = t["departureType"]
+
+ trains[-1]["on_time"] = trains[-1]["scheduled"] == trains[-1]["departure"]
+
+
for t in trains:
t["dest_via"] = t["dest_name"] + (" " if t["via"] else '') + t["via"]
- trains = sorted(trains, key=lambda t: int(t["scheduled"].replace(":", "")))
+ trains = sorted(trains, key=lambda t: t["scheduled"])
trains_filtered = []
@@ -80,73 +105,106 @@ class Module(object):
self.result_map[event["target"].id] = trains_filtered
- trains_string = ", ".join(["%s (%s, %s%s%s)" % (t["dest_via"], t["platform"],
+ trains_string = ", ".join(["%s (%s, %s, %s%s%s)" % (t["dest_via"], t["uid"], t["platform"],
Utils.color(Utils.COLOR_GREEN if t["on_time"] else Utils.COLOR_RED),
t["time"],
Utils.color(Utils.FONT_RESET)
) for t in trains_filtered])
- event["stdout"].write("%s (%s): %s" % (query["locationName"], crs,
+ event["stdout"].write("%s (%s): %s" % (query["locationName"], query["crs"],
trains_string))
def service(self, event):
- colours = [Utils.COLOR_LIGHTBLUE, Utils.COLOR_GREEN, Utils.COLOR_RED]
+ colours = [Utils.COLOR_LIGHTBLUE, Utils.COLOR_GREEN, Utils.COLOR_RED, Utils.COLOR_CYAN]
- token = self.bot.config["nre-api-key"]
- service_id = event["args_split"][0]
- if service_id.isdigit():
- if not event["target"].id in self.result_map:
- event["stdout"].write("No history")
- return
- results = self.result_map[event["target"].id]
- if int(service_id) >= len(results):
- event["stdout"].write("%s is too high. Remember that the first departure is 0" % service_id)
- return
- service_id = results[int(service_id)]["service_id"]
+ service_id = event["args_split"][0]
+ filter = event["args_split"][1] if len(event["args_split"]) > 1 else ""
+ token = self.bot.config["nre-api-key"]
client = Client(URL)
-
header_token = client.factory.create('ns2:AccessToken')
header_token.TokenValue = token
client.set_options(soapheaders=header_token)
- query = client.service.GetServiceDetails(service_id)
- stations = [{
- "name": query["locationName"], "crs": query["crs"],
- "scheduled": query["std"],
- "time": query["etd"] if "etd" in query else query["atd"],
- "length": query["length"],
- "called": "atd" in query
- }]
- for station in query["subsequentCallingPoints"][0][0][0]:
+ rid = service_id
+ if len(service_id) <= 8:
+ query = client.service.QueryServices(service_id, datetime.utcnow().date().isoformat(),
+ datetime.utcnow().time().strftime("%H:%M:%S+0000"))
+ if len(query["serviceList"][0]) > 1:
+ event["stderr"].write("Headcode refers to multiple services: " +
+ ", ".join(["%s (%s->%s)" % (a["rid"], a["originCrs"], a["destinationCrs"]) for a in query["serviceList"][0]]))
+ return
+ rid = query["serviceList"][0][0]["rid"]
+
+ query = client.service.GetServiceDetailsByRID(rid)
+
+ stations = []
+ for station in query["locations"][0]:
stations.append(
{"name": station["locationName"],
- "crs": station["crs"],
- "scheduled": station["st"],
- "time": station["et"] if "et" in station else station["at"],
- "length": station["length"],
- "called": "at" in station
+ "crs": (station["crs"] if "crs" in station else station["tiploc"]).rstrip(),
+ "scheduled": datetime.strptime(station["sta"] if "sta" in station else station["std"], "%Y-%m-%dT%H:%M:%S"),
+ "scheduled_type" : "arrival" if "sta" in station else "departure",
+ "scheduled_short": '' if "sta" in station else "d",
+ "called": "atd" in station or "ata" in station,
+ "passing": station["isPass"] if "isPass" in station else False,
+ "prediction": "eta" in station or "etd" in station and not "atd" in station,
+ "first": len(stations) == 0,
+ "last" : False
})
+ stations[-1]["arrival"] = datetime.strptime(station["eta"] if "eta" in station else station["ata"], "%Y-%m-%dT%H:%M:%S") if "eta" in station or "ata" in station else None
+ stations[-1]["departure"] = datetime.strptime(station["etd"] if "etd" in station else station["atd"], "%Y-%m-%dT%H:%M:%S") if "etd" in station or "atd" in station else None
+ stations[-1]["time"], stations[-1]["timeprefix"] = [a for a in [(stations[-1]["arrival"], ''), (stations[-1]["departure"], "d"), (stations[-1]["scheduled"], stations[-1]["scheduled_short"] + "s")] if a[0] != None][0]
+ stations[-1]["time"] = stations[-1]["time"].strftime("%H%M")
+
+ [a for a in stations if a["called"] or a["first"]][-1]["last"] = True
+
+ for station in stations[0:[(k,v) for k,v in enumerate(stations) if v["last"]][0][0]]:
+ if not station["first"]: station["called"] = True
for station in stations:
station["on_time"] = station["time"] == "On time"
station["status"] = 1 if station["on_time"] else 2
if station["called"]: station["status"] = 0
+ if station["passing"]: station["status"] = 3
- if station["time"] == "On time": station["time"] = station["scheduled"]
-
- station["summary"] = "%s (%s, %s %s%s%s)" % (
- station["name"], station["crs"], "at" if station["called"] else "est",
+ station["summary"] = "%s%s(%s, %s%s%s%s)" % (
+ "*" if station["passing"] else '',
+ station["name"] + " " if station["name"] != station["crs"] else '',
+ station["crs"], ("~" if station["prediction"] else '') + station["timeprefix"],
Utils.color(colours[station["status"]]),
station["time"],
Utils.color(Utils.FONT_RESET)
)
+ stations_filtered = []
+ for station in stations:
+ if station["passing"] and filter != "*": continue
+ if station["called"] and filter != "*":
+ if not station["first"] and not station["last"]:
+ continue
+ stations_filtered.append(station)
+ if station["first"] and not station["last"]:
+ stations_filtered.append({"summary": "(...)"})
+
done_count = len([s for s in stations if s["called"]])
total_count = len(stations)
- event["stdout"].write("%s car %s train (%s/%s): %s" % (query["length"],
- query["operator"], done_count, total_count,
- ", ".join([s["summary"] for s in stations])))
+ event["stdout"].write("%s train (%s/%s/%s): %s" % (query["operator"],
+ done_count, len(stations_filtered), total_count,
+ ", ".join([s["summary"] for s in stations_filtered])))
+
+ def head(self, event):
+ service_id = event["args_split"][0]
+
+ token = self.bot.config["nre-api-key"]
+ client = Client(URL)
+ header_token = client.factory.create('ns2:AccessToken')
+ header_token.TokenValue = token
+ client.set_options(soapheaders=header_token)
+
+ query = client.service.QueryServices(service_id, datetime.utcnow().date().isoformat(),
+ datetime.utcnow().time().strftime("%H:%M:%S+0000"))
+ event["stdout"].write(", ".join(["h/%s r/%s u/%s rs/%s %s (%s) -> %s (%s)" % (a["trainid"], a["rid"], a["uid"], a["rsid"], a["originName"], a["originCrs"], a["destinationName"], a["destinationCrs"]) for a in query["serviceList"][0]]))