2018-06-23 01:00:43 +02:00
|
|
|
"""
|
|
|
|
.. moduleauthor:: Fabio Manganiello <blacklight86@gmail.com>
|
|
|
|
"""
|
|
|
|
|
2019-12-20 18:27:47 +01:00
|
|
|
from typing import Any, Optional, Dict, Union
|
|
|
|
|
2018-07-06 02:08:38 +02:00
|
|
|
from platypush.plugins import Plugin, action
|
2018-03-24 03:05:46 +01:00
|
|
|
|
|
|
|
|
|
|
|
class GpioPlugin(Plugin):
|
2018-06-23 01:00:43 +02:00
|
|
|
"""
|
|
|
|
Plugin to handle raw read/write operation on the Raspberry Pi GPIO pins.
|
|
|
|
|
|
|
|
Requires:
|
|
|
|
* **RPi.GPIO** (`pip install RPi.GPIO`)
|
|
|
|
"""
|
|
|
|
|
2019-12-20 18:27:47 +01:00
|
|
|
def __init__(self, pins: Optional[Dict[str, int]] = None, mode: str = 'board', **kwargs):
|
2018-08-22 19:09:17 +02:00
|
|
|
"""
|
2019-12-20 18:27:47 +01:00
|
|
|
:param mode: Specify 'board' if you want to use the board PIN numbers,
|
|
|
|
'bcm' for Broadcom PIN numbers (default: 'board')
|
2018-08-22 19:12:26 +02:00
|
|
|
:param pins: Configuration for the GPIO PINs as a name -> pin_number map.
|
2018-08-22 19:09:17 +02:00
|
|
|
|
|
|
|
Example::
|
|
|
|
|
|
|
|
{
|
2018-08-22 19:12:26 +02:00
|
|
|
"LED_1": 14,
|
|
|
|
"LED_2": 15,
|
|
|
|
"MOTOR": 16,
|
|
|
|
"SENSOR": 17
|
2018-08-22 19:09:17 +02:00
|
|
|
}
|
2019-12-20 18:27:47 +01:00
|
|
|
|
2018-08-22 19:09:17 +02:00
|
|
|
"""
|
|
|
|
|
2019-07-02 14:04:25 +02:00
|
|
|
super().__init__(**kwargs)
|
2019-12-20 18:27:47 +01:00
|
|
|
self.mode = self._get_mode(mode)
|
2018-08-22 19:12:26 +02:00
|
|
|
self.pins_by_name = pins if pins else {}
|
2019-07-02 14:04:25 +02:00
|
|
|
self.pins_by_number = {number: name
|
|
|
|
for (name, number) in self.pins_by_name.items()}
|
2018-08-22 19:09:17 +02:00
|
|
|
|
|
|
|
def _get_pin_number(self, pin):
|
|
|
|
try:
|
|
|
|
pin = int(str(pin))
|
|
|
|
except ValueError:
|
|
|
|
pin = self.pins_by_name.get(pin)
|
|
|
|
if not pin:
|
|
|
|
raise RuntimeError('Unknown PIN name: "{}"'.format(pin))
|
|
|
|
|
|
|
|
return pin
|
|
|
|
|
2019-12-20 18:27:47 +01:00
|
|
|
@staticmethod
|
|
|
|
def _get_mode(mode_str: str) -> int:
|
|
|
|
import RPi.GPIO as gpio
|
|
|
|
|
|
|
|
mode_str = mode_str.upper()
|
|
|
|
assert mode_str in ['BOARD', 'BCM'], 'Invalid mode: {}'.format(mode_str)
|
|
|
|
return getattr(gpio, mode_str)
|
|
|
|
|
2018-07-06 02:08:38 +02:00
|
|
|
@action
|
2019-12-20 18:56:00 +01:00
|
|
|
def write(self, pin: Union[int, str], value: Union[int, bool],
|
2019-12-20 18:27:47 +01:00
|
|
|
name: Optional[str] = None, mode: Optional[str] = None) -> Dict[str, Any]:
|
2018-06-23 01:00:43 +02:00
|
|
|
"""
|
|
|
|
Write a byte value to a pin.
|
|
|
|
|
2018-08-22 19:09:17 +02:00
|
|
|
:param pin: PIN number or configured name
|
2019-07-02 14:04:25 +02:00
|
|
|
:param name: Optional name for the written value (e.g. "temperature" or "humidity")
|
|
|
|
:param value: Value to write
|
2019-12-20 18:27:47 +01:00
|
|
|
:param mode: If a PIN number is specified then you can override the default 'mode'
|
|
|
|
default parameter
|
2018-06-23 01:00:43 +02:00
|
|
|
|
|
|
|
Response::
|
|
|
|
|
|
|
|
output = {
|
2019-07-02 14:04:25 +02:00
|
|
|
"name": <pin or metric name>,
|
2018-06-23 01:00:43 +02:00
|
|
|
"pin": <pin>,
|
2019-07-02 14:04:25 +02:00
|
|
|
"value": <value>,
|
2018-06-23 01:00:43 +02:00
|
|
|
"method": "write"
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
|
|
|
|
import RPi.GPIO as gpio
|
|
|
|
|
2019-07-02 14:04:25 +02:00
|
|
|
name = name or pin
|
2018-08-22 19:09:17 +02:00
|
|
|
pin = self._get_pin_number(pin)
|
2019-12-20 18:27:47 +01:00
|
|
|
mode = self._get_mode(mode) if mode else self.mode
|
|
|
|
gpio.setmode(mode)
|
2018-03-24 03:05:46 +01:00
|
|
|
gpio.setup(pin, gpio.OUT)
|
2019-07-02 14:04:25 +02:00
|
|
|
gpio.output(pin, value)
|
2018-03-24 03:05:46 +01:00
|
|
|
|
2018-07-06 02:08:38 +02:00
|
|
|
return {
|
2019-07-02 14:04:25 +02:00
|
|
|
'name': name,
|
2018-03-24 03:05:46 +01:00
|
|
|
'pin': pin,
|
2019-07-02 14:04:25 +02:00
|
|
|
'value': value,
|
2018-03-24 03:05:46 +01:00
|
|
|
'method': 'write',
|
2018-07-06 02:08:38 +02:00
|
|
|
}
|
2018-03-24 03:05:46 +01:00
|
|
|
|
2018-07-06 02:08:38 +02:00
|
|
|
@action
|
2019-12-20 18:27:47 +01:00
|
|
|
def read(self, pin: Union[int, str], name: Optional[str] = None,
|
|
|
|
mode: Optional[str] = None) -> Dict[str, Any]:
|
2018-06-23 01:00:43 +02:00
|
|
|
"""
|
|
|
|
Reads a value from a PIN.
|
|
|
|
|
2018-08-22 19:09:17 +02:00
|
|
|
:param pin: PIN number or configured name.
|
2019-07-02 14:04:25 +02:00
|
|
|
:param name: Optional name for the read value (e.g. "temperature" or "humidity")
|
2019-12-20 18:27:47 +01:00
|
|
|
:param mode: If a PIN number is specified then you can override the default 'mode'
|
|
|
|
default parameter
|
2018-06-23 01:00:43 +02:00
|
|
|
|
|
|
|
Response::
|
|
|
|
|
|
|
|
output = {
|
2019-07-02 14:04:25 +02:00
|
|
|
"name": <pin number or pin/metric name>,
|
2018-06-23 01:00:43 +02:00
|
|
|
"pin": <pin>,
|
2019-07-02 14:04:25 +02:00
|
|
|
"value": <value>,
|
2018-06-23 01:00:43 +02:00
|
|
|
"method": "read"
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
|
|
|
|
import RPi.GPIO as gpio
|
|
|
|
|
2019-07-02 14:04:25 +02:00
|
|
|
name = name or pin
|
2018-08-22 19:09:17 +02:00
|
|
|
pin = self._get_pin_number(pin)
|
2018-03-24 03:05:46 +01:00
|
|
|
gpio.setmode(gpio.BCM)
|
|
|
|
gpio.setup(pin, gpio.IN)
|
|
|
|
val = gpio.input(pin)
|
|
|
|
|
2018-07-06 02:08:38 +02:00
|
|
|
return {
|
2019-07-02 14:04:25 +02:00
|
|
|
'name': name,
|
2018-03-24 03:05:46 +01:00
|
|
|
'pin': pin,
|
2019-07-02 14:04:25 +02:00
|
|
|
'value': val,
|
2018-03-24 03:05:46 +01:00
|
|
|
'method': 'read',
|
2018-07-06 02:08:38 +02:00
|
|
|
}
|
2018-03-24 03:05:46 +01:00
|
|
|
|
2019-07-02 14:04:25 +02:00
|
|
|
@action
|
|
|
|
def get_measurement(self, pin=None):
|
|
|
|
if pin is None:
|
|
|
|
return self.read_all()
|
|
|
|
return self.read(pin)
|
|
|
|
|
2018-08-22 19:09:17 +02:00
|
|
|
@action
|
|
|
|
def read_all(self):
|
|
|
|
"""
|
2019-07-02 14:04:25 +02:00
|
|
|
Reads the values from all the configured PINs and returns them as a list. It will raise a RuntimeError if no
|
|
|
|
PIN mappings were configured.
|
2018-08-22 19:09:17 +02:00
|
|
|
"""
|
|
|
|
|
|
|
|
if not self.pins_by_number:
|
|
|
|
raise RuntimeError("No PIN mappings were provided/configured")
|
|
|
|
|
|
|
|
values = []
|
2018-08-22 19:20:24 +02:00
|
|
|
for (pin, name) in self.pins_by_number.items():
|
2019-07-02 14:04:25 +02:00
|
|
|
values.append(self.read(pin=pin, name=name).output)
|
2018-08-22 19:09:17 +02:00
|
|
|
|
|
|
|
return values
|
|
|
|
|
2018-03-24 03:05:46 +01:00
|
|
|
|
|
|
|
# vim:sw=4:ts=4:et:
|