You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

138 lines
4.1 KiB

7 years ago
7 years ago
7 years ago
7 years ago
7 years ago
7 years ago
7 years ago
7 years ago
7 years ago
7 years ago
7 years ago
7 years ago
7 years ago
7 years ago
  1. import argparse
  2. import re
  3. import ctabus
  4. from dateutil.parser import parse as date_parse
  5. import datetime
  6. # for logging
  7. import sys
  8. import os.path as osp
  9. def numb_sort(str):
  10. n = 40
  11. try:
  12. return re.sub(r'\d+',lambda match: match.group(0).rjust(n,'0'),str)
  13. except Exception as E:
  14. print(str)
  15. raise E
  16. def pprint_delta(delta):
  17. delta = str(delta)
  18. days= None
  19. s1 = delta.split(', ')
  20. if len(s1) > 1:
  21. days,time = s1
  22. else:
  23. time = s1[0]
  24. time = time.split('.')[0]
  25. hour,minute,second = map(int,time.split(':'))
  26. time = ''
  27. if hour:
  28. time += f'{hour} hour' + ('s' if hour != 1 else '')
  29. if minute:
  30. if time and not time.endswith(', '):
  31. time += ', '
  32. time += f'{minute} minute' + ('s' if minute != 1 else '')
  33. if second:
  34. if time and not time.endswith(', '):
  35. time += ', '
  36. time += f'{second} second' + ('s' if second != 1 else '')
  37. ret = ''
  38. if days:
  39. ret = days + ', ' if time else ''
  40. ret += time
  41. return ret
  42. def gen_list(objs,data,*displays,key = None,sort = 0,num_pic = True):
  43. k = displays[sort]
  44. display_data = {obj[k]:obj[data] for obj in objs}
  45. srt_keys = sorted(display_data.keys(),key=key)
  46. display = sorted(
  47. [
  48. [obj[d] for d in displays] for obj in objs
  49. ],
  50. key = lambda row: key(row[sort]) if key else row[sort]
  51. )
  52. if num_pic:
  53. display = [[i] + data for i,data in enumerate(display)]
  54. opts = {
  55. 'spacer':' ',
  56. 'seperator':' ',
  57. 'interactive': True,
  58. 'bottom':'=',
  59. 'l_end':'<',
  60. 'r_end':'>',
  61. }
  62. print2d(display,**opts)
  63. if num_pic:
  64. which = None
  65. while not which:
  66. try:
  67. which = srt_keys[int(input('Which one?: '))]
  68. except ValueError:
  69. which = None
  70. return display_data[which]
  71. else:
  72. ret = None
  73. while not ret:
  74. try:
  75. ret = display_data[input('Which one?: ')]
  76. except KeyError:
  77. pass
  78. return ret
  79. config = '''\
  80. {route} - {end} ({direction})
  81. {nm}, stop {stop_id}
  82. {delta} ({t})\
  83. '''
  84. if __name__ == "__main__":
  85. parser = argparse.ArgumentParser(prog = 'ctabus')
  86. parser.add_argument('arg',nargs = '+',metavar = '(stop-id | cross streets)')
  87. parser.add_argument('-r','--route',default = None)
  88. parser.add_argument('-d','--direction',default = None)
  89. parser.add_argument('-l','--lucky',action='store_true',help = 'picks first result')
  90. args = parser.parse_args()
  91. sys.stderr = open(osp.join(osp.dirname(__file__),'stderr.log'),'w')
  92. args.arg = ' '.join(args.arg)
  93. if not args.arg.isdecimal():
  94. # save on import time slightly
  95. from print2d import print2d
  96. from search import Search,StopSearch
  97. #routes
  98. if not args.route:
  99. data = ctabus.get_routes()['routes']
  100. route = gen_list(data,'rt','rt','rtnm',num_pic = False,key=numb_sort)
  101. else:
  102. route = args.route
  103. data = ctabus.get_directions(route)['directions']
  104. #direction
  105. if not args.direction:
  106. direction = gen_list(data,'dir','dir')
  107. else:
  108. s = Search(args.direction)
  109. direction = sorted((obj['dir'] for obj in data),key = s)[0]
  110. #direction
  111. stops = ctabus.get_stops(route,direction)['stops']
  112. s = StopSearch(args.arg)
  113. if args.lucky:
  114. stop_id = sorted(stops,key=lambda stop: s(stop['stpnm']))[0]['stpid']
  115. else:
  116. stop_id = gen_list(stops,'stpid','stpnm',key = s)
  117. else:
  118. stop_id = args.arg
  119. times = ctabus.get_times(stop_id)['prd']
  120. today = datetime.datetime.today()
  121. for time in sorted(times,key = lambda t: t["prdtm"]):
  122. arrival = date_parse(time['prdtm'])
  123. if arrival > today:
  124. delta = pprint_delta(arrival-today)
  125. t = arrival.strftime('%H:%M:%S')
  126. route = time['rt']
  127. direction = time['rtdir']
  128. end = time['des']
  129. nm = time['stpnm']
  130. print(
  131. config.format(**globals()),end= '\n'*2
  132. )