Remocon documentation

You will need:
- an API key from your account on https://remocon.tv, going forward <api_key> is to be replaced with your API key. You get an API key right away by creating a robot in your account page.


Video Streaming

Video can be streamed to the service using ffmpeg to our https://stream.remocon.tv server

ffmpeg -f v4l2 -framerate 25 -video_size 640x480 -i /dev/video0 \ -f mpegts -codec:v mpeg1video -s 640x480 -b:v 1000k -bf 0 \ https://stream.remocon.tv/<api_key>

If you are using a newer Raspberry Pi, then this command should work instead:

libcamera-vid -t 0 --width 640 --height 480 --codec mjpeg \ -o - | ffmpeg -i - -f mpegts -codec:v mpeg1video -b:v 1k -muxdelay 0.001 \ https://stream.remocon.tv/<api_key>


API Overview

Our API runs through websockets and returns JSON data.

It works just like any other HTTPS API out there that you are used to, except instead of making HTTPS requests you make a websocket connection, and then interpret the JSON data that you get back.

Interpreting the JSON data is very easy as you will see examples.

This means that it's very simple to set up, to use, and it's compatible with basically everything.

In Python, and MicroPython should be similar:

# run with python3, not with python import websockets # for web sockets import asyncio import json apiKey = "<api_key>" domain = "ws://control.remocon.tv/" async def connectSocket(): url = domain + apiKey async with websockets.connect(url) as websocket: print('websocket connected to ' + url) while True: input = await websocket.recv() try: interpretData(input) except: pass def interpretData(message): #print(message) data = json.loads(message) if (data["type"] == 'keydown' and data["key"] == 87): print('W key pressed') # put your code here on what you want to do # for example you could enable a GPIO pin here # GPIO.output(17,GPIO.HIGH) elif (data["type"] == 'keyup' and data["key"] == 87): print('W key released') # put your code here on what you want to do # for example you could disable a GPIO pin here # GPIO.output(17,GPIO.LOW) elif (data["type"] == 'keydown' and data["key"] == 83): print('S key pressed') # put your code here on what you want to do elif (data["type"] == 'keyup' and data["key"] == 83): print('S key released') # put your code here on what you want to do # run loop try: while True: asyncio.get_event_loop().run_until_complete(connectSocket()) asyncio.get_event_loop().run_forever() except KeyboardInterrupt: pass except: pass

In vanilla JavaScript:

In Node.js:

// Install Websockets with // npm install --save ws const WebSocket = require('ws'); const socket = new WebSocket('ws://control.remocon.tv/<api_key>'); // Replace this with your API key! socket.on('open', () => { console.log('Connected to the WebSocket server'); }); socket.on('message', data => { const jsonData = JSON.parse(data); if (jsonData.type === 'keydown' && jsonData.key === 87) { console.log('W key pressed'); // Put your code here on what you want to do } else if (jsonData.type === 'keyup' && jsonData.key === 87) { console.log('W key released'); // Put your code here on what you want to do } else if (jsonData.type === 'keydown' && jsonData.key === 65) { console.log('A key pressed'); // Put your code here on what you want to do } else if (jsonData.type === 'keyup' && jsonData.key === 65) { console.log('A key released'); // Put your code here on what you want to do } else if (jsonData.type === 'keydown' && jsonData.key === 83) { console.log('S key pressed'); // Put your code here on what you want to do } else if (jsonData.type === 'keyup' && jsonData.key === 83) { console.log('S key released'); // Put your code here on what you want to do } else if (jsonData.type === 'keydown' && jsonData.key === 68) { console.log('D key pressed'); // Put your code here on what you want to do } else if (jsonData.type === 'keyup' && jsonData.key === 68) { console.log('D key released'); // Put your code here on what you want to do } console.log('Received data:', jsonData); }); socket.on('close', () => { console.log('Disconnected from the WebSocket server'); }); socket.on('error', (error) => { console.error('WebSocket error:', error); }); setInterval(function () {socket.send('Keep-Alive')}, 1000);

In C and in Arduino:

That's it now get working.


Raspberry Pi Example

Open two terminal windows in your raspberry pi, run the following in each of them. The first streams the video, the second gets the commands from the website to control the motor and steering for a remote control car. To control anything else you just change the commands there.

First terminal window:

ffmpeg -f v4l2 -framerate 25 -video_size 640x480 -i /dev/video0 \ -f mpegts -codec:v mpeg1video -s 640x480 -b:v 1000k -bf 0 \ https://stream.remocon.tv/<api_key>

If that throws an error then try "sudo apt-get install ffmpeg" first, or reach out to us.

Second terminal window:

First create a file called websockets.py and copy this into it:

import RPi.GPIO as GPIO # for raspberry pi GPIO import websockets # for web sockets import asyncio import json # https://www.w3schools.com/python/python_json.asp # sending data in json format lets us send as much data # as we want and in any order apiKey = "<api_key>" domain = "ws://control.remocon.tv/" # set GPIO pin variables # the standard Raspberry Pi 4 pins are: 17, 27, 22, 5, 6, 13, 26, 23, 24, 25, 12, 16 motor = 17 servo = 27 # raspberry pi GPIO setup GPIO.setmode(GPIO.BCM) GPIO.setwarnings(False) GPIO.setup(motor,GPIO.OUT) GPIO.setup(servo,GPIO.OUT) motorPWM = GPIO.PWM(motor, 50) # servo GPIO for PWM with 50Hz # <-- might have to # play around with the frequency for the motor, # for the servo 50 should work but for the motor I am not sure motorPWM.start(0) servoPWM = GPIO.PWM(servo, 50) # servo GPIO for PWM with 50Hz servoPWM.start(7.5) async def connectSocket(): url = domain+apiKey async with websockets.connect(url) as websocket: print('websocket connected to ' + url) while True: input = await websocket.recv() try: interpretData(input) except: pass def interpretData(message): print(message) # comment this out if you don't want to see every single message data = json.loads(message) if (data["type"] == 'keydown' and data["key"] == 87): print('forward key pressed') motorPWM.ChangeDutyCycle(8) elif (data["type"] == 'keyup' and data["key"] == 87): print('forward key released') motorPWM.ChangeDutyCycle(0) elif (data["type"] == 'keydown' and data["key"] == 83): print('backward key pressed') motorPWM.ChangeDutyCycle(0) # <-- not sure if there # is a PWM duty cycle for reverse, motors # should have it but I am not sure what it is elif (data["type"] == 'keyup' and data["key"] == 83): print('backward key released') motorPWM.ChangeDutyCycle(0) elif (data["type"] == 'keydown' and data["key"] == 65): print('left key pressed') servoPWM.ChangeDutyCycle(2.5) elif (data["type"] == 'keyup' and data["key"] == 65): print('left key released') servoPWM.ChangeDutyCycle(7.5) elif (data["type"] == 'keydown' and data["key"] == 68): print('right key pressed') servoPWM.ChangeDutyCycle(12.5) elif (data["type"] == 'keyup' and data["key"] == 68): print('right key released') servoPWM.ChangeDutyCycle(7.5) asyncio.get_event_loop().run_until_complete(connectSocket()) asyncio.get_event_loop().run_forever()

Then run this in the terminal window:

sudo apt-get install python3 sudo apt-get install python3-pip pip3 install websockets pip3 install RPi python3 websockets.py # <-- this is the file we just created

And you're done, it should work. If you want to have these two run on startup then add them to the crontab file as cron jobs. The benefit of doing it this way is that you can see how it works and then customize what your Raspberry Pi does in response. When you run the websockets file it will show you every command, you can then change what's in the 'if' statements to make the Raspberry Pi do whatever you want.


How to make it run automatically on startup

Still writing this part, although you can do it with cron jobs. This is regarding the Raspberry Pi example above.

crontab -e

Audio Streaming

Audio can also be streamed but it's done using a separate command with a separate but provided API key. The audio API key is found in your account page on your robot under its regular API key.

ffmpeg -f alsa -i default:CARD=Device -f mpegts -codec:a mp2 -b:a 128k -muxdelay 0.001 \ https://stream.remocon.tv/<audio_api_key> -loglevel error

You might have to investigate yourself whether to also use ALSA and what the audio source is (instead of CARD=Device).



Why is this on Neocities rather than on some other documentation site?

It's because it's very easy to use and FOSS and ideologically good.

Last edited on 2025-01-16