Search code examples
pythonimagegoogle-cloud-platformmqttgoogle-cloud-iot

Why can't I send image to Google Cloud IoT Core with MQTT using Python (with this code)?


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()

Solution

  • The answer was provided by @Aaron.

    "1.2Mb is way too big..."

    Telemetry event payload limit is 256 KB and cannot be increased