client.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. # This file is Copyright 2019 by the GPSD project
  2. # SPDX-License-Identifier: BSD-2-Clause
  3. #
  4. # This code run compatibly under Python 2 and 3.x for x >= 2.
  5. # Preserve this property!
  6. # Codacy D203 and D211 conflict, I choose D203
  7. # Codacy D212 and D213 conflict, I choose D212
  8. """gpsd client functions."""
  9. from __future__ import absolute_import, print_function, division
  10. import json
  11. import select
  12. import socket
  13. import sys
  14. import time
  15. from .misc import polystr, polybytes
  16. from .watch_options import *
  17. GPSD_PORT = "2947"
  18. class gpscommon(object):
  19. """Isolate socket handling and buffering from protocol interpretation."""
  20. host = "127.0.0.1"
  21. port = GPSD_PORT
  22. def __init__(self, host="127.0.0.1", port=GPSD_PORT, verbose=0,
  23. should_reconnect=False):
  24. """Init gpscommon."""
  25. self.stream_command = b''
  26. self.linebuffer = b''
  27. self.received = time.time()
  28. self.reconnect = should_reconnect
  29. self.verbose = verbose
  30. self.sock = None # in case we blow up in connect
  31. # Provide the response in both 'str' and 'bytes' form
  32. self.bresponse = b''
  33. self.response = polystr(self.bresponse)
  34. if host is not None and port is not None:
  35. self.host = host
  36. self.port = port
  37. self.connect(self.host, self.port)
  38. def connect(self, host, port):
  39. """Connect to a host on a given port.
  40. If the hostname ends with a colon (`:') followed by a number, and
  41. there is no port specified, that suffix will be stripped off and the
  42. number interpreted as the port number to use.
  43. """
  44. if not port and (host.find(':') == host.rfind(':')):
  45. i = host.rfind(':')
  46. if i >= 0:
  47. host, port = host[:i], host[i + 1:]
  48. try:
  49. port = int(port)
  50. except ValueError:
  51. raise socket.error("nonnumeric port")
  52. # if self.verbose > 0:
  53. # print 'connect:', (host, port)
  54. self.sock = None
  55. for res in socket.getaddrinfo(host, port, 0, socket.SOCK_STREAM):
  56. af, socktype, proto, _canonname, sa = res
  57. try:
  58. self.sock = socket.socket(af, socktype, proto)
  59. # if self.debuglevel > 0: print 'connect:', (host, port)
  60. self.sock.connect(sa)
  61. if self.verbose > 0:
  62. print('connected to tcp://{}:{}'.format(host, port))
  63. break
  64. # do not use except ConnectionRefusedError
  65. # # Python 2.7 doc does have this exception
  66. except socket.error as e:
  67. if self.verbose > 1:
  68. msg = str(e) + ' (to {}:{})'.format(host, port)
  69. sys.stderr.write("error: {}\n".format(msg.strip()))
  70. self.close()
  71. raise # propagate error to caller
  72. def close(self):
  73. """Close the gpsd socket."""
  74. if self.sock:
  75. self.sock.close()
  76. self.sock = None
  77. def __del__(self):
  78. """Close the gpsd socket."""
  79. self.close()
  80. def waiting(self, timeout=0):
  81. """Return True if data is ready for the client."""
  82. if self.linebuffer:
  83. return True
  84. if self.sock is None:
  85. return False
  86. (winput, _woutput, _wexceptions) = select.select(
  87. (self.sock,), (), (), timeout)
  88. return winput != []
  89. def read(self):
  90. """Wait for and read data being streamed from the daemon."""
  91. if None is self.sock:
  92. self.connect(self.host, self.port)
  93. if None is self.sock:
  94. return -1
  95. self.stream()
  96. eol = self.linebuffer.find(b'\n')
  97. if eol == -1:
  98. # RTCM3 JSON can be over 4.4k long, so go big
  99. frag = self.sock.recv(8192)
  100. self.linebuffer += frag
  101. if not self.linebuffer:
  102. if self.verbose > 1:
  103. sys.stderr.write(
  104. "poll: no available data: returning -1.\n")
  105. # Read failed
  106. return -1
  107. eol = self.linebuffer.find(b'\n')
  108. if eol == -1:
  109. if self.verbose > 1:
  110. sys.stderr.write("poll: partial message: returning 0.\n")
  111. # Read succeeded, but only got a fragment
  112. self.response = '' # Don't duplicate last response
  113. self.bresponse = '' # Don't duplicate last response
  114. return 0
  115. else:
  116. if self.verbose > 1:
  117. sys.stderr.write("poll: fetching from buffer.\n")
  118. # We got a line
  119. eol += 1
  120. # Provide the response in both 'str' and 'bytes' form
  121. self.bresponse = self.linebuffer[:eol]
  122. self.response = polystr(self.bresponse)
  123. self.linebuffer = self.linebuffer[eol:]
  124. # Can happen if daemon terminates while we're reading.
  125. if not self.response:
  126. return -1
  127. if 1 < self.verbose:
  128. sys.stderr.write("poll: data is %s\n" % repr(self.response))
  129. self.received = time.time()
  130. # We got a \n-terminated line
  131. return len(self.response)
  132. # Note that the 'data' method is sometimes shadowed by a name
  133. # collision, rendering it unusable. The documentation recommends
  134. # accessing 'response' directly. Consequently, no accessor method
  135. # for 'bresponse' is currently provided.
  136. def data(self):
  137. """Return the client data buffer."""
  138. return self.response
  139. def send(self, commands):
  140. """Ship commands to the daemon."""
  141. lineend = "\n"
  142. if isinstance(commands, bytes):
  143. lineend = polybytes("\n")
  144. if not commands.endswith(lineend):
  145. commands += lineend
  146. if self.sock is None:
  147. self.stream_command = commands
  148. else:
  149. self.sock.send(polybytes(commands))
  150. class json_error(BaseException):
  151. """Class for JSON errors."""
  152. def __init__(self, data, explanation):
  153. """Init json_error."""
  154. BaseException.__init__(self)
  155. self.data = data
  156. self.explanation = explanation
  157. class gpsjson(object):
  158. """Basic JSON decoding."""
  159. def __init__(self):
  160. """Init gpsjson."""
  161. self.data = None
  162. self.stream_command = None
  163. self.enqueued = None
  164. self.verbose = -1
  165. def __iter__(self):
  166. """Broken __iter__."""
  167. return self
  168. def unpack(self, buf):
  169. """Unpack a JSON string."""
  170. try:
  171. # json.loads(,encoding=) deprecated Python 3.1. Gone in 3.9
  172. # like it or not, data is now UTF-8
  173. self.data = dictwrapper(json.loads(buf.strip()))
  174. except ValueError as e:
  175. raise json_error(buf, e.args[0])
  176. # Should be done for any other array-valued subobjects, too.
  177. # This particular logic can fire on SKY or RTCM2 objects.
  178. if hasattr(self.data, "satellites"):
  179. self.data.satellites = [dictwrapper(x)
  180. for x in self.data.satellites]
  181. def stream(self, flags=0, devpath=None):
  182. """Control streaming reports from the daemon,"""
  183. if 0 < flags:
  184. self.stream_command = self.generate_stream_command(flags, devpath)
  185. else:
  186. self.stream_command = self.enqueued
  187. if self.stream_command:
  188. if self.verbose > 1:
  189. sys.stderr.write("send: stream as:"
  190. " {}\n".format(self.stream_command))
  191. self.send(self.stream_command)
  192. else:
  193. raise TypeError("Invalid streaming command!! : " + str(flags))
  194. def generate_stream_command(self, flags=0, devpath=None):
  195. """Generate stream command."""
  196. if flags & WATCH_OLDSTYLE:
  197. return self.generate_stream_command_old_style(flags)
  198. return self.generate_stream_command_new_style(flags, devpath)
  199. @staticmethod
  200. def generate_stream_command_old_style(flags=0):
  201. """Generate stream command, old style."""
  202. if flags & WATCH_DISABLE:
  203. arg = "w-"
  204. if flags & WATCH_NMEA:
  205. arg += 'r-'
  206. elif flags & WATCH_ENABLE:
  207. arg = 'w+'
  208. if flags & WATCH_NMEA:
  209. arg += 'r+'
  210. return arg
  211. @staticmethod
  212. def generate_stream_command_new_style(flags=0, devpath=None):
  213. """Generate stream command, new style."""
  214. if (flags & (WATCH_JSON | WATCH_OLDSTYLE | WATCH_NMEA |
  215. WATCH_RAW)) == 0:
  216. flags |= WATCH_JSON
  217. if flags & WATCH_DISABLE:
  218. arg = '?WATCH={"enable":false'
  219. if flags & WATCH_JSON:
  220. arg += ',"json":false'
  221. if flags & WATCH_NMEA:
  222. arg += ',"nmea":false'
  223. if flags & WATCH_RARE:
  224. arg += ',"raw":1'
  225. if flags & WATCH_RAW:
  226. arg += ',"raw":2'
  227. if flags & WATCH_SCALED:
  228. arg += ',"scaled":false'
  229. if flags & WATCH_TIMING:
  230. arg += ',"timing":false'
  231. if flags & WATCH_SPLIT24:
  232. arg += ',"split24":false'
  233. if flags & WATCH_PPS:
  234. arg += ',"pps":false'
  235. else: # flags & WATCH_ENABLE:
  236. arg = '?WATCH={"enable":true'
  237. if flags & WATCH_JSON:
  238. arg += ',"json":true'
  239. if flags & WATCH_NMEA:
  240. arg += ',"nmea":true'
  241. if flags & WATCH_RARE:
  242. arg += ',"raw":1'
  243. if flags & WATCH_RAW:
  244. arg += ',"raw":2'
  245. if flags & WATCH_SCALED:
  246. arg += ',"scaled":true'
  247. if flags & WATCH_TIMING:
  248. arg += ',"timing":true'
  249. if flags & WATCH_SPLIT24:
  250. arg += ',"split24":true'
  251. if flags & WATCH_PPS:
  252. arg += ',"pps":true'
  253. if flags & WATCH_DEVICE:
  254. arg += ',"device":"%s"' % devpath
  255. arg += "}"
  256. return arg
  257. class dictwrapper(object):
  258. """Wrapper that yields both class and dictionary behavior,"""
  259. def __init__(self, ddict):
  260. """Init class dictwrapper."""
  261. self.__dict__ = ddict
  262. def get(self, k, d=None):
  263. """Get dictwrapper."""
  264. return self.__dict__.get(k, d)
  265. def keys(self):
  266. """Keys dictwrapper."""
  267. return self.__dict__.keys()
  268. def __getitem__(self, key):
  269. """Emulate dictionary, for new-style interface."""
  270. return self.__dict__[key]
  271. def __iter__(self):
  272. """Iterate dictwrapper."""
  273. return self.__dict__.__iter__()
  274. def __setitem__(self, key, val):
  275. """Emulate dictionary, for new-style interface."""
  276. self.__dict__[key] = val
  277. def __contains__(self, key):
  278. """Find key in dictwrapper."""
  279. return key in self.__dict__
  280. def __str__(self):
  281. """dictwrapper to string."""
  282. return "<dictwrapper: " + str(self.__dict__) + ">"
  283. __repr__ = __str__
  284. def __len__(self):
  285. """length of dictwrapper."""
  286. return len(self.__dict__)
  287. #
  288. # Someday a cleaner Python interface using this machinery will live here
  289. #
  290. # End
  291. # vim: set expandtab shiftwidth=4