Introduction
Welcome to the AV Studio API! You can use our API to pair, control and monitor Epiphan devices.
We have language bindings in bash and Python. You can view code examples in the dark area to the right, and you can switch the programming language of the examples with the tabs in the top right.
Big picture
Each encoder device has a long poll connection to AV Studio and waits for commands. Initially, all devices are unpaired, which means that they are not associated with any AV Studio account. To pair a device, log in to your AV Studio account and send a pairing command to AV Studio using the pairing code. AV Studio finds the device that has this code and connects your account to it.
Authorization
curl https://go.avstudio.com/front/api/v2/users/me \
  -H "Authorization: Bearer TOKEN"
api = AVStudioAPI2()
api.setAuthToken(TOKEN)
AV Studio uses bearer token authorization (rfc6750). HTTP API requests must include auth token in their HTTP headers:
Authorization: Bearer <TOKEN>
Issuing a token
Tokens are issued in "Integrations" tab in your AV Studio team settings:
Few notes about tokens:
- Tokens can be issued by the team's owner or admin. Multiple tokens can be issued for a team, but each token is linked to one team. 
- Tokens have admin permissions, without access to tokens and user management functionality. 
- Tokens do not expire. 
Getting All Devices
There're two devices in a newly created AV Studio account, let's retrieve them:
devices = api.Devices.get_all()
for d in devices:
    print d["Id"], d["Name"]
# Output:
# demo_0_d3d68f3c My First Demo Device
# demo_1_4e0a964a-b350-435f-82c9-de6ab5188af2 My Second Demo Device
curl https://go.avstudio.com/front/api/v2/devices \
    -H "Authorization: Bearer TOKEN"
The above command returns array with the device info dictionaries:
[
  {
    "Id": "demo_0_d3d68f3c",
    "Name": "My First Demo Device",
    "Model": "Demo",
    "Status": "Online",
    "Recording": "unknown",
    "StateTime": 1539366714.931,
    "SnapshotURL": "/front/api/v1/devices/demo_0_d3d68f3c/state.jpg",
    "IsUnpaired": false,
    "Telemetry": {}
  }
]
GET /front/api/v2/devices
Getting a Specific Device
device = api.Devices.get("demo_0_d3d68f3c")
curl https://go.avstudio.com/front/api/v2/devices/demo_0_d3d68f3c \
  -H "Authorization: Bearer TOKEN"
Result:
[
  {
    "Id": "demo_0_d3d68f3c",
    "Name": "My First Demo Device",
    "Model": "Demo",
    "Status": "Online",
    "Recording": "unknown",
    "StateTime": 1539366714.931,
    "SnapshotURL": "/front/api/v1/devices/demo_0_d3d68f3c/state.jpg",
    "IsUnpaired": false,
    "Telemetry": {}
  }
]
GET /front/api/v2/devices/DEVICEID
| Parameter | Description | 
|---|---|
| DEVICEID | The ID of the device to retrieve | 
Pairing a Real Device
Let's pair a real device.
Switching the Webcaster X2 Device to AV Studio Mode
There are two ways to switch modes:
Double press the power button on the device until the LCD screen shows Epiphan logo:

Or select AV Studio using a monitor and attached USB mouse:

Pairing the Device
To pair a device we use the following endpoint:
POST /front/api/v2/devices
r = api.Devices.add("5cf06c29", "NEW DEVICE")
device_id = r["ID"]
curl -X POST https://go.avstudio.com/front/api/v2/devices \
  -H "Authorization: Bearer TOKEN" \
  -d '{"DeviceID": "5cf06c29", "Name": "NEW DEVICE"}'
Result:
{
    "Status": "ok",
    "StatusCode": 200,
    "ID": "DEVICEID"
}
Device Commands
Sending Commands to Devices
e.g. setting bitrate to 1 Mbit/s:
api.Devices.run_command(deviceId, "setparam:bitrate=1000")
curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/task \
  -H "Authorization: Bearer TOKEN" \
  --data-binary '{"cmd": "setparam:bitrate=1000"}'
To send commands to devices, POST {"cmd": COMMAND} json to this endpoint:
POST front/api/v2/devices/DEVICEID/task
setparam:
api.Devices.get(deviceId)["Telemetry"]["settings"].keys()
# Output: ['resolution', 'bitrate']
curl https://go.avstudio.com/front/api/v2/devices/DEVICEID \
  -H "Authorization: Bearer TOKEN" \
| jq '.Telemetry.settings | keys'
# Output:
# [
#   "bitrate",
#   "resolution"
# ]
This command sets the value of the device settings parameter. Available parameters are listed in the device info Telemetry/settings dictionary.
Epiphan Webcaster X2's support the following parameters:
| Param | Possible values | Description | 
|---|---|---|
| resolution | 1920x1080, 1280x720, 640x360, 320x180 | Stream frame size | 
| bitrate | up to 4000 | Stream bitrate, in Kbit/s | 
| chunk_duration | 10-3600 | Chunk duration, in seconds | 
recording.start/stop
api.Devices.run_command(deviceId, "recording.start")
api.Devices.run_command(deviceId, "recording.stop")
 curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/task \
  -H "Authorization: Bearer TOKEN" \
  --data-binary '{"cmd": "recording.start"}'
 curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/task \
  -H "Authorization: Bearer TOKEN" \
  --data-binary '{"cmd": "recording.stop"}'
These commands start and stop recording. See "Recording" section for details.
afu.start/stop
api.Devices.run_command(deviceId, "afu.start")
api.Devices.run_command(deviceId, "afu.stop")
 curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/task \
  -H "Authorization: Bearer TOKEN" \
  --data-binary '{"cmd": "afu.start"}'
 curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/task \
  -H "Authorization: Bearer TOKEN" \
  --data-binary '{"cmd": "afu.stop"}'
These commands start and stop uploading the recording footage to AV Studio. See "Recording" section for details.
footage.clear
api.Devices.run_command(deviceId, "footage.clear")
 curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/task \
  -H "Authorization: Bearer TOKEN" \
  --data-binary '{"cmd": "footage.clear"}'
This command deletes recorded footage on the device (the footage that is not uploaded yet). See "Recording" section for details.
rtmp.start:url
api.Devices.run_command(deviceId, "rtmp.start:rtmp://10.1.2.16/live/test")
 curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/task \
  -H "Authorization: Bearer TOKEN" \
  --data-binary '{"cmd": "rtmp.start:rtmp://10.1.2.16/live/test"}'
The status of the RTMP stream is reported in
Telemetry/state/rtmp.
speedis actual_uploading_speed/bitrate ratio, values greater or equal to 1.0 are ok, less than 1.0 means there's not enough bandwidth.
"rtmp": {
    "url": "rtmp://10.1.2.16/live/test", 
    "speed": 1.00,
    "bitrate": 2061,
    "sent": {
        "frames": 896, 
        "bytes": 7723008
    }
}
This command starts an RTMP stream to a given RTMP url.
rtmp.stop
api.Devices.run_command(deviceId, "rtmp.stop")
 curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/task \
  -H "Authorization: Bearer TOKEN" \
  --data-binary '{"cmd": "rtmp.stop"}'
To stop the RTMP stream that was started by rtmp.start
firmware.update
api.Devices.run_command(deviceId, "firmware.update")
 curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/task \
  -H "Authorization: Bearer TOKEN" \
  --data-binary '{"cmd": "firmware.update"}'
Starts a firmware update if a more recent firmware version is available.
unpair
api.Devices.run_command(deviceId, "unpair")
 curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/task \
  -H "Authorization: Bearer TOKEN" \
  --data-binary '{"cmd": "unpair"}'
Unpairs the device from the account that it's paired to.
Recording
Webcaster X2 devices can record footage and upload it to the AV Studio account, to which they are paired.
On Webcaster X2 stream is recorded to SD card, or internal flash memory, if SD card is full or is not available.
Recorder is started by recording.start and stopped by recording.stop commands. If uploading is enabled by afu.start command, it gets uploaded to AV Studio and deleted from device storage.
If uploading is disabled by afu.stop command, footage stays on device, until it's deleted by footage.clear command, or uploading is started.
Recording, streaming and uploading can run at the same time.
The footage, uploaded to AV Studio, is billed according to the account's plan.
Recorded Footage on Device
api.Devices.get(deviceId)["Telemetry"]["state"]["afu"]
# Output:
# {
#   "queue_duration": 4793,    # The duration of stored footage, in seconds
#   "queue_size": 2513406672,  # The size of stored footage, in bytes
#   "speed": 0,                # Uploading speed (if footage is being uploaded)
#   "state": False             # True if uploading enabled, False otherwise
# }
curl https://go.avstudio.com/front/api/v2/devices/DEVICEID \
  -H "Authorization: Bearer TOKEN" \
| jq '.Telemetry.state.afu'
# Output:
# {
#   "queue_duration": 4793,    # The duration of stored footage, in seconds
#   "queue_size": 2513406672,  # The size of stored footage, in bytes
#   "speed": 0,                # Uploading speed (if footage is being uploaded)
#   "state": False             # True if uploading enabled, False otherwise
# }
Status of recorded, but not uploaded footage is reported in state.afu dictionary of device status.
Getting Recordings
Getting List of Uploaded Chunks
curl https://go.avstudio.com/front/api/v2/devices/DEVICEID/thumbnails?from=START&to=END \
  -H "Authorization: Bearer TOKEN"
# Output:
# [
#   {
#    "Start": 1546887235.983,
#    "End": 1546887246.35,
#    "Url": "chunks/DEVICEID/v2/video/000003000000.mp4"
#   },
#   {
#    "Start": 1546887246.383,
#    "End": 1546887256.749,
#    "Url": "chunks/DEVICEID/v2/video/000003000001.mp4"
#   },
#   {
#    "Start": 1546887256.783,
#    "End": 1546887267.016,
#    "Url": "chunks/DEVICEID/v2/video/000003000002.mp4"
#   }
# ]
api.Devices.get_timeline(DEVICEID, START, END)
# Output:
# [
#   {
#    "Start": 1546887235.983,
#    "End": 1546887246.35,
#    "Url": "chunks/DEVICEID/v2/video/000003000000.mp4"
#   },
#   {
#    "Start": 1546887246.383,
#    "End": 1546887256.749,
#    "Url": "chunks/DEVICEID/v2/video/000003000001.mp4"
#   },
#   {
#    "Start": 1546887256.783,
#    "End": 1546887267.016,
#    "Url": "chunks/DEVICEID/v2/video/000003000002.mp4"
#   }
# ]
GET /front/api/v2/devices/DEVICEID/thumbnails?from=START&to=END
| Parameter | Description | 
|---|---|
| DEVICEID | The ID of the device to retrieve | 
| START, STOP | Time range (UNIX timestamps) | 
Downloading Chunks
# Getting url of the first chunk from the range
url=$(curl -s "https://go.avstudio.com/front/api/v2/devices/DEVICEID/thumbnails?from=START&to=STOP" \
 -H "Authorization: Bearer TOKEN" | jq -r .[0].Url)
# Downloading the chunk
curl -o '/tmp/chunk.mp4' "https://go.avstudio.com/front/api/v2/media/$url" \
 -H "Authorization: Bearer TOKEN"
# Getting url of the first chunk from the range
url = api.Devices.get_timeline(DEVICEID, START, END)[0]["Url"]
# Downloading the chunk
api.HTTP.http_download_file(url, "/tmp/chunk.mp4")
GET /front/api/v2/media/URL
| Parameter | Description | 
|---|---|
| URL | Chunk URL from list of chunks | 
 
      