Python Polling Script using minimalmodbus
This was replaced with a more comprehensive C program. See: Home Solar Project
#!/usr/bin/env python3 # # powermeter.py # # I don't like python very much # import minimalmodbus import serial import time from datetime import datetime import csv import psycopg2 import sys #---------------------------------------------------------------------- # ADD_TABLE_ENTRY #---------------------------------------------------------------------- def add_table_entry(sensor_id,stamp,volts,amps,watts,energy,frequency,power_factor,alarm): sql = "INSERT INTO power_monitor\n" sql += " ( pwm_sensor, pwm_stamp, pwm_volts, pwm_amps, pwm_watts, pwm_energy, pwm_frequency, pwm_power_factor, pwm_alarm\n" sql += " ) VALUES (\n" sql += " {},'{}',{},{},{},{},{},{},{}".format(sensor_id, stamp, volts, amps, watts, energy, frequency, power_factor, alarm) sql += " )" if(verbose > 0): print("--") print(sql) print("--") cursor.execute(sql) connection.commit() #---------------------------------------------------------------------- # CSV_ADD_ENTRY #---------------------------------------------------------------------- def csv_add_entry(csv_log,csv_out,sensor_id,stamp,amps,watts,volts,frequency): data = [ sensor_id, stamp, amps, watts, volts, frequency ] csv_out.writerow(data) csv_log.flush() #---------------------------------------------------------------------- # CSV_OPEN_FILE #---------------------------------------------------------------------- def csv_open_file(csv_log,now,filename): new_filename = '/var/log/USER/powermeter.py/{}{:02d}{:02d}'.format(now.year,now.month,now.day) if(new_filename != filename): if(filename != ""): csv_log.close() csv_log = open(new_filename, 'a', encoding='UTF8', newline='') csv_out = csv.writer(csv_log) return csv_log,csv_out,filename #---------------------------------------------------------------------- # NOW_TO_STAMP #---------------------------------------------------------------------- def now_to_stamp(now): stamp = '{:04d}-{:02d}-{:02d} {:02d}:{:02d}:{:02d}'.format(now.year, now.month, now.day, now.hour, now.minute, now.second) return stamp #---------------------------------------------------------------------- # PSENSOR_READ # Read all 10 registers, in one pass . . . Parse them out #---------------------------------------------------------------------- def psensor_read(pwm): try: reg_ar = pwm.read_registers(0,10,4) volts = reg_ar[0] / 10 amps = ((reg_ar[2] << 16) + reg_ar[1]) / 1000 watts = ((reg_ar[4] << 16) + reg_ar[3]) / 10 energy = ((reg_ar[6] << 16) + reg_ar[5]) frequency = reg_ar[7] / 10 power_factor = reg_ar[8] / 100 alarm = reg_ar[9] if(alarm > 0): alarm = 1 if(verbose > 0): print(reg_ar) except IOError: volts = 0 amps = 0 watts = 0 energy = 0 frequency = 0 power_factor = 0 alarm = 1 return volts,amps,watts,energy,frequency,power_factor,alarm #---------------------------------------------------------------------- # PSENSOR_OPEN # Parm 4 = Close on each query #---------------------------------------------------------------------- def psensor_open(psd_id): psd = minimalmodbus.Instrument('/dev/ttyUSB0', psd_id, 'rtu', True) psd.mode = minimalmodbus.MODE_RTU if(psd_id == 11): psd.serial.baudrate = 9600 psd.serial.bytesize = 8 psd.serial.parity = serial.PARITY_NONE psd.serial.stopbits = 1 psd.serial.timeout = 0.2 # psd.BYTEORDER_LITTLE = 1 if(verbose == 1): print('Details of the meter are:') print(psd) return psd #---------------------------------------------------------------------- # MAIN #---------------------------------------------------------------------- verbose = 0 argc = len(sys.argv) # print("argc = {}\n".format(argc)) if(argc > 1): if(sys.argv[1] == "-v"): verbose = 1 sensor_ids = [ 11, 12 ] # Global connection = psycopg2.connect(user="myhouse", password="myhouse", host="127.0.0.1", port="5432", database="myhouse") cursor = connection.cursor() psd_ar = [] for sensor_id in sensor_ids: psd_ar.append(psensor_open(sensor_id)) # csv_log,csv_out,filename = csv_open_file(csv_log,now,"") now = datetime.now() stamp = now_to_stamp(now) cur_stamp = stamp # Run the function to read the psensor meter. while True: for ix in [0,1]: sensor_id = sensor_ids[ix] psensor = psd_ar[ix] volts,amps,watts,energy,frequency,power_factor,alarm = psensor_read(psensor) if(amps != 0): add_table_entry(sensor_id,stamp,volts,amps,watts,energy,frequency,power_factor,alarm) # csv_add_entry(csv_log,csv_out,ix,stamp,amps,watts,volts,frequency) else: if (verbose == 1): print ("Read Failed\n") time.sleep(5) now = datetime.now() stamp = now_to_stamp(now)