/[projects]/android/TrainInfoService/src/dk/thoerup/traininfoservice/banedk/DepartureFetcher.java
ViewVC logotype

Contents of /android/TrainInfoService/src/dk/thoerup/traininfoservice/banedk/DepartureFetcher.java

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1834 - (show annotations) (download)
Tue Aug 21 06:48:57 2012 UTC (11 years, 8 months ago) by torben
File size: 20420 byte(s)
 remove debug output
1 package dk.thoerup.traininfoservice.banedk;
2
3
4 import java.net.URL;
5 import java.net.URLEncoder;
6 import java.util.Collections;
7 import java.util.Comparator;
8 import java.util.Map;
9 import java.util.logging.Logger;
10
11 import org.jsoup.nodes.Document;
12 import org.jsoup.nodes.Element;
13 import org.jsoup.select.Elements;
14
15 import dk.thoerup.android.traininfo.common.DepartureBean;
16 import dk.thoerup.android.traininfo.common.DepartureEntry;
17 import dk.thoerup.android.traininfo.common.StationEntry;
18 import dk.thoerup.circuitbreaker.CircuitBreaker;
19 import dk.thoerup.circuitbreaker.CircuitBreakerManager;
20 import dk.thoerup.genericjavautils.HttpUtil;
21 import dk.thoerup.genericjavautils.TimeoutMap;
22 import dk.thoerup.traininfoservice.Statistics;
23 import dk.thoerup.traininfoservice.TraininfoSettings;
24 import dk.thoerup.traininfoservice.db.StationDAO;
25
26 public class DepartureFetcher {
27
28 enum TrainType{
29 STOG,
30 REGIONAL
31 }
32
33 enum FetchTrainType {
34 STOG,
35 REGIONAL,
36 BOTH
37 }
38
39 Logger logger = Logger.getLogger(DepartureFetcher.class.getName());
40
41 Map<String, DepartureBean> cache;
42
43 StationDAO stationDao = new StationDAO();
44
45
46 private TraininfoSettings settings;
47
48 Comparator<DepartureEntry> departureTimeComparator = new Comparator<DepartureEntry>() {
49
50 @Override
51 public int compare(DepartureEntry arg0, DepartureEntry arg1) {
52 String timeStr1 = arg0.getTime().replace(":","").trim();
53 String timeStr2 = arg1.getTime().replace(":","").trim();
54
55 int time1 = 0;
56 int time2 = 0;
57
58 if (timeStr1.length() > 0)
59 time1 = Integer.parseInt(timeStr1);
60
61 if (timeStr2.length() > 0)
62 time2 = Integer.parseInt(timeStr2);
63
64 //work correctly when clock wraps around at midnight
65 if (Math.abs(time1-time2) < 1200) {
66 if (time1 > time2)
67 return 1;
68 else
69 return -1;
70 } else {
71 if (time1 < time2)
72 return 1;
73 else
74 return -1;
75
76 }
77
78 }
79
80 };
81
82 public DepartureFetcher(TraininfoSettings settings) {
83 this.settings = settings;
84 cache = new TimeoutMap<String,DepartureBean>( settings.getCacheTimeout() );
85 }
86
87
88
89
90 public DepartureBean cachedLookupDepartures(int stationID, boolean arrival, FetchTrainType type) throws Exception {
91
92 final String key = "" + stationID + ":" + arrival + ":" + type.toString();
93
94 DepartureBean departureBean = cache.get(key);
95
96
97 if (departureBean == null) {
98 departureBean = lookupDepartures(stationID, arrival, type);
99 cache.put(key, departureBean);
100 } else {
101 Statistics.getInstance().incrementDepartureCacheHits();
102 logger.info("Departure: Cache hit " + key); //remove before production
103 }
104 return departureBean;
105 }
106
107
108 public DepartureBean lookupDepartures(int stationID, boolean arrival, FetchTrainType type) throws Exception {
109
110 DepartureBean departureBean = new DepartureBean();
111
112 StationEntry station = stationDao.getById(stationID);
113
114 departureBean.stationName = station.getName();
115
116 //TODO: FetchTraintype.Both should be removed some time after 0.9.5 release
117 if (station.getRegional() != null && (type == FetchTrainType.REGIONAL||type == FetchTrainType.BOTH) ) {
118 DepartureBean tempBean = lookupDepartures(station.getRegional(), TrainType.REGIONAL, arrival);
119 departureBean.entries.addAll( tempBean.entries );
120 departureBean.notifications.addAll(tempBean.notifications);
121 }
122
123 if (station.getStrain() != null && (type == FetchTrainType.STOG||type == FetchTrainType.BOTH)) {
124 DepartureBean tempBean = lookupDepartures(station.getStrain(), TrainType.STOG, arrival);
125 departureBean.entries.addAll( tempBean.entries );
126 departureBean.notifications.addAll(tempBean.notifications);
127 }
128
129 if (departureBean.entries.size() == 0) {
130 logger.info("No departures found for station " + stationID);
131 }
132
133 //TODO: FetchTraintype.Both should be removed some time after 0.9.5 release
134 if (type == FetchTrainType.BOTH) { //if we have both S-tog and regional order by departure/arrival time
135 Collections.sort( departureBean.entries, departureTimeComparator);
136 }
137
138 //System.out.println("Trit: " + settings.isTritinfoEnabled() + " " + station.getTritStation() );
139 if ( settings.isTritinfoEnabled() && station.getTritStation() != -1) {
140 try {
141 injectTritinfoData(departureBean, station);
142 } catch (Exception ex) { //det er ikke kritisk at vi får perron numre med
143 ex.printStackTrace();
144 }
145 }
146
147
148 return departureBean;
149 }
150
151 public DepartureBean lookupDepartures(String stationcode, TrainType type, boolean arrival) throws Exception {
152 if ( settings.getBackend() == TraininfoSettings.Backend.Azure) {
153 return lookupDeparturesAzureSite(stationcode, type, arrival);
154 } else {
155 return lookupDeparturesMobileSite(stationcode, type, arrival);
156 }
157 }
158
159 private String getTypeStringAzure(TrainType type) {
160 switch (type) {
161 case STOG:
162 return "S-Tog";
163 case REGIONAL:
164 return "Fjerntog";
165 default:
166 return ""; //Can not happen
167 }
168 }
169
170 private String getTypeStringWww(TrainType type) {
171 switch (type) {
172 case STOG:
173 return "S2";
174 case REGIONAL:
175 return "FJRN";
176 default:
177 return ""; //Can not happen
178 }
179 }
180
181 public DepartureBean lookupDeparturesAzureSite(String stationcode, TrainType type, boolean arrival) throws Exception {
182
183 DepartureBean departureBean = new DepartureBean();
184
185
186 String typeString = getTypeStringAzure(type);
187 String arrivalDeparture = (arrival==false) ? "Afgang" : "Ankomst";
188
189 stationcode = URLEncoder.encode(stationcode,"ISO-8859-1");
190
191 String uri = "http://trafikinfo.bane.dk/Trafikinformation/AfgangAnkomst/" + arrivalDeparture + "/" + stationcode + "/" + typeString + "/UdvidetVisning";
192
193 logger.fine("URI: " + uri);
194 JsoupInvocation wrapper = new JsoupInvocation( new URL(uri), settings.getReplyTimeout() );
195 CircuitBreaker breaker = CircuitBreakerManager.getManager().getCircuitBreaker("banedk");
196
197 Document page = (Document) breaker.invoke(wrapper);
198
199 String tableName = arrival == false ? "afgangtabel" : "ankomsttabel";
200 Element table = page.getElementById(tableName);
201
202 if (table != null) {
203 Elements tableRows = table.getElementsByTag("tr");
204
205 //boolean tidsstregExists = (table.getElementsByAttributeValue("class", "Tidsstreg").size() > 0);
206 //boolean passedTidsstreg = false;
207
208 for (Element currentRow : tableRows) {
209 String rowClass = currentRow.attr("class");
210 /*
211 if (tidsstregExists == true && passedTidsstreg == false) {
212 if (currentRow.getElementsByAttributeValue("class", "Tidsstreg").size() > 0) {
213 passedTidsstreg = true;
214 } else {
215 continue;
216 }
217 }*/
218
219 if (rowClass != null && rowClass.toLowerCase().contains("station") ) {
220
221 Elements fields = currentRow.getElementsByTag("td");
222
223 DepartureEntry departure = new DepartureEntry();
224
225 String time = fields.get(0).text();
226 if (time.equals(""))
227 time = "0:00"; //Bane.dk bug work-around
228 departure.setTime(time);
229
230 int updated = extractUpdated( fields.get(1) );
231 departure.setUpdated(updated);
232
233 String trainNumber = fields.get(2).text();
234 if (type == TrainType.STOG) //If it is S-train we need to extract the trainNumber
235 trainNumber = trainNumber + " " + extractTrainNumberAzure(fields.get(2));
236 departure.setTrainNumber(trainNumber);
237
238 String destination = fields.get(3).text();
239 departure.setDestination(destination);
240
241 String origin = fields.get(4).text();
242 departure.setOrigin(origin);
243
244 String location = fields.get(5).text();
245 departure.setLocation(location);
246
247 String status = fields.get(6).text().trim();
248 departure.setStatus(status);
249
250 String note = extractNote( fields.get(7) );
251 departure.setNote(note);
252
253 departure.setType(typeString);
254
255 departureBean.entries.add( departure );
256 }
257 }
258 } else {
259 logger.warning("No departures found for station=" + stationcode + ", type=" + type);
260 }
261
262 Element notifDiv = page.getElementById("station_planlagte_text");
263 if (notifDiv != null) {
264
265 Elements tables = notifDiv.getElementsByTag("table");
266 for (Element tab : tables) {
267
268 Elements anchors = tab.getElementsByTag("a");
269 if (anchors.size() == 2) {
270 departureBean.notifications.add( anchors.get(1).text() );
271 }
272 }
273
274 }
275
276
277 return departureBean;
278 }
279
280 public DepartureBean lookupDeparturesMobileSite(String stationcode, TrainType traintype, boolean arrival) throws Exception {
281
282 DepartureBean departureBean = new DepartureBean();
283
284
285 String typeString = getTypeStringWww(traintype);
286 String arrivalDeparture = (arrival==false) ? "afgang" : "ankomst";
287
288 stationcode = URLEncoder.encode(stationcode,"ISO-8859-1");
289
290
291 String uri = "http://mobil.bane.dk/mobilStation.asp?artikelID=5332&stat_kode=" + stationcode + "&webprofil=" + typeString +"&beskrivelse=&mode=ankomstafgang&ankomstafgang=" + arrivalDeparture + "&gemstation=&fuldvisning=1";
292 logger.fine("URI: " + uri);
293 JsoupInvocation wrapper = new JsoupInvocation( new URL(uri), settings.getReplyTimeout() );
294 CircuitBreaker breaker = CircuitBreakerManager.getManager().getCircuitBreaker("banedk");
295
296 Document page = (Document) breaker.invoke(wrapper);
297
298
299 Element content = page.getElementsByClass("contentDiv").get(0);
300
301
302 if (content != null) {
303 Elements tableRows = content.child(0).children();
304
305
306
307 for (Element currentRow : tableRows) {
308 if (currentRow.tagName().equals("br") ) {
309 break;
310 }
311
312
313 String link = currentRow.child(0).attr("href");
314
315 logger.fine( currentRow.text() );
316 logger.fine("Href: " + link);
317
318
319 String parts[] = currentRow.text().split(",");
320
321
322 DepartureEntry departure = new DepartureEntry();
323
324 //if we do these things upfront, then we are allowed to use continue statement when row contains no more data
325 departure.setType(typeString);
326 departureBean.entries.add( departure );
327
328 /*
329 http://mobil.bane.dk/mobilStation.asp?artikelID=5332&tognummer=111&webprofil=FJRN&mode=rute&strBemaerkning=Afg%E5r+fra+%C5rhus+H+kl%2E07%3A21++&strRefURL=%2FmobilStation%2Easp%3FartikelID%3D5332%26stat%5Fkode%3DAR%26webprofil%3DFJRN%26beskrivelse%3D%25C5rhus%2BH%26mode%3Dankomstafgang%26ankomstafgang%3Dafgang%26gemstation%3D
330 */
331 int offset = 0;
332
333 String time = parts[offset++];
334 if (time.equals(""))
335 time = "0:00"; //Bane.dk bug work-around
336 departure.setTime(time);
337
338 int updated = 4; //does not exist on mobile
339 departure.setUpdated(updated);
340
341 String trainNumber = extractTrainNumberMobile(link);
342 /*if (traintype == TrainType.STOG) //If it is S-train we need to extract the trainNumber
343 trainNumber = trainNumber + " " + extractTrainNumberAzure(fields.get(2));*/
344 departure.setTrainNumber(trainNumber);
345
346 if (traintype == TrainType.STOG) { //if it is stog the next vield is the "Line" code - this should be used somewhere, but skippint ahead for now
347 String stogLine = parts[offset++].trim();
348 departure.setTrainNumber(stogLine + " " + trainNumber);
349 }
350
351 String destination = parts[offset++].trim();;
352 departure.setDestination(destination);
353
354 String origin = "-"; // fields.get(4).text(); does not exist on mobile
355 departure.setOrigin(origin);
356
357 String location = ""; // fields.get(5).text(); does not exist on mobile
358 departure.setLocation(location);
359
360 if (offset == parts.length) {
361 continue;
362 }
363
364 if (parts[offset].trim().equalsIgnoreCase("NB!")) {
365 offset++;
366 }
367
368 if (offset == parts.length) {
369 continue;
370 }
371
372 String status = parts[offset++].trim();; //fields.get(6).text().trim(); - extract from url
373 departure.setStatus(status);
374
375 String note = ""; //extractNote( fields.get(7) ); - extract from url
376 departure.setNote(note);
377
378 }
379 } else {
380 logger.warning("No departures found for station=" + stationcode + ", type=" + traintype);
381 }
382
383 return departureBean;
384 }
385
386
387
388 public static String cleanText(String input) {
389 //apparently JSoup translates &nbsp; characters on www.bane.dk to 0xA0
390 return input.replace((char) 0xA0, (char)0x20).trim();
391 }
392
393
394 // old www site is not available any more
395 @Deprecated
396 public DepartureBean lookupDeparturesWwwSite(String stationcode, TrainType trainType, boolean arrival) throws Exception {
397
398 DepartureBean departureBean = new DepartureBean();
399
400 String type = getTypeStringWww(trainType);
401
402 stationcode = URLEncoder.encode(stationcode, "ISO-8859-1");
403
404
405 String uri = "http://www.bane.dk/visStation.asp?ArtikelID=4275&W=" + type + "&S=" + stationcode;
406 logger.fine("URI:" + uri);
407
408
409 JsoupInvocation wrapper = new JsoupInvocation( new URL(uri), settings.getReplyTimeout() );
410 CircuitBreaker breaker = CircuitBreakerManager.getManager().getCircuitBreaker("banedk");
411
412 Element page = (Element) breaker.invoke(wrapper);
413
414 String tableName = arrival == false ? "afgangtabel" : "ankomsttabel";
415 Element table = page.getElementById(tableName);
416
417
418
419 if (table != null) {
420 Elements tableRows = table.getElementsByTag("tr");
421
422 //boolean passedTidsstreg = false;
423 //boolean tidsstregExists = (table.getElementsByAttributeValue("class", "Tidsstreg").size() > 0);
424
425 for (Element currentRow : tableRows) {
426 String rowClass = currentRow.attr("class");
427 /*
428 if (tidsstregExists == true && passedTidsstreg == false) {
429 if (currentRow.getElementsByAttributeValue("class", "Tidsstreg").size() > 0) {
430 passedTidsstreg = true;
431 } else {
432 continue;
433 }
434 }*/
435
436
437 if (rowClass != null && rowClass.toLowerCase().contains("station") ) {
438 Elements fields = currentRow.getElementsByTag("td");
439
440 DepartureEntry departure = new DepartureEntry();
441
442
443
444 String time = cleanText( fields.get(0).getAllElements().get(2).text() );
445 if (time.equals(""))
446 time = "0:00"; //Bane.dk bug work-around
447 departure.setTime(time);
448
449 int updated = extractUpdated( fields.get(1) );
450 departure.setUpdated(updated);
451
452 String trainNumber = cleanText( fields.get(2).text() );
453 if (type.equalsIgnoreCase("S2")) //If it is S-train we need to extract the trainNumber
454 trainNumber = trainNumber + " " + extractTrainNumberWww(fields.get(2));
455 departure.setTrainNumber(trainNumber);
456
457 String destination = cleanText( fields.get(3).text() );
458 departure.setDestination(destination);
459
460 String origin = cleanText( fields.get(4).text() );
461 departure.setOrigin(origin);
462
463 String location = cleanText( fields.get(5).text() );
464 departure.setLocation(location);
465
466 String status = cleanText( fields.get(6).text() );
467 departure.setStatus(status);
468
469 String note = cleanText( extractNote( fields.get(7) ) );
470 departure.setNote(note);
471
472 departure.setType(type);
473
474 departureBean.entries.add(departure);
475
476
477 }
478 }
479 } else {
480 logger.warning("No departures found for station=" + stationcode + ", type=" + type);
481 }
482
483
484 return departureBean;
485 }
486
487
488 private int extractUpdated(Element updatedTd) { //extract the digit (in this case: 4) from "media/trafikinfo/opdater4.gif"
489 int updated = -1;
490
491 Elements updatedImgs = updatedTd.getElementsByTag("img");
492 String updatedStr = updatedImgs.get(0).attr("src");
493
494 if (updatedStr != null) {
495 for (int i=0; i<updatedStr.length(); i++) {
496 char c = updatedStr.charAt(i);
497 if ( Character.isDigit(c)) {
498 updated = Character.digit(c, 10);
499 break;
500 }
501 }
502 }
503 return updated;
504 }
505
506 private String extractNote(Element noteTd) {
507 String note = noteTd.text().trim();
508
509
510 Elements elems = noteTd.getElementsByClass("bemtype");
511 if (elems.size() > 0 && note.charAt(note.length()-1) == 'i')
512 note = note.substring(0,note.length() -1 );
513
514 return note.trim();
515 }
516
517 private String extractTrainNumberAzure(Element trainTd) {
518 Element anchorElement = trainTd.getElementsByTag("a").get(0);
519 String href = anchorElement.attr("href");
520
521 int pos = href.lastIndexOf('/');
522 String number = href.substring(pos+1);
523
524 return number;
525 }
526
527 private String extractTrainNumberMobile(String link) {
528 Map<String,String> elements = HttpUtil.decodeParams(link);
529
530 return elements.get("tognummer");
531 }
532
533 private String extractTrainNumberWww(Element trainTd) {
534 String number = "";
535 Element anchorElement = trainTd.getElementsByTag("a").get(0);
536 String href = anchorElement.attr("href");
537
538 String argstring = href.split("?")[1];
539 Map<String,String> elements = HttpUtil.decodeParams(argstring);
540 number = elements.get("TogNr");
541
542
543 /*String argstring = href.substring( href.indexOf('?') + 1);
544 String args[] = argstring.split("&");
545 for (String arg : args) {
546 String pair[] = arg.split("="); // Key=pair[0], Value=pair[1]
547
548 if (pair[0].equalsIgnoreCase("TogNr"))
549 number = pair[1];
550 }*/
551
552
553 return number;
554 }
555
556
557 private void injectTritinfoData(DepartureBean departureBean, StationEntry station) throws Exception {
558 String uri = "http://tritinfo.pallas.dk/webtavle?page=stationcontent&staid=" + station.getTritStation();
559 logger.fine("URI:" + uri);
560 System.out.println("URI:" + uri);
561
562
563 JsoupInvocation wrapper = new JsoupInvocation( new URL(uri), settings.getReplyTimeout() );
564 CircuitBreaker breaker = CircuitBreakerManager.getManager().getCircuitBreaker("tritinfo");
565
566 Element page = (Element) breaker.invoke(wrapper);
567
568 Element table = page.getElementsByClass("passages").get(0);
569
570 Elements trains = table.getElementsByClass("train");
571
572 for (int i=0; i<trains.size(); i++) {
573 Element train = trains.get(i);
574
575 //String trainType = train.getElementsByClass("trainType").get(0).text();
576 String trainNumber = train.getElementsByClass("trainNumber").get(0).text();
577
578 Elements trackElems = train.getElementsByClass("plannedTrack");
579 if (trackElems.size() == 0) {
580 trackElems = train.getElementsByClass("expectedTrack");
581 }
582 String track = trackElems.get(0).text().trim();
583 String trackType = train.getElementsByClass("trackType").get(0).text().trim();
584
585
586 String platform = track;
587 if ( !trackType.equals("")) {
588 platform += trackType;
589 }
590
591 for (DepartureEntry entry : departureBean.entries) {
592 String entryTrainId = entry.getTrainNumber().split(" ")[1];
593 if ( entryTrainId.equals(trainNumber)) {
594
595 entry.setPlatform(platform);
596 break;
597 }
598 }
599 }
600 }
601
602
603 //test
604 /*
605 public static void main(String args[]) throws Exception {
606 DepartureFetcher f = new DepartureFetcher();
607 List<DepartureBean> deps = f.lookupDepartures("AR", "FJRN");
608 for(DepartureBean d : deps) {
609 System.out.println( d.getTime() + ";" + d.getUpdated() + ";" + d.getTrainNumber() + ";" +
610 d.getDestination() + ";" + d.getOrigin() + ";" + d.getLocation() + ";" + d.getStatus() + ";" + d.getNote() );
611 }
612
613 System.out.println("--------------------------");
614 }*/
615 }

  ViewVC Help
Powered by ViewVC 1.1.20