6.10 Writing the Script

The package delivery monitoring script needs to perform several functions, from listening for and reacting to triggers from the package delivery monitoring hardware to sending an email alert about the event and everything in between. Specifically, the script needs to do the following:

  1. Listen for threshold exceeded events (i.e., package delivery and removal) sent via the soft serial port communications between the XBee radios.

  2. Timestamp these events generated by the force sensitive resistor in the deliverystatus table.

  3. If a high value is received (i.e., a package is delivered), query tracking numbers stored in the tracking table. If a low value is received (i.e., a package is removed), send an email alert stating such and return execution of the script back to listening for delivery events.

  4. If a high value is received, wait for a specified time before querying the tracking table to allow time for the courier to update the delivery records.

  5. Iterate over undelivered tracking numbers and poll the appropriate courier’s web service records for delivery confirmation status.

  6. If the courier’s web service results report a delivery confirmation, change the status of the tracking number record in the local tracking database table to 1 (i.e., boolean value for delivered).

  7. Send an email via Gmail’s secure SMTP gateway that contains the results of delivery activity in the body of the message. Note that you will need login access to an active Gmail account for this function to work.

  8. Return to listening for additional package delivery events.

With those steps in mind, here’s the complete script.

PackageDeliveryDetector/packagedeliverydetector.py
  from datetime import datetime ​
  import packagetrack​
  from packagetrack import Package​
  import serial​
  import smtplib​
  import sqlite3​
  import time​
  import os​
  import sys​
  # Connect to the serial port
  ​XBeePort = serial.Serial('/dev/tty.YOUR_SERIAL_DEVICE', \​
  ​ baudrate = 9600, timeout = 1) ​
  ​​
  def send_email(subject, message): ​
  ​ recipient = 'YOUR_EMAIL_RECIPIENT@DOMAIN.COM'
  ​ gmail_sender = 'YOUR_GMAIL_ACCOUNT_NAME@gmail.com'
  ​ gmail_password = 'YOUR_GMAIL_ACCOUNT_PASSWORD'
  ​​
  # Establish secure TLS connection to Gmail SMTP gateway
  ​ gmail_smtp = smtplib.SMTP('smtp.gmail.com',587)​
  ​ gmail_smtp.ehlo()​
  ​ gmail_smtp.starttls()​
  ​ gmail_smtp.ehlo​
  ​​
  # Log into Gmail
  ​ gmail_smtp.login(gmail_sender, gmail_password)​
  ​​
  # Format message
  ​ mail_header = 'To:' + recipient + '\n' + 'From: ' + gmail_sender + '\n' \​
  ​ + 'Subject: ' + subject + '\n'
  ​ message_body = message​
  ​ mail_message = mail_header + '\n ' + message_body + ' \n\n'
  ​​
  # Send formatted message
  ​ gmail_smtp.sendmail(gmail_sender, recipient, mail_message)​
  ​ print("Message sent")​
  ​​
  # Close connection
  ​ gmail_smtp.close()​
  ​​
  def process_message(msg): ​
  try:​
  # Remember to use the full correct path to the
  # packagedelivery.sqlite file
  ​ connection = sqlite3.connect("packagedelivery.sqlite")​
  ​ cursor = connection.cursor()​
  ​​
  # Get current date and time and format it accordingly
  ​ timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S")​
  ​​
  ​ sqlstatement = "INSERT INTO delivery (id, time, event) \​
  VALUES(NULL, \"%s\", \"%s\")" % (timestamp, msg)​
  ​ cursor.execute(sqlstatement)​
  ​ connection.commit()​
  ​ cursor.close()​
  except:​
  ​ print("Problem accessing delivery table in the " \​
  ​ + "packagedelivery database")​
  ​​
  if (msg == "Delivery"):​
  ​​
  # Wait 5 minutes (300 seconds) before polling the various couriers
  ​ time.sleep(300)​
  ​​
  try:​
  ​ connection = sqlite3.connect("packagedelivery.sqlite")​
  ​ cursor = connection.cursor()​
  ​ cursor.execute('SELECT * FROM tracking WHERE '\​
  ​ + 'delivery_status=0')​
  ​ results = cursor.fetchall()​
  ​ message = ""
  ​​
  for x in results:​
  ​ tracking_number = str(x[1])​
  ​ description = str(x[2])​
  print tracking_number​
  ​​
  ​ package = Package(tracking_number)​
  ​ info = package.track()​
  ​ delivery_status = info.status​
  ​ delivery_date = str(info.delivery_date)​
  ​​
  if (delivery_status.lower() == 'delivered'):​
  ​ sql_statement = 'UPDATE tracking SET \​
  delivery_status = "1", delivery_date = \​
  "' + delivery_date + \​
  '" WHERE tracking_number = "' \​
  ​ + tracking_number + '";'
  ​ cursor.execute(sql_statement)​
  ​ connection.commit()​
  ​ message = message + description \​
  ​ + ' item with tracking number ' \​
  ​ + tracking_number \​
  ​ + ' was delivered on ' \​
  ​ + delivery_date +'\n\n'
  ​​
  # Close the cursor
  ​ cursor.close()​
  ​​
  # If delivery confirmation has been made, send an email
  if (len(message) > 0):​
  print message​
  ​ send_email('Package Delivery Confirmation', message)​
  else:​
  ​ send_email('Package Delivery Detected', 'A ' \​
  ​ + 'package delivery event was detected, ' \​
  ​ + 'but no packages with un-confirmed ' \​
  ​ + 'delivery tracking numbers in the database ' \​
  ​ + 'were able to be confirmed delivered by ' \​
  ​ + 'the courier at this time.')​
  except:​
  ​ print("Problem accessing tracking table in the " \​
  ​ + "packagedelivery database")​
  ​​
  else:​
  ​ send_email('Package(s) Removed', 'Package removal detected.')​
  ​​
  if sys.platform == "win32": ​
  ​ os.system("cls")​
  else:​
  ​ os.system("clear")​
  ​​
  ​print("Package Delivery Detector running...\n")​
  try:​
  while 1:​
  # listen for inbound characters from the XBee radio
  ​ XBee_message = XBeePort.readline()​
  ​​
  # Depending on the type of delivery message received,
  # log and lookup accordingly
  if "Delivery" in XBee_message:​
  # Get current date and time and format it accordingly
  ​ timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S")​
  ​ print("Delivery event detected - " + timestamp)​
  ​ process_message("Delivery")​
  ​​
  if "Empty" in XBee_message:​
  # Get current date and time and format it accordingly
  ​ timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S")​
  ​ print("Parcel removal event detected - " + timestamp)​
  ​ process_message("Empty")​
  ​​
  except KeyboardInterrupt: ​
  ​ print("\nQuitting the Package Delivery Detector.\n")​
  pass

Begin by importing the script’s dependencies on the custom packagetrack library along with the serial, smtplib, sqlite3, time, os, and sys standard Python libraries.

Identify the serial port of the XBee radio attached to the computer’s serial port. This radio will listen for incoming transmissions from the paired XBee radio attached to the Arduino connected to the force sensitive resistor. Replace the ’/dev/tty.YOUR_SERIAL_DEVICE’ placeholder with the actual path of the your XBee radio’s attached serial port value.

This is the send_mail routine that is used in the process_message routine, and thus it needs to be declared first. Replace the recipient, gmail_sender, and gmail_password placeholder values with your desired recipient and Gmail account credentials.

The process_message routine is where most of the action happens in the script. Connect to the packagedelivery.sqlite SQLite database and log the type of event received. If a delivery message is received, the script waits for five minutes before polling the FedEx and UPS web services to give the shipper enough time to log the delivery status to the central servers. Then, the tracking table in the packagedelivery.sqlite database is queried for any undelivered tracking numbers. These numbers are submitted one at a time to the respective web service. If a delivery confirmation is returned, its positive response is logged to the database with the confirmed delivery date, as well as appended to the body of the email message to be sent via the send_email routine.

This is the main loop of the script. Begin by clearing the screen and listening for a “Delivery” or “Empty” message from the Arduino-attached XBee radio and invoke the process_message routine.

Gracefully exit the script if a Ctrl-C keypress is detected.

Save the script as packagedelivery.py and execute it with the python packagedelivery.py command. If any errors arise, check syntax and code-line indentations, since Python is very strict about line formatting. If the script starts up without any complaints, you’re ready to test it out.

Programming Your Home
cover.xhtml
f_0000.html
f_0001.html
f_0002.html
f_0003.html
f_0004.html
f_0005.html
f_0006.html
f_0007.html
f_0008.html
f_0009.html
f_0010.html
f_0011.html
f_0012.html
f_0013.html
f_0014.html
f_0015.html
f_0016.html
f_0017.html
f_0018.html
f_0019.html
f_0020.html
f_0021.html
f_0022.html
f_0023.html
f_0024.html
f_0025.html
f_0026.html
f_0027.html
f_0028.html
f_0029.html
f_0030.html
f_0031.html
f_0032.html
f_0033.html
f_0034.html
f_0035.html
f_0036.html
f_0037.html
f_0038.html
f_0039.html
f_0040.html
f_0041.html
f_0042.html
f_0043.html
f_0044.html
f_0045.html
f_0046.html
f_0047.html
f_0048.html
f_0049.html
f_0050.html
f_0051.html
f_0052.html
f_0053.html
f_0054.html
f_0055.html
f_0056.html
f_0057.html
f_0058.html
f_0059.html
f_0060.html
f_0061.html
f_0062.html
f_0063.html
f_0064.html
f_0065.html
f_0066.html
f_0067.html
f_0068.html
f_0069.html
f_0070.html
f_0071.html
f_0072.html
f_0073.html
f_0074.html
f_0075.html
f_0076.html
f_0077.html
f_0078.html
f_0079.html
f_0080.html
f_0081.html
f_0082.html
f_0083.html
f_0084.html
f_0085.html
f_0086.html
f_0087.html
f_0088.html
f_0089.html
f_0090.html
f_0091.html
f_0092.html
f_0093.html
f_0094.html
f_0095.html
f_0096.html
f_0097.html
f_0098.html
f_0099.html
f_0100.html
f_0101.html
f_0102.html
f_0103.html
f_0104.html
f_0105.html
f_0106.html
f_0107.html
f_0108.html
f_0109.html
f_0110.html
f_0111.html
f_0112.html
f_0113.html
f_0114.html
f_0115.html
f_0116.html
f_0117.html
f_0118.html
f_0119.html
f_0120.html
f_0121.html
f_0122.html
f_0123.html
f_0124.html
f_0125.html