forked from SolentAirWatch/sniffy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpmsX003.py
114 lines (103 loc) · 3.86 KB
/
pmsX003.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
import bme280 # driver for Bosche BME280 - local file
import serial # pyserial for Plantower PMS5003
import paho.mqtt.client as mqtt # MQTT stream to opensesors
import time # for sleep function
import datetime # for timestampping data
import json
import csv
from pprint import pprint # makes data more pretty
monitorID = '10' # id 0 is reserved for test
monitorLocation = [50.9262, -1.4092]
port = serial.Serial('/dev/ttyS0', baudrate=9600, timeout=2.0)
broker = "" #"mqtt.opensensors.io" # "46.101.13.195" # test broker
topic = "mqtt-data" #"/orgs/solentairwatch/sniffy"
csvFile = "PMSx003.csv" # keep a local copy for debug
def on_connect(client, userdata, flags, rc):
print("Connected with result code " + str(rc))
# do nothing we're connected
pass
def on_publish(client, userdata, mid):
print(mid)
# print('published a message')
pass
def on_disconnect(client, userdata, rc):
if rc !=0:
print('unexpected disconect with code' + str(rc))
client.reconnect()
# function to parse a line of binary data from PMSx003 -- needs updating for python 3
def read_pm_line(_port):
rv = b''
while True:
ch1 = _port.read()
if ch1 == b'\x42':
ch2 = _port.read()
if ch2 == b'\x4d':
rv += ch1 + ch2
rv += _port.read(28)
return rv
# define a function to read env data from BME280 over i2c
def readBME():
temp, pres, humid = bme280.readBME280All()
temp = round(float(temp),1)
pres = round(float(pres),1)
humid = round(float(humid),1)
return (temp, pres, humid)
# set up objects
f = open(csvFile,'a') # open the csv file but 'a'ppend it if it already exists
# Authenticate with opensensors.io
client = mqtt.Client(client_id=clientID)
client.username_pw_set("mqtt-user", password="mqtt-password" )
# set up callbacks
client.on_connect = on_connect
client.on_publish = on_publish
client.connect(broker) # (address, port, timeout (sec) )
# write a blank message to CSV so we can add headers at the start of the file
message = {
'time': "init",
'id': monitorID,
'cityName': "",
'stationName': "",
'averaging': 0,
'latitude': monitorLocation[0],
'longitude': monitorLocation[1],
'PM1': 0,
'PM25': 0,
'PM10': 0}
w = csv.DictWriter(f, message.keys())
w.writeheader()
client.loop_start() # start the network loop
while True: # PMSx003 sensor by default streams data and non-uniform intervals - replace with timed polling
try:
rcv = read_pm_line(port)
temp, pres, humid = readBME()
message = {
'time': str(datetime.datetime.now()),
'id': monitorID,
'cityName': "Southampton",
'stationName': "Common#1",
'latitude': monitorLocation[0],
'longitude': monitorLocation[1],
'averaging': 0,
'PM1': ord(rcv[4]) * 256 + ord(rcv[5]),
'PM25': ord(rcv[6]) * 256 + ord(rcv[7]),
'PM10': ord(rcv[8]) * 256 + ord(rcv[9]),
'tempreature': temp,
'pressure': pres,
'humidity': humid,
#'PM10_STD': ord(rcv[10]) * 256 + ord(rcv[11]),
#'PM25_STD': ord(rcv[12]) * 256 + ord(rcv[13]),
#'PM100_STD': ord(rcv[14]) * 256 + ord(rcv[15]),
#'gr03um': ord(rcv[16]) * 256 + ord(rcv[17]),
#'gt05um': ord(rcv[18]) * 256 + ord(rcv[19]),
#'gr10um': ord(rcv[20]) * 256 + ord(rcv[21]),
#'gr25um': ord(rcv[22]) * 256 + ord(rcv[23]),
#'gr50um': ord(rcv[24]) * 256 + ord(rcv[25]),
#'gr100um': ord(rcv[26]) * 256 + ord(rcv[27])
}
pprint(message)
client.publish(topic, payload=json.dumps(message), qos=0, retain=False)
w = csv.DictWriter(f, message.keys())
w.writerow(message)
time.sleep(0.1) # wait 100 millisonds
except KeyboardInterrupt:
break