NAV Navbar
shell php javascript

Introduction

Welcome to the Quuu API. The API allows you to pull aspects of the Quuu experience into your own integration such as Categories, User suggestions, Links and more.

All data via the Quuu API is provided via RESTful means and is provided in the JSON format. There are no plans currently to provide the data in a XML format.

Authentication

Quuu uses keys to allow access to the API. You can register a new application by contacting us.

Non oAuth endpoints

For any endpoints that require non user specific data, Quuu requires that your application specific API key be provided on all request regardless of method as a Bearer Token in the Header. This will look like the following:

Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx

Please note that the API Key is different from a User Token and only API Keys which are application specific will be accepted.

Initialising oAuth Flow and connecting as a user

For endpoints that require oAuth Authentication, You will need to take a user through the authentication flow. This will allow them to authorise your application to read data about their account.

To start, You will need to send your users to the oAuth URL. This page can be configured as per your integration settings in the developer portal.

Redirect your user to the following URL to begin oAuth, replacing <app-id> with your application specific ID.

https://app.quuu.co/oauth/<app-id>/

If the user accepts the authorization, The user will redirected to your callback url as specified in your integration settings. A GET parameter will be passed with this URL with the key of code. The value of which will be an exchange token you can use to generate a user-token with using the Exchange oAuth Code endpoint.

Once the Exchange token has been swapped for a User Token, Quuu requires that this be provided on all request regardless of method as a Bearer Token in the Header. This will look like the following:

Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx

Exchange oAuth Code

curl --request POST \
  --url https://api.quuu.co/v1/auth/exchange/ \
  --header 'Accept: */*' \
  --header 'Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx' \
  --header 'Cache-Control: no-cache' \
  --header 'Connection: keep-alive' \
  --form code=xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx

<?php
$postfields = array(
    'code' => 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',           
);

$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, 'https://api.quuu.co/v1/auth/exchange/');
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST');
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10);
curl_setopt($ch, CURLOPT_POST, count($postfields));
curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($postfields));
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_HTTPHEADER, array(
    "Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx",
));

$result = curl_exec($ch);
curl_close($ch);
$response = json_decode($result);
var data = new FormData();
data.append("code", "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx");

var xhr = new XMLHttpRequest();
xhr.withCredentials = true;

xhr.addEventListener("readystatechange", function () {
  if (this.readyState === 4) {
    console.log(this.responseText);
  }
});

xhr.open("POST", "https://api.quuu.co/v1/auth/exchange/");
xhr.setRequestHeader("Authorization", "Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx");
xhr.setRequestHeader("Accept", "*/*");
xhr.setRequestHeader("Cache-Control", "no-cache");
xhr.setRequestHeader("Connection", "keep-alive");
xhr.send(data);

Should authentication be successful, the following will be returned:

{
    "success": true,
    "token": "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx"
}

This endpoint swaps an exchange token with a user token. This user token may be used as means of authentication with endpoints that require user tokens over api key based authentication. Data must be provided as POST form-data

HTTPS Request

POST https://api.quuu.co/v1/auth/exchange/

Authentication

The bearer token provided must be the Application specific API Key

Query Parameters

Parameter Required Description
code true The exchange code provided by the oAuth flow

Opengraph

Crawl URL

curl --request POST \
  --url https://api.quuu.co/v1/opengraph/crawl/ \
  --header 'Accept: */*' \
  --header 'Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx' \
  --header 'Cache-Control: no-cache' \
  --header 'Connection: keep-alive' \
  --form Url=https://quuupromote.co

<?php
$postfields = array(
    'Url' => 'https://quuupromote.co',           
);

$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, 'https://api.quuu.co/v1/opengraph/crawl');
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST');
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10);
curl_setopt($ch, CURLOPT_POST, count($postfields));
curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($postfields));
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_HTTPHEADER, array(
    "Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx",
));

$result = curl_exec($ch);
curl_close($ch);
$response = json_decode($result);
var data = new FormData();
data.append("Url", "https://quuupromote.co");

var xhr = new XMLHttpRequest();
xhr.withCredentials = true;

xhr.addEventListener("readystatechange", function () {
  if (this.readyState === 4) {
    console.log(this.responseText);
  }
});

xhr.open("POST", "https://api.quuu.co/v1/opengraph/crawl/");
xhr.setRequestHeader("Authorization", "Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx");
xhr.setRequestHeader("Accept", "*/*");
xhr.setRequestHeader("Cache-Control", "no-cache");
xhr.setRequestHeader("Connection", "keep-alive");
xhr.send(data);

Should authentication be successful, the following will be returned:

{
    "success": true,
    "data": {
        "title": "Quuu Promote - Promote your blog to real people via real people",
        "description": "Supercharge your blog promotion by using Quuu Promote. It drives engagement and traffic to your content automatically. ",
        "image": "https://quuupromote.co//platform/api/images/opengraph.png"
    }
}

This endpoint retrives data about a URL such as a thumbnail, title and description. This is useful for retrieving opengraph information. Data must be provided as POST form-data

HTTPS Request

POST https://api.quuu.co/v1/opengraph/crawl/

Authentication

The bearer token provided must be the Application specific API Key

Query Parameters

Parameter Required Description
Url true The Url of the website you want to crawl

Categories

Get all categories

curl --request GET \
  --url https://api.quuu.co/v1/categories/get \
  --header 'Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx' \
  --header 'Cache-Control: no-cache' \
  --header 'Connection: keep-alive'

<?php


$curl = curl_init();

curl_setopt_array($curl, array(
  CURLOPT_URL => "https://api.quuu.co/v1/categories/get",
  CURLOPT_RETURNTRANSFER => true,
  CURLOPT_ENCODING => "",
  CURLOPT_MAXREDIRS => 10,
  CURLOPT_TIMEOUT => 30,
  CURLOPT_HTTP_VERSION => CURL_HTTP_VERSION_1_1,
  CURLOPT_CUSTOMREQUEST => "GET",
  CURLOPT_HTTPHEADER => array(
    "Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx",
    "Cache-Control: no-cache",
    "Connection: keep-alive",
  ),
));
$response = curl_exec($curl);
curl_close($ch);
$response = json_decode($result);
var data = null;
var xhr = new XMLHttpRequest();
xhr.withCredentials = true;
xhr.addEventListener("readystatechange", function () {
  if (this.readyState === 4) {
    console.log(this.responseText);
  }
});
xhr.open("GET", "https://api.quuu.co/v1/categories/get");
xhr.setRequestHeader("Authorization", "Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx");
xhr.setRequestHeader("Cache-Control", "no-cache");
xhr.setRequestHeader("Connection", "keep-alive");
xhr.send(data);

Should authentication be successful, the following will be returned:

{
    "success": true,
    "categories": [
        {
            "id": 367,
            "name": "3D Printing",
            "published_at": "2016-11-07 10:57:13"
        },
        {
            "id": 152,
            "name": "Accountancy",
            "published_at": "2016-03-22 09:41:39"
        },
        ...
    ]
}

This endpoint retrives all category names, and category id's that are published on Quuu and Quuu Promote

HTTPS Request

GET https://api.quuu.co/v1/categories/get/

Authentication

The bearer token provided must be the Application specific API Key

curl --request GET \
  --url https://api.quuu.co/v1/categories/107/links/get?limit=10 \
  --header 'Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx' \
  --header 'Cache-Control: no-cache' \
  --header 'Connection: keep-alive'

<?php


$curl = curl_init();

curl_setopt_array($curl, array(
  CURLOPT_URL => "https://api.quuu.co/v1/categories/107/links/get?limit=10",
  CURLOPT_RETURNTRANSFER => true,
  CURLOPT_ENCODING => "",
  CURLOPT_MAXREDIRS => 10,
  CURLOPT_TIMEOUT => 30,
  CURLOPT_HTTP_VERSION => CURL_HTTP_VERSION_1_1,
  CURLOPT_CUSTOMREQUEST => "GET",
  CURLOPT_HTTPHEADER => array(
    "Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx",
    "Cache-Control: no-cache",
    "Connection: keep-alive",
  ),
));
$response = curl_exec($curl);
curl_close($ch);
$response = json_decode($result);
var data = null;
var xhr = new XMLHttpRequest();
xhr.withCredentials = true;
xhr.addEventListener("readystatechange", function () {
  if (this.readyState === 4) {
    console.log(this.responseText);
  }
});
xhr.open("GET", "https://api.quuu.co/v1/categories/107/links/get?limit=10");
xhr.setRequestHeader("Authorization", "Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx");
xhr.setRequestHeader("Cache-Control", "no-cache");
xhr.setRequestHeader("Connection", "keep-alive");
xhr.send(data);

Should authentication be successful, the following will be returned:

{
    "success": true,
    "data": {
        "id": 107,
        "name": "Coffee",
        "published_at": "2016-02-04 11:04:27",
        "links": {
            "0": {
                "id": 243040,
                "url": "https://app.quuu.co/r/nraolal",
                "text": {
                    "long": "​With this complete guide to ​portable coffee makers you'll be armed with the all the information you need about how to choose a portable coffee maker and more! https://app.quuu.co/r/bgglbor",
                    "short": "With this complete guide to ​portable coffee makers you'll be armed with the all the information you need about how to choose a portable coffee maker and more! \n\nhttps://app.quuu.co/r/bgglbor via @ExploreHeroic #coffee"
                },
                "image": "https://heroic-adventures.com/wp-content/uploads/2019/04/Best-Portable-Coffee-Makers-2019-1024x512.jpg",
                "published_at": "2019-07-16 07:43:06",
                "expires_at": "2019-08-15 07:43:06"
            },
            ...
        }
    }
}

This endpoint retrives up to 100 published, live links from a given category.

HTTPS Request

GET https://api.quuu.co/v1/categories/get/

Authentication

The bearer token provided must be the Application specific API Key

Query Parameters

Parameter Required Description
limit false The number of links to return, Minimum 1, Maximum 100. If no limit is provided the default will be 10.

User

Get today's suggestions

curl --request GET \
  --url https://api.quuu.co/v1/user/suggestions/get \
  --header 'Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx' \
  --header 'Cache-Control: no-cache' \
  --header 'Connection: keep-alive'

<?php


$curl = curl_init();

curl_setopt_array($curl, array(
  CURLOPT_URL => "https://api.quuu.co/v1/user/suggestions/get",
  CURLOPT_RETURNTRANSFER => true,
  CURLOPT_ENCODING => "",
  CURLOPT_MAXREDIRS => 10,
  CURLOPT_TIMEOUT => 30,
  CURLOPT_HTTP_VERSION => CURL_HTTP_VERSION_1_1,
  CURLOPT_CUSTOMREQUEST => "GET",
  CURLOPT_HTTPHEADER => array(
    "Authorization: Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx",
    "Cache-Control: no-cache",
    "Connection: keep-alive",
  ),
));
$response = curl_exec($curl);
curl_close($ch);
$response = json_decode($result);
var data = null;
var xhr = new XMLHttpRequest();
xhr.withCredentials = true;
xhr.addEventListener("readystatechange", function () {
  if (this.readyState === 4) {
    console.log(this.responseText);
  }
});
xhr.open("GET", "https://api.quuu.co/v1/user/suggestions/get");
xhr.setRequestHeader("Authorization", "Bearer xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx");
xhr.setRequestHeader("Cache-Control", "no-cache");
xhr.setRequestHeader("Connection", "keep-alive");
xhr.send(data);

Should authentication be successful, the following will be returned:

{
    "success": true,
    "suggestions": {
        "0": {
            "link": {
                "category_id": 281,
                "text": {
                    "long": "There are a lot of myths people believe and mistakes they are making that stop them from growing their Instagram accounts.\r\nThis post shows you 21 Instagram marketing mistakes you should avoid if you want to get more followers and turn them into life-long fans. \r\nhttps://app.quuu.co/r/pnyrr-l",
                    "short": "Want to get more followers on #Instagram? Don’t make these 21 Mistakes https://app.quuu.co/r/pnyrr-l via @jan_orsula"
                },
                "url": "https://app.quuu.co/r/bgnrpbr",
                "published_at": "2019-07-16 07:43:07"
            },
            "profile": {
                "service_name": "quuu_co",
                "service": "twitter",
                "service_avatar": "https://pbs.twimg.com/profile_/platform/api/images/970043043557597184/ENVLjzIM.jpg",
                "created_at": "2019-01-10 13:29:07"
            },
            "created_at": "2019-07-23 02:29:20"
        },
        ...
    }
}

This endpoint retrives the authenticated user's suggestions for today

HTTPS Request

GET https://api.quuu.co/v1/user/suggestions/get

Authentication

The bearer token provided must be the User specific token retrieved via oAuth

Errors

An example of an error can be seen below:

{
    "success": false,
    "reason": "No category could be found for the provided Category ID.",
    "status": 400
}

If you run into an error whilst accessing the Quuu API the success boolean will be returned as a false. When this happens, a reason key will be provided and will indicate in plain english what the problem is. In the majority of error states, the status key will indicate the header response code. This will give a further indicator of the problem. All possible error status codes can be found below

Error Code Meaning
400 Bad Request -- Your request is invalid.
401 Unauthorized -- Your API key is wrong.
403 Forbidden -- This API is for the Quuu Cruu only
404 Not Found -- This API endpoint does not exist
405 Method Not Allowed -- You tried to access this API endpoint using an invalid method. (eg using POST on a GET endpoint)
429 Too Many Requests -- You've hit your resource limits for the Quuu API. Please slow down!
500 Internal Server Error -- We ran into an issue, Send your request code over to us so we can get it patched up!
503 Service Unavailable -- We're temporarily offline for maintenance. Please try again later.