NAV
javascript graphql json shell python

Introduction

Welcome to the Unleash live API suite, where innovation meets efficiency!

Media Drive API: The Media Drive API unlocks the power of your Unleash live account's media storage and allows you to upload, manage, and retrieve media assets.

Mission API: The Mission API provides you with a comprehensive set of tools to streamline mission planning and execution. Using this API, you can programmatically list, upload and download the smart inspect or waypoint missions, allowing you to automate and optimize your unmanned aerial vehicle (UAV) operations.

Flight Logs API: With the Flight Logs API, you can access the flight data captured by your UAVs through Unleash live. Retrieve detailed flight logs, including telemetry, sensor readings, and performance metrics. Integrate your flight management system (eg AVCRM) to easily comply with regulatory requirements around tracking flight logs.

Analytics API: The Analytics API opens up a world of A.I. generated analytics data derived from your Unleash live account. Empower your in-house platforms with data-driven decision-making capabilities.

Jobs & Tasks API: The Jobs & Tasks API empowers you to create jobs and task withing your team.

Unleash CLI: Unleash Command Line Interface (CLI) is an easy way to deploy the models into production. Users need to prepare their model files in a folder and upload them to their account using their command line. We encourage you to explore and try our samples to use your Unleash live API. To access our samples, simply visit Samples API Github repository.

API Documentation

Version: 2.1.0

Authentication

HTTP

API request required authentication header with JWT token being received from AWS COGNITO service. You will need a user login and user password to send such request. We recommend to use aws sdk library as a handy helper. List of official AWS SDK in multiple programing language

We have prepared example code which shows authorization flow written in node.js

Basic node.js configuration required to run below script:

{
  "name": "unleash-live-api",
  "dependencies": {
    "amazon-cognito-identity-js": "^6.1.2",
    "aws-sdk": "^2.1336.0",
    "graphql-request": "^5.2.0",
    "jwt-decode": "^3.1.2"
  }
}

exampleConfig.js

export const API_URL_DEFAULT = "api.unleashlive.com";
export const FLIGHTS_CDN_DEFAULT = "https://flights.unleashlive.com";
export const USER_NAME = 'test@user-name.com';
export const PASSWORD = 'test-password';
export const GRAPHQL_ENDPOINT = 'https://mediadrive-api.unleashlive.com/graphql';
export const FOLDER_ID = 'test-folder-id';
export const UPLOAD_BUCKET_NAME = 'library-upload-syd-cloud';
export const MEDIA_DRIVE_BUCKET_NAME = 'library-syd-cloud';
export const ACCOUNT_ID = '590171865298';

"test@user-name.com": The same login email that is used to login in cloud.unleashlive.com.

"test-pass": Password to the above login email.

cognito.js

import AmazonCognitoIdentity from "amazon-cognito-identity-js";
import jwt_decode from "jwt-decode";
import AWS from "aws-sdk";
import {ACCOUNT_ID} from "../examples/exampleConfig.js";

const REGION = 'ap-southeast-2';
const IDENTITY_POOL_ID = 'ap-southeast-2:359c9d8b-02e4-457f-b185-4f3ff3f639d7';
const POOL_DATA = {
  UserPoolId: 'ap-southeast-2_UnpJmDeN5',
  ClientId: '7jhe1lfkb5pvhbkicc3d14m0ej'
};

export function initClient(idToken) {
  AWS.config.region = REGION;
  AWS.config.credentials = new AWS.CognitoIdentityCredentials({
    AccountId: ACCOUNT_ID,
    IdentityPoolId: IDENTITY_POOL_ID,
    Logins: {
      [`cognito-idp.${REGION}.amazonaws.com/${POOL_DATA.UserPoolId}`]: idToken
    },
  });
}
export function getUserPool() {
  return new AmazonCognitoIdentity.CognitoUserPool(POOL_DATA);
}

export function getCognitoUser(email) {
  const userData = {
    Username: email,
    Pool: getUserPool()
  };
  return new AmazonCognitoIdentity.CognitoUser(userData);
}

export function decodeJWTToken(token) {
  const {email, exp, auth_time, token_use, sub} = jwt_decode(token.idToken);
  return {token, email, exp, uid: sub, auth_time, token_use};
}

export function getAuthDetails(email, password) {
  const authenticationData = {
    Username: email,
    Password: password,
  };
  return new AmazonCognitoIdentity.AuthenticationDetails(authenticationData);
}

export function getCredentials() {
  return new AWS.CognitoIdentityCredentials({
    AccountId: ACCOUNT_ID,
    IdentityPoolId: AWS.config.credentials.params.IdentityPoolId,
    Logins: AWS.config.credentials.params.Logins,
  });
}

action.js, general scripts used before any action

import {initClient, decodeJWTToken, getAuthDetails, getCognitoUser} from "./cognito.js";

export function signIn(email, password) {
  return new Promise((resolve) => {
    getCognitoUser(email).authenticateUser(getAuthDetails(email, password), {
      onSuccess: (result) => {

        const jwtTokenPayload = result.getIdToken().decodePayload();
        const token = {
          accessToken: result.getAccessToken().getJwtToken(),
          idToken: result.getIdToken().getJwtToken(),
          refreshToken: result.getRefreshToken().getToken(),
          identityId: jwtTokenPayload['custom:identityId'],
          companyId: jwtTokenPayload['custom:companyId'],
          teamId: jwtTokenPayload['custom:teamId'],
        }
        initClient(token.idToken);
        return resolve({statusCode: 200, response: decodeJWTToken(token)});
      },

      onFailure: (err) => {
        return resolve({statusCode: 400, response: err.message || JSON.stringify(err)});
      },
    });
  });
}

exampleSignIn.js

import {signIn} from "../src/actions.js";

signIn(USER_NAME, PASSWORD)
.then(async (response) => {
  // Perform media drive actions
})
.catch(e => {
  console.error(e);
});

Every graphQL request should be sent with obligatory header authorization. Every REST request should be sent with obligatory header X-Amz-Cognito-Security-Token.

Quick Hint: If you are using other programming language, you can use the following command in your terminal to generate Security token: aws cognito-idp initiate-auth --auth-flow USER_PASSWORD_AUTH --region ap-southeast-2 --client-id 7jhe1lfkb5pvhbkicc3d14m0ej --auth-parameters USERNAME=<USER_LOGIN_EMAIL>,PASSWORD=<PASSWORD>

API KEY

This authentication method is mainly used to fetch analytics data generated during your AI sessions. Please contact here or your Account Manager to sort out API keys for you. Further, you can also email us at support@unleashlive.com.

Media Drive

Pre-requisites

Operation Types

Media drive api is build with support of GraphQL. It allows to perform few types of operations:

Name Description
Mutation Create, Update, Delete media drive item
Query Get , List existing items

Data Types

Media Drive item

Name Type Constraints Description
id String optional Item Id
teamId String optional Item teamId resource belongs to
parentId String optional Parent item Id. (In most cases it represents folder id)
tags String optional String which represent assigned tags to the item
type Enum optional Enum of custom types
location String optional Representation of hierarchy where item is placed. Each Id is seperated by "#"
deviceId String optional Id of device which was used to create an Item (if any)
name String optional The name of an item
createdAt String optional UTC Timestamp of item creation
updatedAt String optional UTC Timestamp of item last update
ownerId String optional Id of an author user
s3Path String optional Physical location of a file assigned with an item
metadata Object optional File Metadata
mimeType String optional Any available mime type plus custom types
annotations String optional Stringify Annotations object

Media Drive Metadata

Name Type Constraints Description
size Number Optional ...
childItemsNumber Int Optional ...
addonId String Optional ...
deviceId String Optional ...
annotationCount Int Optional ...
make String Optional ...
model String Optional ...
orientation Int Optional ...
software String Optional ...
camaperturevalue Float Optional ...
camfnumber Float Optional ...
camfocallength Float Optional ...
camfocallengthin35mmformat Int Optional ...
camiso Int Optional ...
gpsalt Float Optional ...
gpsaltref Int Optional ...
gpslat Float Optional ...
gpslatref String Optional ...
gpslng Float Optional ...
gpslngref String Optional ...
baseAltitude Float Optional ...
exifimagewidth Int Optional ...
exifimageheight Int Optional ...
gimbalpitchdegree Float Optional ...
automationid String Optional ...
createdate AWSTimestamp Optional ...
modifydate AWSTimestamp Optional ...
lastmodified AWSTimestamp Optional ...
type String Optional ...
width Int Optional ...
height Int Optional ...
bounds [[Float]] Optional ...
oldpath String Optional ...
oldPath String Optional ...
is360 Boolean Optional ...
duration Float Optional ...
isPanoramic Boolean Optional ...

Media Drive Custom MimeType possible values

Name Description
application/vnd.unleashlive.folder Media drive item representing folder
application/vnd.unleashlive.model Media drive item representing model wrapper
application/vnd.unleashlive.model.2d Media drive item representing 2D model
application/vnd.unleashlive.model.3d Media drive item representing 3D model
application/vnd.unleashlive.model.vr Media drive item representing VR model
application/vnd.unleashlive.model.pc Media drive item representing Point Cloud model

Media Drive item.type possible values

Key Value
I Image
V Video
M Model
R Report
FL Flight log
D General document
U Undefined type

Media Drive Annotations Object type

Key Value
{addonId} Annotation

Media Drive Annotation Object type

Key Type Constrains
addonId String Required
labels Image labels map Optional
sessionInfo Sessions info Optional

Media Drive ImageLabelsMap type

Key Value
{id} SingleLabel

Media Drive SessionsInfo type

Name Type Constraints Description
title String Optional ...
value String or String[] Required ...
order String Optional ...
placeholder String Optional ...

Media Drive SingleLabel type

Key Type Constrains
shapeType Shape types Required
category String Required
severity Number Optional
comment String Optional
color String Optional
updatedAt UTC Timestamp Required
bbox Number[] Optional
area Number Optional
distance Number Optional
isAI Boolean Optional
isAccepted Boolean Optional
isModified Boolean Optional

Media Drive ShapeTypes enum

Key Value
Polygon POLYGON
Rectangle RECTANGLE
Point POINT

Authenticate requests

Each request has to be authenticated with a valid JWT token. The token is passed in the authorization header. To learn on how to get JWT token go to

Get media

Example get query:

query GetLibraryItem {
  get(
    item: {id: "ItemId1"}
  ) {
    teamId
    location
    type
    tags
    createdAt
    deviceId
    id
    name
    parentId
    s3Path
    updatedAt
    mimeType
    annotations
    metadata {
      addonId
      camaperturevalue
      camfnumber
      camfocallength
      camfocallengthin35mmformat
      camiso
      childItemsNumber
      exifimageheight
      exifimagewidth
      gpsalt
      gpslat
      gpslatref
      gpslng
      gpslngref
      height
      isPanoramic
      lastmodified
      make
      model
      orientation
      size
      software
      type
      width
    }
  }
}

exampleGet.js

import * as graphQlRequest from 'graphql-request'
import {signIn} from "../src/actions.js";
import {FOLDER_ID, GRAPHQL_ENDPOINT, PASSWORD, USER_NAME} from "./exampleConfig.js";

const GraphQLClient = graphQlRequest.GraphQLClient;
const gql = graphQlRequest.gql;

export function getFolderItem(signInResponse, id) {
  const graphQLClient = new GraphQLClient(GRAPHQL_ENDPOINT, {
    headers: {
      authorization: signInResponse.response.token.idToken,
    },
  })

  const getQuery = gql`
    query GetLibraryItem{
      get(item: {id: "${id}"}) {
        id,
        location
      }
    }`

  return graphQLClient.request(getQuery);
}

signIn(USER_NAME, PASSWORD)
  .then(async (signInResponse) => {
    const folderItem = (await getFolderItem(signInResponse,FOLDER_ID)).get;
    console.log(folderItem);
  })
  .catch(e => {
    console.error(e);
  });

Example get response:

{
  "data": {
    "get": {
      "teamId": "teamId1",
      "location": "teamId1",
      "type": "F",
      "tags": "aiResultsNum:1",
      "createdAt": 1678802206809,
      "deviceId": null,
      "id": "ItemId1",
      "name": "Sample folder",
      "parentId": "teamId1",
      "s3Path": "12345/78910.mp4",
      "updatedAt": 1678976523362,
      "mimeType": "application/vnd.unleashlive.folder",
      "annotations": null,
      "metadata": {
        "addonId": null,
        "camaperturevalue": null,
        "camfnumber": null,
        "camfocallength": null,
        "camfocallengthin35mmformat": null,
        "camiso": null,
        "childItemsNumber": 13,
        "exifimageheight": null,
        "exifimagewidth": null,
        "gpsalt": null,
        "gpslat": null,
        "gpslatref": null,
        "gpslng": null,
        "gpslngref": null,
        "height": null,
        "isPanoramic": null,
        "lastmodified": null,
        "make": null,
        "model": null,
        "orientation": null,
        "size": 0,
        "software": null,
        "type": null,
        "width": null,
      },
    }
  }
}

Operation type: Query. Used to fetch full object by known itemId.

Query input parameters

Key Type Description Constrain
item {id: String} Object with selected item's Id Required

Query output type

Query output type

List media

Example list query:

query list {
  list(
    sort: desc
    location: "teamId1/folderId1"
    limit: 48
  ) {
    items {
      teamId
      location
      type
      tags
      createdAt
      deviceId
      id
      name
      parentId
      s3Path
      updatedAt
      mimeType
      metadata {
        isPanoramic
        duration
      }
    }
    nextToken {
      pk
      sk
      locationCreatedAt
      teamId
      teamIdType
      createdAt
      searchNameCreatedAt
      searchName
      deviceId
      type
    }
  }
}

exampleList.js

import {signIn} from "../src/actions.js";
import {FOLDER_ID, GRAPHQL_ENDPOINT, PASSWORD, USER_NAME} from "./exampleConfig.js";
import * as graphQlRequest from "graphql-request";
import {getFolderItem} from "./exampleGet.js";
const GraphQLClient = graphQlRequest.GraphQLClient;
const gql = graphQlRequest.gql;

export function listItems(signInResponse, location) {
  const graphQLClient = new GraphQLClient(GRAPHQL_ENDPOINT, {
    headers: {
      authorization: signInResponse.response.token.idToken,
    },
  })

  const listQuery = gql`
    query list {
      list(
        sort: desc
        location: "${location}"
        limit: 2
      ) {
        items {
          teamId
          location
          type
          tags
          createdAt
          deviceId
          id
          name
          parentId
          s3Path
          updatedAt
          mimeType
          metadata {
            isPanoramic
            duration
          }
        },
        nextToken {
          pk
          sk
          locationCreatedAt
          teamId
          teamIdType
          createdAt
          searchNameCreatedAt
          searchName
          deviceId
          type
        }
      }
    }`

  return graphQLClient.request(listQuery);
}

signIn(USER_NAME, PASSWORD)
  .then(async (signInResponse) => {
    const folderItem = (await getFolderItem(signInResponse, FOLDER_ID)).get;
    const location = [folderItem.location, folderItem.id].join("/");

    const listResponse = await listItems(signInResponse, location);
    console.log(listResponse);
  })
  .catch(e => {
    console.error(e);
  });

Example graphql list response:

{
  "data": {
    "list": {
      "items": [
        {
          "teamId": "teamId1",
          "location": "teamId1/folderId1",
          "type": "D",
          "tags": null,
          "createdAt": 1678976522491,
          "deviceId": "deviceId1",
          "id": "1",
          "name": "testDocument.json",
          "parentId": "folderId1",
          "s3Path": "1234/2b0d6d59-d221-4dc1-9cf4-78677a002127.json",
          "updatedAt": 1678976539957,
          "mimeType": "application/json",
          "metadata": {
            "isPanoramic": null,
            "duration": null,
          },
        },
        {
          "teamId": "teamId1",
          "location": "teamId1/folderId1",
          "type": "V",
          "tags": null,
          "createdAt": 1678976459317,
          "deviceId": "deviceId1",
          "id": "2",
          "name": "testClip.mp4",
          "parentId": "folderId1",
          "s3Path": "1234/2453702b-8764-4f55-bd81-487f58ec556a.mp4",
          "updatedAt": 1678976491797,
          "mimeType": "video/mp4",
          "metadata": {
            "isPanoramic": null,
            "duration": 1202.003,
          },
        },
        {
          "teamId": "teamId1",
          "location": "teamId1/folderId1",
          "type": "F",
          "tags": "aiResultsNum:1",
          "createdAt": 1678124970355,
          "deviceId": "deviceId1",
          "id": "3",
          "name": "Test folder",
          "parentId": "folderId1",
          "s3Path": "<path-of-file>",
          "updatedAt": 1678976549956,
          "mimeType": "application/vnd.unleashlive.folder",
          "metadata": {
            "isPanoramic": null,
            "duration": null,
          },
        },
        {
          "teamId": "teamId1",
          "location": "teamId1/folderId1",
          "type": "I",
          "tags": "MTY2MDg0NzMyNDMzOSNmYjcwNGIwMy02MjhhLTQyNDEtYmY5OC00NWNkNTFiN2MyNzc#MTY1ODc1NzkwNTUzOSM0OGUxZThiNi0zODA0LTRkMTQtYTJkYi0yOGY1ZmM4ZjUyMmM#aiResultsNum:1",
          "createdAt": 1524474282000,
          "deviceId": "deviceId1",
          "id": "4",
          "name": "DJI_0590.JPG",
          "parentId": "folderId1",
          "s3Path": "1234/efd72fa2-11a8-4369-84c0-8a159d8d3189.jpg",
          "updatedAt": 1678907063236,
          "mimeType": "image/jpeg",
          "metadata": {
            "isPanoramic": null,
            "duration": null,
          },
        }
      ],
      "nextToken": null,
    }
  }
}

List operation is a Query type operation. It should be used to fetch all subitems of known parentId. This variable can be fetched by:

Move media

Example move mutation:

mutation MoveItems {
    move(
        moveItems: [{id: "itemId1"}]
        to: {id: "itemId11"}
    ) {
        id
        parentId
        location
    }
}

exampleMove.js

import * as graphQlRequest from 'graphql-request'
import {signIn} from "../src/actions.js";
import {FOLDER_ID, GRAPHQL_ENDPOINT, PASSWORD, USER_NAME} from "./exampleConfig.js";
import {getFolderItem} from "./exampleGet.js";
const GraphQLClient = graphQlRequest.GraphQLClient;
const gql = graphQlRequest.gql;

export function moveItem(signInResponse, id, toId) {
  const graphQLClient = new GraphQLClient(GRAPHQL_ENDPOINT, {
    headers: {
      authorization: signInResponse.response.token.idToken,
    },
  })

  const moveQuery = gql`
    mutation MoveItems {
      move(
        moveItems: [{id: "${id}"}]
        to: {id: "${toId}"}
      ) {
        id
        parentId
        location
      }
    }
  `
  return graphQLClient.request(moveQuery);
}

signIn(USER_NAME, PASSWORD)
  .then(async (signInResponse) => {
    const moveResponse = await moveItem(signInResponse, FOLDER_ID, 'itemId11');
    console.log(moveResponse);

    const folderItem = (await getFolderItem(signInResponse, FOLDER_ID)).get;
    console.log(folderItem);
  })
  .catch(e => {
    console.error(e);
  });

Example move response:

{
  "data": {
    "move": [
      {
        "id": "itemId1",
        "parentId": "itemId11",
        "location": "teamId1/itemId11",
      }
    ]
  }
}

Action will move list of selected items along with it's all sub items to destination item.

Move mutation, input parameters

Key Type Description Constrain
moveItems {id: String}[] List of objets with selected item's Id Required
to {id: String} Destination object Required

Rename media

Example rename request:

mutation RenameItems {
  rename(
    item: {id: "itemId1"}
    newName: "New name"
  ) {
    id
  }
}

exampleRename.js

import * as graphQlRequest from 'graphql-request'
import {signIn} from "../src/actions.js";
import {FOLDER_ID, GRAPHQL_ENDPOINT, PASSWORD, USER_NAME} from "./exampleConfig.js";
import {getFolderItem} from "./exampleGet.js";
const GraphQLClient = graphQlRequest.GraphQLClient;
const gql = graphQlRequest.gql;

export function renameItem(signInResponse, id) {
  const graphQLClient = new GraphQLClient(GRAPHQL_ENDPOINT, {
    headers: {
      authorization: signInResponse.response.token.idToken,
    },
  })

  const renameQuery = gql`
    mutation RenameItems {
      rename(
        item: {id: "${id}"}
        newName: "New name"
      ) {
        id
      }
    }
  `
  return graphQLClient.request(renameQuery);
}

signIn(USER_NAME, PASSWORD)
  .then(async (signInResponse) => {
    const renameResponse = await renameItem(signInResponse, FOLDER_ID);
    console.log(renameResponse);

    const folderItem = (await getFolderItem(signInResponse, FOLDER_ID)).get;
    console.log(folderItem);
  })
  .catch(e => {
    console.error(e);
  });

Example rename response:

{
  "data": {
    "rename": {
      "id": "itemId1",
    }
  }
}

Action will rename selected item

Rename mutation, input parameters

Key Type Description Constrain
item {id: String} Object with selected item's Id Required
newName String New name which will be stored Required

Upload media

exampleUpload.js

import AWS from 'aws-sdk';
import {signIn} from "../src/actions.js";
import {FOLDER_ID, PASSWORD, UPLOAD_BUCKET_NAME, USER_NAME} from "./exampleConfig.js";
import * as uuid from "uuid";
import * as fs from "fs";
import {getCredentials} from "../src/cognito.js";
import {getFolderItem} from "./exampleGet.js";

const FILE_NAME = 'logo.png';
const FILE_PATH = `./assets/${FILE_NAME}`;

function getFileKey(companyId, teamId, identityId, fileName) {
  // return [companyId, teamId, identityId, uuid.v4(), fileName].join('/')
  return [identityId, uuid.v4(), fileName].join('/')
}

signIn(USER_NAME, PASSWORD)
  .then(async (response) => {
    const companyId = response.response.token.companyId;
    const teamId = response.response.token.teamId;
    const identityId = response.response.token.identityId;

    const folderItem = (await getFolderItem(FOLDER_ID)).get;
    const parentId = folderItem.id;
    const location = [folderItem.location, folderItem.id].join("/");

    // Upload file to it's destination
    fs.readFile(FILE_PATH, (err, data) => {
      if (err) throw err;
      const s3Client = new AWS.S3({
        credentials: getCredentials(),
        params: {Bucket: UPLOAD_BUCKET_NAME}
      });

      const params = {
        Bucket: UPLOAD_BUCKET_NAME, // pass your bucket name
        Key: getFileKey(companyId, teamId, identityId, FILE_NAME),
        Body: data,
        Metadata: {
          location: location,
          parentId: parentId,
          name: FILE_NAME,
        }
      };
      s3Client.upload(params, function(s3Err, data) {
        if (s3Err) throw s3Err
        console.log(`File uploaded successfully at ${data.Location}`)
      });
    });
  })
  .catch(e => {
    console.error(e);
  });

We recommend using AWS SDK lib to perform upload.

Metadata which can be passed with a file

Key Description Constrain Example value
deviceid Device which will be assign with item Optional deviceId1
location Location to which file will be uploaded Optional teamId1/itemId11
name Name which will be assign to a media drive item Optional New name
parentId Parent to which file will be uploaded Optional itemId11 (Deprecated)

Download media

exampleDownload.js

import { signIn } from "../../auth/src/actions.js";
import {
  FOLDER_ID,
  MEDIA_DRIVE_CDN,
  PASSWORD,
  USER_NAME,
} from "../../config/variables.js";
import { listItemsImages } from "./list.js";
import { getFolderItem } from "./get.js";

import * as https from "https";
import * as fs from "fs";
import {download} from "../../common/downloadFile.js";

signIn(USER_NAME, PASSWORD)
  .then(async (signInResponse) => {
    const folderItem = (await getFolderItem(signInResponse, FOLDER_ID)).get;
    const location = [folderItem.location, folderItem.id].join("/");

    const listResponse = (await listItemsImages(signInResponse, location)).list
      .items;

    const imageItem = listResponse.find((item) => item.type === "I");
    return download(MEDIA_DRIVE_CDN, imageItem.s3Path, imageItem.name);
  })
  .catch((e) => {
    console.error(e);
  });

Example script presents how to download file from library. Example flow does below steps:

Delete media

Example delete mutation:

mutation DeleteItems {
  delete(
    deleteItems: [{id: "itemId1"}, {id: "itemId2"}]
  ) {
    deleteItems {
      id
    }
    ownerId
  }
}

exampleDelete.js

import * as graphQlRequest from 'graphql-request'
import {signIn} from "../src/actions.js";
import {GRAPHQL_ENDPOINT, PASSWORD, USER_NAME} from "./exampleConfig.js";

const GraphQLClient = graphQlRequest.GraphQLClient;
const gql = graphQlRequest.gql;

export function deleteItem(signInResponse) {
  const graphQLClient = new GraphQLClient(GRAPHQL_ENDPOINT, {
    headers: {
      authorization: signInResponse.response.token.idToken,
    },
  })

  const deleteQuery = gql`
    mutation DeleteItems {
      delete(
        deleteItems: [{id: "itemId1"}, {id: "itemId2"}]
      ) {
        deleteItems {
          id
        }
        ownerId
      }
    }
  `
  return graphQLClient.request(deleteQuery);
}

signIn(USER_NAME, PASSWORD)
.then(async (signInResponse) => {
  const deleteResponse = (await deleteItem(signInResponse))
  console.log(deleteResponse);

  const getQuery = gql`
      query GetLibraryItem{
        get(item: {id: "itemId1"}) {
          id,
          name
        }
      }`

  const graphQLClient = new GraphQLClient(GRAPHQL_ENDPOINT, {
    headers: {
      authorization: signInResponse.response.token.idToken,
    },
  })

  const getResponse = await graphQLClient.request(getQuery);
  console.log(getResponse);
})
.catch(e => {
  console.error(e);
});

Example delete response:

{
  "data": {
    "delete": {
      "deleteItems": [
        {
          "id": "itemId1"
        }, {
          "id": "itemId2"
        }
      ],
      "ownerId": "userId1",
    }
  }
}

Action will remove list of selected items along with it's all subitems.

Delete mutation, input parameters

Key Type Description Constrain
deleteItems {id: String}[] List of objets with selected item's Id Required

Flight Logs

List logs

List logs is basic REST endpoint

GET: /v1/flights

Query Parameters:

dateFrom (optional): Returns flight logs from the specified timestamp, including that timestamp.

dateTo (optional): Returns flight logs up to the specified timestamp, including that timestamp.

For instance:

https://api.unleashlive.com/v1/flights?dateFrom=1709654015526 returns everything from particular timestamp including that timestamp

https://api.unleashlive.com/v1/flights?dateTo=1709654015526 returns everything to particular timestamp including that timestamp

https://api.unleashlive.com/v1/flights?dateFrom=1709654015525&dateTo=1709654015528 returns everything in between those 2 timestamps

The timestamp needs to be in UNIX timestamp format, in milliseconds.

flights/examples/list.js

You can either use programming language of your choice or you can try the endpoint with simple CURL.

export async function listFlightLogs(signInResponse) {
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: '/v1/flights',
    method: 'GET',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendGet(requestOptions);
}

Download log

In order to download a file you need to know its path and a FLIGHTS_CDN which is: https://flights.unleashlive.com

To store a file locally with the same name which is presented on unleash live platform use name = flightLog.name.

import https from "https";
import fs from "fs";

export function download(cdn, filePath, name) {
  const file = fs.createWriteStream(name);
  return https.get(
    [cdn, filePath].join("/"),
    function (response) {
      response.pipe(file);

      // after download completed close filestream
      file.on("finish", () => {
        file.close();
        console.log("Download Completed");
      });
    }
  );

}


function getFileNameFromPath(filePath) {
  const filePathParts = filePath.split('/');
  return filePathParts[filePathParts.length - 1];
}

try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const flightLogs = await listFlightLogs(signInResponse);
  const firstItem = flightLogs[0];
  const localFileName = getFileNameFromPath(firstItem.s3Path);
  await download(FLIGHTS_CDN, firstItem.s3Path, localFileName);
} catch (e) {
  console.error(e);
}

Missions

Upload mission

Uploading mission is a basic post request to below endpoint

POST: /v1/mission

mission/examples/upload.js

export async function uploadMission(signInResponse) {
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: '/v1/mission',
    method: 'POST',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendPost(requestOptions, BODY);
}


try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const uploadResponse = await uploadMission(signInResponse);
  console.log(uploadResponse);
} catch (e) {
  console.error(e);
}

Analytics

Description

This API exposes Unleash live read-only endpoints, allowing developers and users to retrieve AI-generated analytics data.

API access is currently an optional addon, please contact your account manager to request a demo or for more information.

URL

The current analytics API can be found at:

https://api.unleashlive.com/v1

Parameters

All in-use parameters appear here, consult each endpoint for more information on what is required.

Headers

 curl -X GET -H "x-api-key: <API-KEY>" <URL>
import urllib3
http = urllib3.PoolManager()
r = http.request('GET', <ENDPOINT>, headers={'x-api-key': <API-KEY>})
r.data

Only one custom header is required to use the Analytics API - your individual customer API key.

Information about how to get api key API KEY

x-api-key: <API KEY>

Variables

Name Type Constraints Description Example
deviceId string required, uuid format Unique Device ID Wabcdb35be35bb72
unixUTCTimeStampFrom integer required, uuid format Unix epoch, milliseconds 1597630156000
unixUTCTimeStampTo integer required, uuid format Unix epoch, milliseconds 1597630156000
pageNumber integer required, uuid format Page number 0

Endpoints

All current valid endpoints.

Version

 curl -X GET -H "x-api-key: <API-KEY>" https://api.unleashlive.com/v1/analytics/version
import urllib3
http = urllib3.PoolManager()
r = http.request('GET', 'https://api.unleashlive.com/v1/analytics/version', headers={'x-api-key': <API-KEY>})
r.data
{"1.0.1"}

Retrieve current API version number.

Returns version number in the format "1.0.1", and should be used for testing and confirmation of valid API key.

API Endpoint

GET /analytics/version

Devices

curl -X GET -H "x-api-key: <API-KEY>" https://api.unleashlive.com/v1/analytics/devices
import urllib3
http = urllib3.PoolManager()
r = http.request('GET', 'https://api.unleashlive.com/v1/analytics/devices', headers={'x-api-key': <API-KEY>})
r.data
["Wb8fabc31378ce07","W022abc5be35bb72","Weea288bc1faaddd"]

Returns a list of all valid devices for a given API Key and account in the format:

["Wabcdb35be35bb72", ...]

Where each item corresponds to an Unleash live Device ID.

API Endpoint

GET /analytics/devices

Sessions

curl -X GET -H "x-api-key: <API-KEY>" https://api.unleashlive.com/v1/analytics/sessions
import urllib3
http = urllib3.PoolManager()
r = http.request('GET', 'https://api.unleashlive.com/v1/analytics/sessions', headers={'x-api-key': <API-KEY>})
r.data
["1595418530512","1595452859625","1595492155987"]

Returns a list of all sessions for a given API Key and account in the format:

["1595418530512", ...]

Where each item corresponds to an Unleash live session ID.

API Endpoint

GET /analytics/sessions

Size Optimized Data

 curl -X GET -H "x-api-key: <API-KEY>" https://api.unleashlive.com/v1/analytics/{deviceId}/{unixUTCTimeStampFrom}/{unixUTCTimeStampTo}/{pageNumber}
import urllib3
http = urllib3.PoolManager()
r = http.request('GET', 'https://api.unleashlive.com/v1/analytics/{deviceId}/{unixUTCTimeStampFrom}/{unixUTCTimeStampTo}/{pageNumber}', headers={'x-api-key': <API-KEY>})
r.data
{"keys": ["class_name","count","device_id","frame_number","model_id","session_id","source_file_name","timestamp"],
"total_num_pages":1,
"data":[["person",1,"Wabcdb35be35bb72",110251,"starter-ai","1597622730877","Wabcdb35be35bb72?token=sneaky-firefox-288","2020-08-17T02:08:49.000Z"],
  ["person",3,"Wabcdb35be35bb72",110252,"starter-ai","1597622730878","Wabcdb35be35bb72?token=sneaky-firefox-288","2020-08-17T02:08:49.000Z"]]}

Retrieve raw analytics data generated between two given dates.

This endpoint is size-optimized - three fields are returned:

Field Type Description
keys list (string) A list of all keys used in the data field, in order of use.
total_num_pages integer The number of pages available for the requested data.
data list (mixed) List containing all requested data.
API Endpoint

GET /analytics/{deviceId}/{unixUTCTimeStampFrom}/{unixUTCTimeStampTo}/{pageNumber}

Tableau Formatted Data

 curl -X GET -H "x-api-key: <API-KEY>" https://api.unleashlive.com/v1/analytics/tableau/{deviceId}/{unixUTCTimeStampFrom}/{unixUTCTimeStampTo}/{pageNumber}
import urllib3
http = urllib3.PoolManager()
r = http.request('GET', 'https://api.unleashlive.com/v1/analytics/tableau/{deviceId}/{unixUTCTimeStampFrom}/{unixUTCTimeStampTo}/{pageNumber}', headers={'x-api-key': <API-KEY>})
r.data
{"total_num_pages":1,
"data":[{"timestamp":"2020-08-17T02:08:49.000Z","class_name":"person","count":1,"device_id":"Wabcdb35be35bb72","source_file_name":"Wabcdb35be35bb72?token=sneaky-firefox-288","frame_number":110251,"model_id":"starter-ai","session_id":"1597622730877"},
  {"timestamp":"2020-08-17T02:08:49.000Z","class_name":"truck","count":1,"device_id":"Wabcdb35be35bb72","source_file_name":"Wabcdb35be35bb72?token=sneaky-firefox-288","frame_number":110252,"model_id":"starter-ai","session_id":"1597622730877"}]}

Retrieve tableau formatted analytics data generated between two given dates.

This endpoint is descriptive, allowing data to be read nativily by services such as Tableau and salesforce easily.

Field Type Description
total_num_pages integer The number of pages available for the requested data.
data list (mixed) List containing all requested data.
API Endpoint

GET /analytics/tableau/{deviceId}/{unixUTCTimeStampFrom}/{unixUTCTimeStampTo}/{pageNumber}

Jobs & Tasks

Job & tasks types

UserTaskStatus
DRAFT
PUBLISHED
IN_PROGRESS
COMPLETED
FAILED

Create job

export async function createJob(signInResponse) {
  const body = {
    title: 'Job1',
    description: 'Detailed job description'
    // It's possible to attach users who are withing the same team as author of the job
    // Author of the job is assigned to the job automatically
    // userIds: ['user-id-1', 'user-id-2', ....]
  };

  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: '/v1/job',
    method: 'POST',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendPost(requestOptions, body);
}

try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const newJob = await createJob(signInResponse);
  console.log(newJob);
} catch (e) {
  console.error(e);
}

Example response:

{
  "id": "123456",
  "createdAt": 1234,
  "updatedAt": 1234,
  "totalTasks": 0,
  "completedTasks": 0,
  "title": "Job1",
  "description": "Detailed job description",
  "userIds": [],
  "ownerId": "1234",
  "teamId": "1234",
  "companyId": "1234",
  "status": "DRAFT",
  "pk": "1234",
  "sk": "1234"
}

Send post request to create job

POST /v1/job

Body parameters:

Key Type Description Constrain
title String Job name Required
description String Job description Required
usrIds String[] Job name Optional
status UserTaskStatus[] Job status Optional

Get job

{
  "id": "123456",
  "createdAt": 1234,
  "updatedAt": 1234,
  "totalTasks": 0,
  "completedTasks": 0,
  "title": "Job1",
  "description": "Detailed job description",
  "userIds": [],
  "ownerId": "1234",
  "teamId": "1234",
  "companyId": "1234",
  "status": "DRAFT",
  "pk": "1234",
  "sk": "1234"
}
export async function getJob(signInResponse) {
  const jobId = '1736424501360-483d4874-b1b5-4361-a0ba-b4fdb65cee13'
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: `/v1/job/${jobId}`,
    method: 'GET',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendGet(requestOptions);
}

try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const newJob = await getJob(signInResponse);
  console.log(newJob);
} catch (e) {
  console.error(e);
}

Fetch the job from database

GET /v1/job/{jobId}

Example response:

List assigned jobs

Example response:

[
  {
    "companyId": "1234",
    "assignedId": "1234",
    "updatedAt": 1234,
    "createdAt": 1234,
    "jobId": "1234",
    "sk": "JU#1234",
    "teamId": "1234",
    "id": "1234",
    "pk": "1234",
    "title": "Job1"
  }
]
export async function listJobs(signInResponse) {
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: '/v1/job',
    method: 'GET',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendGet(requestOptions);
}


try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const response = await listJobs(signInResponse);
  console.log(response);
} catch (e) {
  console.error(e);
}

Fetch all jobs assigned to my user

GET /v1/job

List active team jobs

Example response:

{
  "items": [
    {
      "id": "123456",
      "createdAt": 1234,
      "updatedAt": 1234,
      "totalTasks": 0,
      "completedTasks": 0,
      "title": "Job1",
      "description": "Detailed job description",
      "userIds": [],
      "ownerId": "1234",
      "teamId": "1234",
      "companyId": "1234",
      "status": "DRAFT",
      "pk": "1234",
      "sk": "1234"
    }
  ]
}
export async function listTeamJobs(signInResponse) {
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: '/v1/job/team',
    method: 'GET',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendGet(requestOptions);
}


try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const response = await listTeamJobs(signInResponse);
  console.log(response);
} catch (e) {
  console.error(e);
}

Fetch all jobs for current team. Require admin privileges.

GET /v1/job/team

Update job

export async function updateJob(signInResponse) {
  const jobId = '12345';
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: `/v1/job/${jobId}`,
    method: 'PATCH',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendPatch(requestOptions, {'title': 'New job title'});
}


try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const response = await updateJob(signInResponse);
  console.log(response);
} catch (e) {
  console.error(e);
}

Example response

{
  "updatedAt": 1234,
  "title": "Updated field"
}

Use it to update job's fields values. Api returns only updated fields

PATCH /v1/job/{jobId}

Body parameters:

Key Type Description Constrain
title String Job name Optional
description String Job description Optional
usrIds String[] Job name Optional
status UserTaskStatus[] Job status Optional

Delete job

export async function deleteJob(signInResponse) {
  const jobId = '1736427631226-cf674aaa-d2a6-4ee9-b701-2271b0037cd9\'';
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: `/v1/job/${jobId}`,
    method: 'DELETE',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendDelete(requestOptions);
}


try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const deleteResponse = await deleteJob(signInResponse);
  console.log(deleteResponse);
} catch (e) {
  console.error(e);
}

Example response:

{
  "id": "123456",
  "createdAt": 1234,
  "updatedAt": 1234,
  "totalTasks": 0,
  "completedTasks": 0,
  "title": "Job1",
  "description": "Detailed job description",
  "userIds": [],
  "ownerId": "1234",
  "teamId": "1234",
  "companyId": "1234",
  "status": "DRAFT",
  "pk": "1234",
  "sk": "1234"
}

Delete and returns deleted job

DELETE /v1/job/{jobId}

Create task

export async function createTask(signInResponse) {
  const body = {
    title: 'Task title',
    description: 'Task description',
    // Task should be assigned to the job. Without that assigning user is forbidden
    jobId: '1234',
    type: 'FLIGHT_REQUEST',
    // It is possible to bind a mission to the task. That operation can be done
    // on creation or on update as well.
    // context: {
    //   mission?: Mission
    // }
    // mission?: Mission;

    // Optional field, task can be assigned to the user.
    // Note if the task is not assigned to the job it then assigning it to the user will be rejected
    // userId?: string;
    lat: -33.86355993143128,
    lng: 151.2021623286489
  }

  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: '/v1/task',
    method: 'POST',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendPost(requestOptions, body);
}

try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const response = await createTask(signInResponse);
  console.log(response);
} catch (e) {
  console.error(e);
}

Example response:

{
  "id": "1234",
  "createdAt": 1234,
  "updatedAt": 1234,
  "title": "Task title",
  "description": "Task description",
  "jobId": "1234",
  "type": "FLIGHT_REQUEST",
  "lat": -33.86355993143128,
  "lng": 151.2021623286489,
  "ownerId": "1234",
  "teamId": "1234",
  "companyId": "1234",
  "status": "PUBLISHED",
  "pk": "1234",
  "sk": "T#1234"
}

Add task to existing job

POST /v1/task

TaskType
FLIGHT_REQUEST
FLIGHT_REQUEST_MANUAL

Flight History: >

Key Type Description Constrain
flightStart Number begging of the flight Required
flightEnd Number end of the flight Required
distance Number distance Required
s3Path Number path of the file with flight route Required

TaskContext: >

Key Type Description Constrain
mission Partial Optional
asset Partial end of the flight Optional
properties Record<String, String additional metadata describing task Optional

Body parameters:

Key Type Description Constrain
jobId String job id Required
type TaskType task type Required
description String description Optional
status UserTaskStatus[] task status Optional
lat Number latitude Optional
lng Number longitude Optional
flightHistory FlightHistory[] History of flights Optional
context TaskContext Additional task context Optional

Create task with mission

const BODY =  {
  "jobId": "1234",
  "type": 'FLIGHT_REQUEST',
  "desc": "Sample-mission",
  "name": "test mission 2",
  "speed": 2,
  "route": [
    {
      "pole": "Pole1",
      "wo": "00800521567",
      "lat": -33.72577124,
      "lng": 151.18167339,
      "altitude": 163.144,
      "altEGM": 187.144,
      "si": "dist-3"
    },
    {
      "pole": "WP",
      "wo": "00800521567",
      "lat": -33.72577124,
      "lng": 151.18167339,
      "altitude": 171.144,
      "altEGM": 195.144,
      "si": "waypoint"
    },
    {
      "pole": "WP",
      "wo": "00800521567",
      "lat": -33.72582000,
      "lng": 151.18118322,
      "altitude": 168.144,
      "altEGM": 192.144,
      "si": "waypoint"
    },
    {
      "pole": "Pole2",
      "wo": "00800521567",
      "lat": -33.72582000,
      "lng": 151.18118322,
      "altitude": 186.938,
      "altEGM": 188.938,
      "si": "dist-1"
    },
    {
      "pole": "Pole3",
      "wo": "00800521568",
      "lat": -33.72554444,
      "lng": 151.18123573,
      "altitude": 163.383,
      "altEGM": 187.383,
      "si": "dist-3"
    }
  ]
}

export async function createCorridorMissionTask(signInResponse) {
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: '/v1/task/mission',
    method: 'POST',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendPost(requestOptions, BODY);
}


try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const response = await createCorridorMissionTask(signInResponse);
  console.log(response);
} catch (e) {
  console.error(e);
}

Example response:

{
  "id": "1234",
  "createdAt": 1234,
  "updatedAt": 1234,
  "type": "FLIGHT_REQUEST",
  "title": "test mission 2",
  "ownerId": "1234",
  "teamId": "1234",
  "companyId": "1234",
  "lat": -33.72577124,
  "lng": 151.18167339,
  "context": {
    "mission": {
      "id": "1234"
    }
  },
  "jobId": "1234",
  "status": "PUBLISHED",
  "pk": "1234",
  "sk": "T#1234"
}

That endpoint allows to create a task with mission attached in one go. The mission can be then reused in other tasks just by passing it's id to context.mission.id

POST /v1/task/mission

Create task with corridor mission

const BODY =  {
  "jobId": "1234",
  "desc": "Mission corridor",
  "name": "test mission 2",
  "speed": 5,
  "route": [
    {
      "pole": "RegularPoint",
      "wo": "00800521566",
      "lat": 52.253057,
      "lng": 21.067224,
      "altitude": 39.99,
      "altEGM": 20.74,
      "heading": 168,
      "pitch": -50,
      "speed": 2,
      "si": "waypoint",
      "actions": [
        {
          "action": "TAKE_PHOTO"
        }
      ]
    },
    {
      "pole": "RegularPoint",
      "wo": "00800521567",
      "lat": 52.253257,
      "lng": 21.067424,
      "altitude": 41.1,
      "altEGM": 41.1,
      "si": "dist-1"
    },
    {
      "pole": "Pole2",
      "wo": "00800521567",
      "lat": 52.253258,
      "lng": 21.068002,
      "altitude": 44.1,
      "altEGM": 44.1,
      "si": "dist-2"
    },
    {
      "pole": "Pole3",
      "wo": "00800521568",
      "lat": 52.253311,
      "lng": 21.068288,
      "altitude": 38,
      "altEGM": 38,
      "si": "dist-3"
    }
  ]
};

export async function createMissionTask(signInResponse) {
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: '/v1/task/mission',
    method: 'POST',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendPost(requestOptions, {
    jobId: "1736427631226-cf674aaa-d2a6-4ee9-b701-2271b0037cd9",
    ...BODY
  });
}


try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const response = await createMissionTask(signInResponse);
  console.log(response);
} catch (e) {
  console.error(e);
}

Example response:

{
  "id": "1234",
  "createdAt": 1234,
  "updatedAt": 1234,
  "type": "FLIGHT_REQUEST",
  "title": "test mission 2",
  "ownerId": "1234",
  "teamId": "1234",
  "companyId": "1234",
  "lat": 52.253057,
  "lng": 21.067224,
  "context": {
    "mission": {
      "id": "1234"
    }
  },
  "jobId": "1234",
  "status": "PUBLISHED",
  "pk": "1234",
  "sk": "T#1234"
}

That endpoint allows to create a task with corridor mission attached in one go. The mission can be then reused in other tasks just by passing it's id to context.mission.id

POST /v1/task/mission-corridor

Update task

export async function updateTask(signInResponse) {
  const taskId = '1234';
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: `/v1/task/${taskId}`,
    method: 'PATCH',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendPatch(requestOptions, {'title': 'Update title'});
}


try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const response = await updateTask(signInResponse);
  console.log(response);
} catch (e) {
  console.error(e);
}
{
  "companyId": "c4177ce0-ac52-414c-bffb-2f5bf72f135a",
  "lng": 21.067224,
  "status": "PUBLISHED",
  "createdAt": 1234,
  "jobId": "1234",
  "teamId": "1234",
  "ownerId": "1234",
  "context": {
    "mission": {
      "id": "1234"
    }
  },
  "updatedAt": 1234,
  "sk": "T#1234",
  "id": "1234",
  "lat": 52.253057,
  "pk": "1234",
  "title": "Task title updated",
  "type": "FLIGHT_REQUEST"
}

Endpoint used for task updates. Api returns whole new task object

PATCH /v1/task/{taskId}

Get task

export async function getTask(signInResponse) {
  const taskId = '1234'
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: `/v1/task/${taskId}`,
    method: 'GET',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendGet(requestOptions);
}

try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const newTask = await getTask(signInResponse);
  console.log(newTask);
} catch (e) {
  console.error(e);
}

Example response:

{
  "companyId": "1234",
  "lng": 21.067224,
  "status": "PUBLISHED",
  "createdAt": 1234,
  "jobId": "1234",
  "teamId": "1234",
  "ownerId": "1234",
  "context": {
    "mission": {
      "id": "1234"
    }
  },
  "updatedAt": 1234,
  "sk": "T#1234",
  "id": "1234",
  "pk": "1234",
  "lat": 52.253057,
  "type": "FLIGHT_REQUEST",
  "title": "test mission 2"
}

Use to fetch task from database.

GET /v1/task/{taskId}

List assigned tasks

export async function listAssignedTasks(signInResponse) {
  const limit = 1;
  const nextToken = 'abcd';

  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: `/v1/task?limit=${limit}&nextToken=${nextToken}`,
    method: 'GET',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendGet(requestOptions);
}


try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const response = await listAssignedTasks(signInResponse);
  console.log(response);
} catch (e) {
  console.error(e);
}

Example response:

{
  "items": [
    {
      "companyId": "1234",
      "assignedId": "1234",
      "lng": 21.067224,
      "status": "PUBLISHED",
      "createdAt": 1234,
      "jobId": "1234",
      "teamId": "1234",
      "assignedAt": 1234,
      "ownerId": "1234",
      "updatedAt": 1234,
      "context": {
        "mission": {
          "id": "1234",
          "createdAtId": "1234",
          "companyId": "1234",
          "updatedAt": 1234,
          "route": [
            {
              "altitude": 39.99,
              "lng": 21.067224,
              "heading": 168,
              "si": "waypoint",
              "wo": "00800521566",
              "pitch": -50,
              "pole": "RegularPoint",
              "actions": [
                {
                  "action": "TAKE_PHOTO"
                }
              ],
              "lat": 52.253057,
              "speed": 2,
              "altEGM": 20.74
            },
            {
              "altitude": 41.1,
              "lng": 21.067424,
              "si": "dist-1",
              "wo": "00800521567",
              "pole": "RegularPoint",
              "lat": 52.253257,
              "altEGM": 41.1
            },
            {
              "altitude": 44.1,
              "lng": 21.068002,
              "si": "dist-2",
              "wo": "00800521567",
              "pole": "Pole2",
              "lat": 52.253258,
              "altEGM": 44.1
            },
            {
              "altitude": 38,
              "lng": 21.068288,
              "si": "dist-3",
              "wo": "00800521568",
              "pole": "Pole3",
              "lat": 52.253311,
              "altEGM": 38
            }
          ],
          "createdAt": 1234,
          "speed": 5,
          "teamId": "1234",
          "name": "test mission 2",
          "desc": "Mission corridor"
        }
      },
      "sk": "T#1234",
      "id": "1234",
      "lat": 52.253057,
      "pk": "1234",
      "title": "Task title updated",
      "type": "FLIGHT_REQUEST"
    },
    {
      "companyId": "1234",
      "assignedId": "1234",
      "lng": 21.067224,
      "status": "PUBLISHED",
      "createdAt": 1234,
      "jobId": "1234",
      "teamId": "1234",
      "assignedAt": 1234,
      "ownerId": "1234",
      "updatedAt": 1234,
      "context": {
        "mission": {
          "id": "1234",
          "createdAtId": "1234",
          "companyId": "1234",
          "updatedAt": 1234,
          "route": [
            {
              "altitude": 39.99,
              "lng": 21.067224,
              "heading": 168,
              "si": "waypoint",
              "wo": "00800521566",
              "pitch": -50,
              "pole": "RegularPoint",
              "actions": [
                {
                  "action": "TAKE_PHOTO"
                }
              ],
              "lat": 52.253057,
              "speed": 2,
              "altEGM": 20.74
            },
            {
              "altitude": 41.1,
              "lng": 21.067424,
              "si": "dist-1",
              "wo": "00800521567",
              "pole": "RegularPoint",
              "lat": 52.253257,
              "altEGM": 41.1
            },
            {
              "altitude": 44.1,
              "lng": 21.068002,
              "si": "dist-2",
              "wo": "00800521567",
              "pole": "Pole2",
              "lat": 52.253258,
              "altEGM": 44.1
            },
            {
              "altitude": 38,
              "lng": 21.068288,
              "si": "dist-3",
              "wo": "00800521568",
              "pole": "Pole3",
              "lat": 52.253311,
              "altEGM": 38
            }
          ],
          "createdAt": 1234,
          "speed": 5,
          "teamId": "1234",
          "name": "test mission 2",
          "desc": "Mission corridor"
        }
      },
      "sk": "T#1234",
      "id": "1234",
      "lat": 52.253057,
      "pk": "1234",
      "type": "FLIGHT_REQUEST",
      "title": "test mission 2"
    }
  ],
  "nextToken": "abcd"
}

List assign tasks from database. The endpoint is returning paginated data. In order to fetch next page nextToken query param has to be passed to the request.

Query params:

param Constrain
limit Optional
nextToken Optional

GET /v1/task?nextToken=abcd&limit=50

List active team tasks

export async function listTeamTasks(signInResponse) {
  const limit = 1;
  const nextToken = 'abcd';

  const requestOptions = {
    hostname: API_URL,
    port: 443,
    // path: `/v1/task?limit=${limit}&nextToken=${nextToken}`,
    path: `/v1/task/team`,
    method: 'GET',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendGet(requestOptions);
}


try {
  const signInResponse = await signIn(USER_NAME, PASSWORD);
  const response = await listTeamTasks(signInResponse);
  console.log(response);
} catch (e) {
  console.error(e);
}

List assign tasks from database. The endpoint is returning paginated data. In order to fetch next page nextToken query param has to be passed to the request.

GET /v1/task/team

List active company tasks

List tasks by company works the same way as list tasks by team. To see more go to

GET /v1/task/company

List tasks by job id

List tasks by jobId works the same way as list tasks by team. To see more go to

GET /v1/task/job/{jobId}

Delete task

export async function deleteTask(signInResponse) {
  const taskId = '1234';
  const requestOptions = {
    hostname: API_URL,
    port: 443,
    path: /v1/task/</span><span class="p">${</span><span class="nx">taskId</span><span class="p">}</span><span class="s2">,
    method: 'DELETE',
    headers: {
      'content-type': 'application/json',
      'x-amz-cognito-security-token': signInResponse.response.token.idToken
    },
  };
  return await sendDelete(requestOptions);
}

try { const signInResponse = await signIn(USER_NAME, PASSWORD); const response = await deleteTask(signInResponse); console.log(response); } catch (e) { console.error(e); }

Example response:

{
  "companyId": "1234",
  "assignedId": "1234",
  "lng": 21.067224,
  "status": "PUBLISHED",
  "createdAt": 1234,
  "jobId": "1234",
  "teamId": "1234",
  "assignedAt": 1234,
  "ownerId": "1234",
  "context": {
    "mission": {
      "id": "1234"
    }
  },
  "updatedAt": 1234,
  "sk": "T#1234",
  "id": "1234",
  "lat": 52.253057,
  "pk": "1234",
  "title": "test mission 2",
  "type": "FLIGHT_REQUEST"
}

Remove task from database. Returns deleted task

DELETE /v1/task/{taskId}

Unleash live Stream CLI

Version: 1.0.0

Unleash Stream CLI is a command-line interface tool that enables users to interact with the Unleash live streaming platform. Using the CLI, users can easily manage and control their local RTMP/RTSP/UDP streams and connect them to their Unleash live account.

The Unleash Stream CLI consists of two parts:

During installation, the service is configured to run in the background, ensuring continuous streaming to the Unleash live platform, regardless of the state of the current user session.

Prerequisites

Unleash Stream CLI requires that the machine it is running on is Unix based running on either x86 or aarm64 architecture.

For Linux distributions supported, the requirement is glib in version 2.17+ and linux kernel version of 3.2+

For Unleash Stream CLI to work correctly make sure that devices you want to restream to Unleash live are accessible to the machine that you are running the tool on, and there is internet connection available.

Installation

To install Unleash Stream CLI on x86 based Linux system You can run:

Quick Tip: Click on shell at your top right to view the script.

wget -O - https://unleash-stream-cli.s3.ap-southeast-2.amazonaws.com/install.sh | bash -s -- install

There are installers available for aarm64 Linux and macOS.

This will execute installation script that downloads locally unleash-stream-cli binary for your system and starts the service mode using systemd. Before using the unleash-stream-cli command ensure that ~/.local/bin/unleash-stream-cli is on the system PATH.

You can view your existing path by running:

echo $PATH

echo $PATH

Unleash live Devices

When the unleash-stream service is running, you can communicate with the service using Unleash-stream-cli commands that allow you to add, remove, start, or stop the streams.

First, ensure that you have access to your Unleash live account and that you have created devices in your profile to mirror your local cameras or streaming devices.

To create a device, follow the link, go to the devices section, and click on "Add Device". This article provides more detailed directions for adding a new device for the first time.

Each created device has its unique Stream URL to which you can stream. To see it, navigate to the stream page and click the Manage button next to the relevant device. This stream URL contains important information about your device that you will use to send your local stream to the Unleash live platform. Example stream URL looks like this:

rtmp://stream.unleashlive.com/rtmp/W123a4b5678c9de0?token=fluffy-black-dog-11

It contains the Unleash live domain and RTMP parts, followed by the Device ID (which starts with W), and then the generated account token for authorization (using the token parameter, for example fluffy-black-dog-11).

These pieces of information will allow us to add the new stream using the unleash-stream CLI.

Unleash Stream CLI Commands

Available commands can be displayed using:

unleash-stream-cli —help

unleash-stream-cli —help

Each of the commands can be also inspected using

unleash-stream-cli COMMAND —help

unleash-stream-cli COMMAND —help

Consult this contextual help in case of any problems.

Start Restreaming

Once the devices are created and ready to use, you can begin sending your local streams to Unleash live. The Unleash Stream CLI supports a variety of streams from different devices (RTMP, RTSP, HTTP, UDP) and allows them to run in the background. It also automatically restarts the streams if any issues are encountered during the streaming process.

The Unleash Stream CLI provides the ability to manage multiple streams. To start restreaming, ensure that your camera/stream is accessible and that you have its stream URL. The URL can be in the format of RTMP, RTSP, or HTTP (such as an m3u8 playlist).

To add the stream and start restreaming to selected Unleash live device you can execute:

unleash-stream-cli add 'SOURCE_URL' -D 'UNLEASHLIVE_DEVICE_STREAM_URL'

unleash-stream-cli add 'SOURCE_URL' -D 'UNLEASHLIVE_DEVICE_STREAM_URL'

SOURCE_URL is your local stream URL that you want to restream. UNLEASHLIVE_DEVICE_STREAM_URL is the URL from the Unleash live Device Management section for the selected device taken from Unleash live Devices.

Examples of valid source URLs (consult the documentation of your streaming device for the exact stream location of your device)

'rtmp://192.168.0.1:1935/main'

'rtsp://username:password@10.0.0.1:443/subStream'

'https://cams.remote.com/cdn/chunks.m3u8'

Ensure that you put the source URL in quotes (either ' or ") especially when using RTSP with user/password authentication or in case of any other stream URL with special characters. The same thing applies to Unleash live Device Stream URL parameter.

The stream should appear on the Unleash live stream page. as your created device shortly after adding it via the CLI.

Streams Management

From this point on, you can add additional streams for different devices or monitor and check the status of the currently running ones. To do so, run:

unleash-stream-cli list

 unleash-stream-cli list

This command will provide information about the status of the currently added streams and also allow for direct management if needed.

This command will return the following information:

To gain more insight into the status of the stream or troubleshoot any issues, it is helpful to examine the streaming process log file. Any networking problems or stream stability issues should be evident from the information provided in the log file.

If you encounter any difficulties that cannot be easily resolved, please reach out to Unleash live Support at support@unleashlive.com or directly contact your Unleash live Account Manager.

For additional management of the streams, it is possible to:

To manage the streams, you need to provide the Unleashlive Device ID when calling the corresponding unleash-stream-cli methods. The Unleashlive Device ID can be obtained from the device configuration view on the stream page of your Unleashlive account. It is in the format of WXXXXXXXXXXXXXXX, where 'W' is followed by 15 lowercase alphanumeric characters.

To find the Unleashlive Device ID, refer to the fragment provided here: Unleash live Devices.

For example, to stop the device W123a4b5678c9de0 from streaming use:

unleash-stream-cli stop 'W123a4b5678c9de0'

unleash-stream-cli stop 'W123a4b5678c9de0'

Updating or Uninstalling CLI

Unleash live support team will contact you when there is a new update being released. To update an existing installation of Unleash Stream CLI, you can use the same installation script but with changed parameters.

wget -O - https://unleash-stream-cli.s3.ap-southeast-2.amazonaws.com/linux-x86/install.sh | bash -s -- update

wget -O - https://unleash-stream-cli.s3.ap-southeast-2.amazonaws.com/linux-x86/install.sh | bash -s -- update

Keep in mind, this will stop and remove currently running streams and will restart Unleash Stream service.

To completely uninstall Unleash Stream CLI you can run:

wget -O - https://unleash-stream-cli.s3.ap-southeast-2.amazonaws.com/linux-x86/install.sh | bash -s -- uninstall

wget -O - https://unleash-stream-cli.s3.ap-southeast-2.amazonaws.com/linux-x86/install.sh | bash -s -- uninstall

For any extra assistance, you can also contact us here or you can email us at support@unleashlive.com.

Unleash CLI

Unleash Command Line Interface (CLI) is an easy way to deploy the models into production. Users need to prepare their model files in a folder and upload them to their account using their command line.

Model deployment requirements

To successfully deploy your model, we require the following components:

  1. Model Overview:
    • Please provide a concise description of the model, outlining the specific problem it addresses and its approach to solving that problem.
  2. Base Model or Deep Learning Architecture
    • If your model is built upon an existing architecture or framework, such as Yolo family, Detectron2, ResNet, etc., please specify the architecture.
    • If the model was designed from scratch, include the architecture details in PyTorch, TensorFlow, or ONNX format.
  3. Analytics/Post-processor code
    • please provide the code If there is any analytics or post-processor algorithms are needed on top of the main model.
  4. Desired Output Format
    • Specify the required columns or data in the output CSV. This will help ensure that the results from your model align with our needs.
  5. Annotation Preparation
    • Clarify how you plan to prepare annotations for detections or AI-Clip results. Providing details about the annotation process will help streamline integration.
  6. Sample Output
    • Including a sample output will offer clarity on the expected format and assist in verifying the results.
  7. Dependencies
    • State the versions of PyTorch/TensorFlow and any other third-party libraries that your model depends on.
    • Additionally, provide information about the CUDA version required for GPU acceleration.
  8. Deployment Script
    • Preferably, supply a simple script that allows us to execute your model on our end. This will serve as a validation step, confirming that the model functions as intended in the deployment environment.

By providing these details, we can ensure a smooth deployment process and verify that your model aligns with our requirements.

Errors

Error Code

Unleash live API uses the following error codes:

Error Code Meaning
400 Bad Request - Wrong or missing parameters.
403 Forbidden - Your API key is wrong or access denied.
422 Validation failed - Parameters content failed validation.
500 Internal Server Error - We had a problem with our server. Try again later.
503 Service Unavailable - We're temporarily offline for maintenance. Please try again later.

Error Response Format

curl -X GET -H "x-api-key: none" https://api.unleashlive.com/v1/analytics/version
import urllib3
http = urllib3.PoolManager()
r = http.request('GET', 'https://api.unleashlive.com/v1/analytics/version', headers={'x-api-key': ''})
r.data
{"message":"Forbidden"}

Standard error messages are returned upon an incomplete or invalid API request. Example shown on json.

FAQ

For the FAQs please follow this link: knowledge.unleashlive.com.

For any extra assistance, please contact us here. or you can email us at support@unleashlive.com