0

Below is my attempt to send an image to Google Cloud IoT Core with MQTT. I have read the following posts which have helped me somewhat, but my code still isn't working: How Can I Publish File to AWS- IoT using Mosquitto in Python and How can I publish a file using Mosquitto in python?.

I would guess my error has to do either with the qos in the client.publish or how I have used the looping, but I'm afraid my experimenting with these factors hasn't helped me so far (trying qos = 0/1/2 and e.g. client.loop_forever()). My image is 1.2 Mb, so this shouldn't be a problem as far as I understand.

#!/usr/bin/python

from picamera import PiCamera
import datetime
import time
import jwt
import paho.mqtt.client as mqtt
from time import sleep

# Define some project-based variables to be used below. This should be the only
# block of variables that you need to edit in order to run this script

ssl_private_key_filepath = 'FILE1.pem'
ssl_algorithm = 'RS256' # Either RS256 or ES256
root_cert_filepath = 'FILE2.PEM'
project_id = 'PROJECT_ID'
gcp_location = 'LOCATION'
registry_id = 'REG_ID'
device_id = 'DEVICE_ID'

# end of user-variables

run = True
cur_time = datetime.datetime.utcnow()

def create_jwt():
  token = {
      'iat': cur_time,
      'exp': cur_time + datetime.timedelta(minutes=60),
      'aud': project_id
  }

  with open(ssl_private_key_filepath, 'r') as f:
    private_key = f.read()

  return jwt.encode(token, private_key, ssl_algorithm)

_CLIENT_ID = 'projects/{}/locations/{}/registries/{}/devices/{}'.format(project_id, gcp_location, registry_id, device_id)
_MQTT_TOPIC = '/devices/{}/events'.format(device_id)

client = mqtt.Client(client_id=_CLIENT_ID)
# authorization is handled purely with JWT, no user/pass, so username can be whatever
client.username_pw_set(
    username='unused',
    password=create_jwt())

def error_str(rc):
    return '{}: {}'.format(rc, mqtt.error_string(rc))

def on_connect(unusued_client, unused_userdata, unused_flags, rc):
    print('on_connect', error_str(rc))

def on_publish(unused_client, unused_userdata, unused_mid):
    print('on_publish')
    run = False


client.on_connect = on_connect
client.on_publish = on_publish

client.tls_set(ca_certs=root_cert_filepath) # Replace this with 3rd party cert if that was used when creating registry
client.connect('mqtt.googleapis.com', 8883)


camera = PiCamera()
camera.start_preview()
sleep(5)
camera.capture('/tmp/picture.jpg')
camera.stop_preview()

with open("/tmp/picture.jpg", 'rb') as f:
    imagestring = f.read()
byteArray = bytes(imagestring)


try:
    client.publish(_MQTT_TOPIC, byteArray, qos=2)
except:
    print('Error')

while run:
    client.loop()

client.disconnect()
Ingrid
  • 516
  • 6
  • 18
  • 3
    1.2Mb is way too big....https://cloud.google.com/iot/quotas – Aaron Jan 28 '20 at 15:12
  • I also wonder if `client.connect()` is asynchronous, and you are moving on with your script before the client connects to the server, and you are getting an error because it isn't connected yet. `try: except:` blocks should generally only be for errors you expect to happen like from bad user input, and exceptions like `StopIteration`. you may find the error message is more informative than `print('Error')` – Aaron Jan 28 '20 at 15:23
  • @Aaron Great, thanks! I only checked the limit for MQTT messages - forgot about the quotas at the Google end... I did a resize of the image when capturing, which solved the problem. – Ingrid Jan 28 '20 at 16:06

1 Answers1

1

The answer was provided by @Aaron.

"1.2Mb is way too big..."

Telemetry event payload limit is 256 KB and cannot be increased

Tlaquetzal
  • 2,760
  • 1
  • 12
  • 18