__init__.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  1. #!/usr/bin/env python3.5
  2. # Copyright 2017 Digital
  3. #
  4. # This file is part of DigiLib.
  5. #
  6. # DigiLib is free software: you can redistribute it and/or modify
  7. # it under the terms of the GNU General Public License as published by
  8. # the Free Software Foundation, either version 3 of the License, or
  9. # (at your option) any later version.
  10. #
  11. # DigiLib is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # You should have received a copy of the GNU General Public License
  17. # along with DigiLib. If not, see <http://www.gnu.org/licenses/>.
  18. # Python modules
  19. import logging
  20. import threading
  21. import traceback
  22. # Third party modules
  23. import blinker
  24. import curio
  25. import digilib.network
  26. log = logging.getLogger(__name__+"")
  27. lgpio = logging.getLogger(__name__+".gpio")
  28. # Error messages
  29. ERROR_TAKES_ARGUMENTS = "{} takes {} {} argument(s): {}"
  30. # respond(ERROR_TAKES_ARGUMENTS.format(
  31. # command, "one", "positional", "<name>"))
  32. _pins_for_cleanup = set()
  33. _gpio = None
  34. def _make_gpio_warpper():
  35. _gpio = GPIOWrapper()
  36. # class FakeGPIO(object):
  37. # OUT = "out"
  38. # IN = "in"
  39. # BCM = "broadcom"
  40. # pin_values = {}
  41. # def cleanup(self,*args):
  42. # lgpio.debug("cleanup with these args '{}'!".format(args))
  43. # def setup(self,pins,value):
  44. # lgpio.debug("pin(s) {} set to {}".format(pins,value))
  45. # def setmode(self,mode):
  46. # lgpio.debug("mode set to {}".format(mode))
  47. # def output(self,pins,value):
  48. # lgpio.debug("setting pin(s) {} to value {}".format(
  49. # pins, value
  50. # ))
  51. # if type(pins) == int:
  52. # pins = [pins]
  53. # for p in pins:
  54. # self.pin_values[p] = value
  55. # def read(self,pin):
  56. # return self.pin_values[pin]
  57. #
  58. # try:
  59. # import RPi.GPIO as gpio
  60. # except:
  61. # # print("Using FakeGPIO because RPi.GPIO was not found")
  62. # gpio = FakeGPIO()
  63. #
  64. # def call_gpio_cleanup(*args):
  65. # log.info(args)
  66. # gpio.cleanup(_pins_for_cleanup)
  67. class GPIOWrapper(object):
  68. gpio = None
  69. OUT = "out"
  70. IN = "in"
  71. # BCM = "broadcom"
  72. pin_values = {}
  73. def __init__(self):
  74. super(GPIOWrapper,self).__init__()
  75. self.load_rpi_gpio()
  76. lgpio.debug("setting pin numbering to broadcom")
  77. if self.gpio:
  78. self.gpio.setmode(self.gpio.BCM)
  79. sig_exit = blinker.signal("global-exit")
  80. sig_exit.connect(self.cleanup)
  81. def cleanup(self,*args):
  82. lgpio.debug("cleanup! ({})".format(args))
  83. if self.gpio:
  84. self.gpio.cleanup(_pins_for_cleanup)
  85. def load_rpi_gpio(self):
  86. try:
  87. self.gpio = False
  88. self.gpio = __import__("RPi.GPIO",fromlist="GPIO")
  89. except ImportError as e:
  90. lgpio.debug("failed to import RPi.GPIO")
  91. print("134")
  92. except Exception as e:
  93. lgpio.debug("unknown error occured", exc_info=e)
  94. print("137")
  95. finally:
  96. self.OUT = getattr(self.gpio,"OUT",self.OUT)
  97. self.IN = getattr(self.gpio,"IN",self.IN)
  98. # self.BCM = getattr(self.gpio,self.BCM)
  99. def output(self,pins,state,*args):
  100. lgpio.debug("setting pin(s) {:0>2} to value {}".format(
  101. pins,state))
  102. if type(pins) is int:
  103. pins = [pins]
  104. _pins_for_cleanup.update(pins)
  105. if self.gpio:
  106. self.gpio.output(pins,state)
  107. else:
  108. lgpio.debug("no gpio module")
  109. lgpio.debug("gpio module: {}".format(self.gpio))
  110. def parse_to_int_list(self,parse,*args):
  111. if type(parse) is list:
  112. return parse
  113. elif type(parse) is int:
  114. return [parse]
  115. elif type(parse) is bool:
  116. return [int(parse)]
  117. elif type(parse) is not str:
  118. raise ValueError("Failed to parse value {}".format(parse))
  119. if parse.lower() in ["true","high"]:
  120. return [1]
  121. elif parse.lower() in ["false","low"]:
  122. return [0]
  123. parse = parse.split(".")
  124. for part in parse.copy():
  125. parse.remove(part)
  126. parse.append(int(part))
  127. return parse
  128. def read(self,pin,*args):
  129. if self.gpio:
  130. state = self.gpio.read(pin)
  131. else:
  132. state = -1
  133. lgpio.debug("reading pin {:0>2}: {}".format(
  134. pin,state))
  135. return state
  136. def setup(self,pins,value,*args):
  137. lgpio.debug("setting pin(s) {} to {}".format(pins,value))
  138. if self.gpio:
  139. self.gpio.setup(pins,value)
  140. class PinBase(object):
  141. """PinBase is the base class for all classes representing a gpio pin"""
  142. pin_number = None
  143. value = None
  144. def __init__(self,pin_number,mode):
  145. super(PinBase,self).__init__()
  146. if _gpio == None:
  147. _make_gpio_warpper()
  148. self.pin_number = pin_number
  149. self.value = self.value_low
  150. _gpio.setup(self.pin_number,_gpio.OUT)
  151. def output(self,value):
  152. [value] = gpioself.parse_to_int_list(value)
  153. self.value = value
  154. _gpiooutput(self.pin_number,value)
  155. def read(self):
  156. value = _gpioread(self.pin_number,value)
  157. return value
  158. class DigitalPin(PinBase):
  159. value_high = True
  160. value_low = False
  161. def __init__(self,pin_number,mode):
  162. super(DigitalPin,self).__init__(pin_number,mode)
  163. class AnalogPin(PinBase):
  164. value_high = 1
  165. value_low = 0
  166. def __init__(self,pin_number):
  167. super(AnalogPin,self).__init__(pin_number)
  168. class PinControllerBase(object):
  169. """docstring for PinControllerBase.
  170. PinControllerBase is the base class for all classes controlling one or more physical devices connected to a gpio header
  171. """
  172. pins = []
  173. def __init__(self,pins):
  174. super(PinControllerBase, self).__init__()
  175. self.pins.extend(pins)
  176. # def make_digital_pin(self,*args):
  177. # return DigitalPin(*args)
  178. # def make_analog_pin(self,*args):
  179. # return AnalogPin(*args)
  180. class PinAPIBase(object):
  181. """docstring for PinAPI.
  182. PinAPIBase is the base class for all classes providing an api to multiple
  183. PinController.
  184. """
  185. controllers = []
  186. def __init__(self):
  187. super(PinAPIBase, self).__init__()
  188. class PCEngine(PinControllerBase):
  189. """Test Class"""
  190. max_speed=1
  191. speed = 0
  192. turn_on_speed = 1
  193. is_on = False
  194. def __init__(self,pin_on_off,pin_analog):
  195. super(PCEngine, self).__init__([pin_on_off,pin_analog])
  196. # self.pins.append(pin_on_off)
  197. # self.pins.append(pin_analog)
  198. self.pin_on_off = DigitalPin(pin_on_off,_gpio.OUT)
  199. self.pin_analog = DigitalPin(pin_analog,_gpio.OUT)
  200. # self.pins.append(self.pin_on_off)
  201. # self.pins.append(self.pin_analog)
  202. # gpio.setup(self.pins,_gpio.OUT)
  203. # gpio.setup(self.pin_analog,_gpio.OUT)
  204. self.set_speed(1)
  205. self.set_state(1)
  206. def set_speed(self,speed):
  207. self.pin_analog.output(speed)
  208. self.speed = speed
  209. def set_state(self,state):
  210. """state is integer and specifies if the engine should be turned on (1) or turned off (0) """
  211. if state == self.is_on:
  212. return
  213. # self.set_speed(speed)
  214. self.pin_on_off.output(state)
  215. self.is_on = state
  216. class EnginesController(PinAPIBase):
  217. engine_left = None
  218. engine_right = None
  219. action_in_process = False
  220. def __init__(self,left,right):
  221. super(EnginesController,self).__init__()
  222. self.engine_right = self.make_engine(*right)
  223. self.engine_left = self.make_engine(*left)
  224. def __enter__(self):
  225. if self.action_in_process:
  226. lgpio.debug("action already in process, not entering")
  227. return False
  228. def make_engine(self,*args):
  229. return digilib.pin.PCEngine(*args)
  230. def set_engine_state(self,args=[],command=None,respond=None):
  231. if len(args) != 2:
  232. respond("missing argument(s): <engine> <state>")
  233. return
  234. [engine,state] = args
  235. if engine == "right":
  236. self.engine_right.set_state(state)
  237. elif engine == "left":
  238. self.engine_left.set_state(state)
  239. else:
  240. respond("unknown engine. use 'left' or 'right'")
  241. def set_engine_speed(self,args=[],command=None,respond=None):
  242. if len(args) != 2:
  243. respond("missing argument(s): <engine> <speed>")
  244. return
  245. [engine,speed] = args
  246. if engine == "right":
  247. self.engine_right.set_speed(speed)
  248. elif engine == "left":
  249. self.engine_left.set_speed(speed)
  250. else:
  251. respond("unknown engine. use 'left' or 'right'")
  252. async def turn(self,args=[],command=None,respond=None):
  253. if len(args) != 1:
  254. respond("one missing argument: direction")
  255. return
  256. [direction] = args
  257. # lgpio.debug(threading.current_thread())
  258. # with self ad
  259. lgpio.info("turning {}".format(direction))
  260. right_state = self.engine_right.is_on
  261. right_speed = self.engine_right.speed
  262. left_state = self.engine_left.is_on
  263. left_speed = self.engine_left.speed
  264. if direction == "right":
  265. self.engine_right.set_state(False)
  266. self.engine_left.set_state(True)
  267. self.engine_left.set_speed(1)
  268. elif direction == "left":
  269. self.engine_right.set_state(True)
  270. self.engine_right.set_speed(1)
  271. self.engine_left.set_state(False)
  272. await curio.sleep(2)
  273. # time.sleep(2)
  274. self.engine_right.set_state(right_state)
  275. self.engine_right.set_speed(right_speed)
  276. self.engine_left.set_state(left_state)
  277. self.engine_left.set_speed(left_speed)
  278. lgpio.info("done turning {}".format(direction))
  279. # set engines to previous values
  280. class LED(DigitalPin):
  281. def __init__(self,pin):
  282. super(DigitalPin,self).__init__(pin,_gpio.OUT)
  283. self.on()
  284. def on(self,args=[],command=None,respond=None):
  285. self.output(True)
  286. def off(self,args=[],command=None,respond=None):
  287. self.output(False)
  288. def set(self,args=[],command=None,respond=None):
  289. if len(args) != 1:
  290. respond("one missing argument: state")
  291. return
  292. [state] = args
  293. self.output(state)
  294. class LEDRGB(PinControllerBase):
  295. def __init__(self,pin_red,pin_green):
  296. super(LEDRGB,self).__init__([pin_red,pin_green])
  297. self.pin_red = DigitalPin(pin_red,_gpio.OUT)
  298. self.pin_green = DigitalPin(pin_green,_gpio.OUT)
  299. self.red()
  300. def red(self,args=[],command=None,respond=None):
  301. if len(args) > 1:
  302. respond("{} takes one optinal agument: <state>".foramt(command))
  303. return
  304. elif len(args) == 1:
  305. state = int(*args)
  306. else:
  307. state = 1
  308. self.pin_red.output(state)
  309. def green(self,args=[],command=None,respond=None):
  310. if len(args) > 1:
  311. respond("{} takes one optinal agument: <state>".foramt(command))
  312. return
  313. elif len(args) == 1:
  314. state = int(*args)
  315. else:
  316. state = 1
  317. self.pin_green.output(state)
  318. class DebugPinController(PinControllerBase):
  319. def output(self,args=[],command=None,respond=None):
  320. if len(args) != 2:
  321. respond(ERROR_TAKES_ARGUMENTS.format(
  322. command, "two", "positional", "<name>"))
  323. return False
  324. pins = _gpioparse_to_int_list(args[0])
  325. [state] = _gpioparse_to_int_list(args[1])
  326. gpio.output(pins,state)
  327. def read(self,args=[],command=None,respond=None):
  328. if len(args) != 2:
  329. respond(ERROR_TAKES_ARGUMENTS.format(
  330. command, "two", "positional", "<name>"))
  331. return False
  332. pins = _gpioparse_to_int_list(args[0])
  333. [state] = _gpioparse_to_int_list(args[1])
  334. return _gpioread(pins,state)
  335. if __name__ == "__main__":
  336. pass
  337. #