2
votes
***Python code:***
import serial
import pandas as pd
import time
import re
import xlrd
from msvcrt import getch
import numpy as np
i = 0
x = 0
y = 0

df = pd.read_excel(r'C:\Users\lynchfamily\Desktop\mlglovesdata.xls')
# Read COM9
# Read from COM10 as well
# Readline() only works with a timeout (IMPORTANT)
serHC = serial.Serial('COM9', 115200,timeout=.250,parity=serial.PARITY_NONE,rtscts=1) # This is the JY
serRN = serial.Serial('COM10', 115200,timeout=.250,parity=serial.PARITY_NONE,rtscts=1) # This is the silvermate


def serialin():
    # Sensor lists
    sensor_names = list()
    sensor_values = list()
    global i
    # Read a certain amount of bytes from serial and then continue
    # Regular expressions for finding the proper data
    while i < 6:
        # print(i) for debugging
        global serHC
        global serRN
        #searchObj = re.search(r'(A\d?\d*)?(\d*)?',serHC.read(4).decode(),re.I)
        #searchObjRN = re.search(r'(A\d?\d*)?(\d*)?',serRN.read(4).decode(),re.I)
        # Serial data stops while in loop
        # The if statements keep the false values out of the program
        #if searchObj.group(1):
        sensor_names.append(serHC.read(2))
        #if searchObj.group(2):
        sensor_values.append(serHC.read(2))
        #if searchObjRN.group(1):
        sensor_names.append(serRN.read(2))
        #if searchObjRN.group(2):
        sensor_values.append(serRN.read(2))
        i = i + 1
    while 1:
        # Get the key from the msvcrt module
        key = getch().decode('ASCII')

        # If key is pressed, do something
        if key:
            print(key)
            # Zip them together
            # Final 2D list
            final_2d_list = zip(sensor_names,sensor_values)
            print(list(sorted(final_2d_list)))
            #vals = df.Dataframe([
            #df.append(vals)
            #print(sorted_array_1stdim[r])
            #sensor_values = [0] * 10
            # Thread for reading definition
            break
            # Fancy recursion
    sensor_values.clear()
    sensor_names.clear()
    i = 0
    serialin()
serialin()

Arduino Code:

// The device with green colored wires
void setup() {

  Serial.begin(115200);

}

void loop() {
   // It won't work with the I2C while loop for some reason. Perhaps it is getting stuck up on it

   Serial.print("A4");
   Serial.print(analogRead(0)); // Read the local analog signal
   delay(5);
   Serial.print("A5");
   Serial.print(analogRead(1)); // Read the local analog signal
   delay(5);
   Serial.print("A6");
   Serial.print(analogRead(2)); // Read the local analog signal
   delay(5);
   Serial.print("A7");
   Serial.print(analogRead(3)); // Read the local analog signal


}

I'm trying to send analog data from sensors over through bluetooth silver mate from sparkfun, and HC-06 modules to python.
I have to read the analog data at a delay of 5 seconds between each, so that the readings aren't conflicted.
The data comes through serial ports COM9 and COM10. I know that serial in python can be blocking, that's why I attempted to read it first, and then put it in a list.
I also knows that once serial has been read through, it appears to be non-blocking. When I was using serHC.readline() and serRN.readline(), I was getting something like what I'd expect to see.
However, the data in the list were not updating according to the change in the sensors. I have to admit python is not my main programming language, so that is why I'm asking for help.
I thought maybe using multiple threads might work, but I wasn't able to get the serHC and serRN variables in the main thread.

Any help will be appreciated!!

1

1 Answers

1
votes

As you have discovered it is not possible to read sequentially from serial ports: a blocking read over one port implies a loss of data simultaneous sent over the other port.

Use a thread based approach.

The following sketch should be enough to get started:

import serial
import time
import re
import threading

BYTES_TO_READ = 6

# read from serial port
def read_from_serial(board, port):
    print("reading from {}: port {}".format(board, port))
    payload = b''
    ser = serial.Serial(port, 115200,timeout=.250, parity=serial.PARITY_NONE, rtscts=1)
    bytes_count = 0
    while bytes_count < BYTES_TO_READ:
        read_bytes = ser.read(2)

        # sum number of bytes returned (not 2), you have set the timeout on serial port
        # see https://pythonhosted.org/pyserial/pyserial_api.html#serial.Serial.read
        bytes_count = bytes_count + len(read_bytes)
        payload = payload + read_bytes

    # here you have the bytes, do your logic
    # ...
    print("READ from {}: [{}]".format(board, payload))
    return


def main():

    board = {
        'JY': 'COM9',
        'SILVER': 'COM10'
    }

    threads = []
    for b in board:

        t = threading.Thread(target=read_from_serial, args=(b, board[b],))
        threads.append(t)
        t.start()

    # wait for all threads termination
    for t in threads:
        t.join()

main()

For learning about threading: https://pymotw.com/3/threading/

Periodic read from serials

Below a sketch for reading each TIME_PERIOD seconds. A parte the infinite while loop around the read there is a "thread" loop with a nested try/catch block for catching serials communication problems and retrying to connect after TIME_PERIOD.

Take it just as a starting example!

import serial
import time
import re
import threading

BYTES_TO_READ = 6

TIME_PERIOD = 5

def read_message(board, port, handle):
    payload = b''
    bytes_count = 0
    while bytes_count < BYTES_TO_READ:
        read_bytes = handle.read(2)
        bytes_count = bytes_count + len(read_bytes)
        payload = payload + read_bytes
    # here you have the bytes, do your logic
    # ...
    print("READ from {}: [{}]".format(board, payload))

def serial_thread(board, port):
    print("reading from {}: port {}".format(board, port))

    while True:
        try:
            handle = serial.Serial(port, 115200,timeout=.250, parity=serial.PARITY_NONE, rtscts=1)

            while True:
                read_message(board, port, handle)
                time.sleep(TIME_PERIOD)
        except Exception as e:
            print("ERROR: {}".format(e))
            print("retrying in {} seconds".format(TIME_PERIOD))
            handle.close()
            time.sleep(TIME_PERIOD)

def main():
    board = {
        'JY': '/dev/ttyUSB0',
        'SILVER': '/dev/ttyACM0'
    }
    threads = []
    for b in board:
        t = threading.Thread(target=serial_thread, args=(b, board[b],))
        threads.append(t)
        t.start()
    # wait for all threads termination
    for t in threads:
        t.join()

main()