adapt to (tarc.)rideralerts.com api

This commit is contained in:
steven 2025-03-22 20:12:49 -04:00
parent 2f7d67aa5a
commit 0db90601d9
3 changed files with 17 additions and 94 deletions

43
app.py
View file

@ -1,50 +1,23 @@
from flask import Flask, jsonify, render_template, send_from_directory
import requests
import re
import json
app = Flask(__name__)
routes={}
txt=requests.get("https://www.ridetarc.org/getting-around/routes/").text
for r in re.findall(r'"rtNo">(.*)</span>[\s]*(.*)</h5>[\s\S]*?href="([^"]*)',txt):
stops=json.loads(re.search(r'triangleCoords = (.*);',requests.get(r[2]).text).group(1))
e=dict()
for i in stops:
code=i["stop_code"]
if code in e:
e[code].append(i["stop_sequence"])
e[code]=e[code][:3]+sorted(e[code][3:])
else:
e[code]=[i["lat"],i["lng"],i["stop_name"],i["stop_sequence"]]
# check for stops in txt, for some reason [4, 2, 6] are not there, and the sequence numbers are gone
routes[int(r[0])]={"name":r[1],"stops":e}
data=requests.get("https://tarc.rideralerts.com/InfoPoint/rest/Routes/GetVisibleRoutes").json()
routes = ",".join(str(r.get('RouteId')) for r in data)
@app.route("/")
def index():
return render_template("index.html",routes=routes)
@app.route("/routez")
def routez():
return routes
return render_template("map.html")
@app.route("/bus.svg")
def favicon():
return send_from_directory(app.static_folder, "bus.svg")
return send_from_directory(app.static_folder,"bus.svg")
@app.route('/<int:route>')
def map(route):
if route not in routes: return "not a valid route"
return render_template("map.html",route=route,stops=json.dumps(routes[route]["stops"],separators=(',',':')))
@app.route('/<int:route>.csv')
def api(route):
if route not in routes: return "400"
u='https://www.ridetarc.org/wp-admin/admin-ajax.php?action=route_vehicle&route_id='+('0'+str(route))[-2:]
d=requests.get(u).json()
result=[]
for i in d["data"]: result.append(f"{i["vehicle"]["id"]},{i["timestamp"]},{i["position"]["latitude"]},{i["position"]["longitude"]}")
return "\n".join(result)
@app.route('/tarc.csv')
def tarc():
d=requests.get("https://tarc.rideralerts.com/InfoPoint/rest/Vehicles/GetAllVehiclesForRoutes?routeIDs=0,"+routes).json()
return "\n".join(f"{v['Latitude']},{v['Longitude']},{v['RouteId']},{v['Name']},{v['OnBoard']},{v['DirectionLong']},{v['Speed']}" for v in d)
if __name__ == '__main__':
app.run()

View file

@ -1,27 +0,0 @@
<!DOCTYPE html>
<html lang="en">
<head>
<title>TARC Routes</title>
<meta charset="UTF-8">
<meta content="Realtime TARC bus maps" property="og:title">
<meta content='https://tarc.stevenalexander.org/bus.svg' property='og:image'>
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="icon" type="image/x-icon" href="/bus.svg">
<style>
body {
font-family: monospace;
font-size: 1cm;
padding: 0;
margin: 0;
}
</style>
</head>
<body>
Find <a href="https://www.ridetarc.org/about-us/news/">news</a> and <a href="https://www.ridetarc.org/alerts/">alerts</a> at ridetarc.org<br>
Some busses don't provide location data.<br>
This service is provided without warranty.<br>
{%- for route in routes %}
<a href="{{route}}">{{'%02d'%route}} {{routes[route]["name"]}}</a><br>
{%- endfor %}
</body>
</html>

View file

@ -1,59 +1,36 @@
<!DOCTYPE html>
<html lang="en">
<head>
<title>Route {{route}}</title>
<title>TARC Routes</title>
<meta charset="UTF-8">
<meta content="Route {{route}}" property="og:title">
<meta content="TARC Routes" property="og:title">
<meta content='https://tarc.stevenalexander.org/bus.svg' property='og:image'>
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="icon" type="image/x-icon" href="/bus.svg">
<script src="https://unpkg.com/leaflet@1.9.4/dist/leaflet.js" integrity="sha256-20nQCchB9co0qIjJZRGuk2/Z9VM+kNiyxNV1lvTlZBo=" crossorigin=""></script>
<link rel="stylesheet" href="https://unpkg.com/leaflet@1.9.4/dist/leaflet.css" integrity="sha256-p4NxAoJBhIIN+hmNHrzRCf9tD/miZyoHS5obTRR9BMY=" crossorigin="" />
<style>
html, body {
height: 100%;
margin: 0;
}
#map {
height: 100vh;
}
#clock {
background-color: #000;
color: #FFF;
font: 16px monospace;
}
html,body{height:100%;margin:0;}
#map{height:100vh;}
#clock{background-color:black;color:white;font:16px monospace;}
</style>
</head>
<body>
<div id="map"></div>
<script>
let map=L.map("map",{center:[38.21,-85.75],zoom:13,layers:[L.tileLayer("https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png")]});
L.control.scale({"position":"topright"}).addTo(map);
document.querySelector(".leaflet-top.leaflet-right").addEventListener("click",function() {
map.getContainer().requestFullscreen()
})
let s={{stops|safe}};
for (i in s) {
L.circle([s[i][0],s[i][1]],{radius:20,color:"#0092da"}).bindPopup(`<b>${s[i][2]} (#${i})</b><br>Sequence #${s[i].slice(3)}`).addTo(map)
}
let busMarkers=L.layerGroup().addTo(map);
function updateMap() {
fetch("{{route}}.csv").then(response=>response.text()).then(data=>{
fetch("tarc.csv").then(response=>response.text()).then(data=>{
if (data == "") return;
busMarkers.clearLayers();
data.split("\n").forEach(line=>{
var [n,t,lat,lng]=line.split(",");
L.marker([lat,lng]).bindPopup("<b>"+n+"</b> at<br>"+new Date(t*1000).toLocaleTimeString()).addTo(busMarkers)
var [lat,lng,r,id,p,d,s]=line.split(",");
L.marker([lat,lng]).bindPopup(`<b>Route ${r}</b><br>#${id}<br>${p} on board<br>${d}<br>@${s}mph`).addTo(busMarkers)
})
})
}
document.querySelector(".leaflet-bottom.leaflet-right").innerHTML='<div id="clock"></div>';
let clockElement=document.querySelector("#clock");
function updateClock() {
clockElement.textContent=busMarkers.getLayers().length+" TARC(s) - "+new Date().toLocaleTimeString()
}
setInterval(updateMap,5000); // use websocket?
setInterval(updateClock,1000);
setInterval(updateMap,5000); // use websocket
let locationMarker=L.marker([0,0]),locationRadius=L.circle([0,0]);
let locationLayer=L.layerGroup([locationMarker,locationRadius]).addTo(map);