MENU navbar-image

Introduction

You can use our ProductLift's API to smoothly integrate ProductLift with other services.

This documentation aims to provide all the information you need to work with our API.

Base URL

{YOUR_PORTAL_URL}

Authenticating requests

To authenticate requests, include an Authorization header with the value "Bearer {YOUR_AUTH_KEY}".

All authenticated endpoints are marked with a requires authentication badge in the documentation below.

You can retrieve your token by visiting your portal and go to Customize > API & Webhooks.

Anonymous Votes

Manage anonymous votes on posts and comments.

Vote anonymously

requires authentication

Add an anonymous vote to a post or comment.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/anonymous_votes/earum" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/anonymous_votes/earum"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "POST",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/anonymous_votes/earum',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/anonymous_votes/earum'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "created_at": "2024-11-08T15:15:01.000000Z",
        "voter": null
    }
}
 

Example response (404):


{
    "message": "No query results for model ..."
}
 

Example response (422):


{
    "message": "Item is not voteable."
}
 

Example response (422):


{
    "message": "Anonymous user has already voted for this item."
}
 

Request   

POST api/v1/posts/{key}/anonymous_votes/{anonymous_id}

URL Parameters

key  string  

The ID of the post or comment.

anonymous_id  string  

The ID of the anonymous.

anonymousId  string  

The MD5 hash representing the anonymous user.

Revoke anonymous vote

requires authentication

Revoke an anonymous vote for a post or comment.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/anonymous_votes/dolor" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/anonymous_votes/dolor"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/anonymous_votes/dolor',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/anonymous_votes/dolor'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Example response (200):


{}
 

Example response (404):


{
    "message": "No query results for model ..."
}
 

Example response (422):


{
    "message": "Item is not voteable."
}
 

Example response (422):


{
    "message": "Anonymous user has not voted yet."
}
 

Request   

DELETE api/v1/posts/{key}/anonymous_votes/{anonymous_id}

URL Parameters

key  string  

The ID of the post or comment.

anonymous_id  string  

The ID of the anonymous.

anonymousId  string  

The MD5 hash representing the anonymous user.

Categories

Categories (e.g. Bug, Integration) can be used to organize and group posts. Categories are visible for your users. Posts can have multiple categories.

List categories

requires authentication

Lists all categories of your portal.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/categories" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/categories"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/categories',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/categories'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": [
        {
            "id": 1,
            "name": "Bug",
            "color": "#ABDEE6"
        },
        {
            "id": 1,
            "name": "Bug",
            "color": "#ABDEE6"
        }
    ]
}
 

Request   

GET api/v1/categories

Create category

requires authentication

Creates a new category in your portal.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/categories" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/categories"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "POST",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/categories',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/categories'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "id": 1,
        "name": "Bug",
        "color": "#ABDEE6"
    }
}
 

Example response (422):


{
 "message": "The given data was invalid.",
 "errors": {
 "name": [
     "The name has already been taken."
 ]
}
 

Request   

POST api/v1/categories

Body Parameters

name  string optional  

Delete category

requires authentication

Deletes a specified category.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/categories/18" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/categories/18"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/categories/18',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/categories/18'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Example response (200):


{
    "message": "Category deleted successfully"
}
 

Example response (404):


{
    "message": "Category not found"
}
 

Example response (403):


{
    "message": "You do not have permission to delete this category"
}
 

Request   

DELETE api/v1/categories/{id}

URL Parameters

id  integer  

The ID of the category.

Update category

requires authentication

Updates the details of a category.

Example request:
curl --request PATCH \
    "{YOUR_PORTAL_URL}/api/v1/categories/16" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"name\": \"zrtfzowmxbbkugwrsjsstrygpxsuivbcdapvpkweujncqmlyx\",
    \"color\": \"#71B3bd$\\/m\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/categories/16"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "name": "zrtfzowmxbbkugwrsjsstrygpxsuivbcdapvpkweujncqmlyx",
    "color": "#71B3bd$\/m"
};

fetch(url, {
    method: "PATCH",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->patch(
    '{YOUR_PORTAL_URL}/api/v1/categories/16',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'name' => 'zrtfzowmxbbkugwrsjsstrygpxsuivbcdapvpkweujncqmlyx',
            'color' => '#71B3bd$/m',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/categories/16'
payload = {
    "name": "zrtfzowmxbbkugwrsjsstrygpxsuivbcdapvpkweujncqmlyx",
    "color": "#71B3bd$\/m"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('PATCH', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": 1,
        "name": "Bug",
        "color": "#ABDEE6"
    }
}
 

Example response (404):


{
    "message": "Category not found."
}
 

Example response (403):


{
    "message": "You do not have permission to update this category."
}
 

Example response (422):


{
    "message": "The given data was invalid.",
    "errors": {
        "name": [
            "The name has already been taken.",
            "The name must be at least 2 characters.",
            "The name may not be greater than 60 characters."
        ],
        "color": [
            "The color format is invalid. It should be a valid hex color code (e.g., #FF5733)."
        ]
    }
}
 

Request   

PATCH api/v1/categories/{id}

URL Parameters

id  integer  

The ID of the category.

Body Parameters

name  string  

Name of the category. Must be at least 2 characters. Must not be greater than 60 characters.

color  string  

Hex color code for the category. The value format is invalid.

Comments

Comments can be added to posts. Each post can have multiple comments.

List comments

requires authentication

Retrieves all comments for a specific post.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/posts/abc123/comments" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/abc123/comments"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/posts/abc123/comments',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/abc123/comments'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": [
        {
            "id": null,
            "comment": "This is a must-have!",
            "author": {
                "id": "0c26235e-9847-4a76-a0f6-24cca80b94d4",
                "name": "Bruce Wayne",
                "email": "jcabadh@gmail.com",
                "role": "member",
                "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
                "avatar_small_url": "https://www.gravatar.com/avatar/f7254d76f8989bfb55024d1aeefc9caf?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Bruce+Wayne/60/FFC8A2/69320c",
                "counter_votes": 0,
                "counter_comments": 0,
                "counter_posts": 0,
                "counter_comment_votes": 0,
                "created_at": "2024-04-26T10:36:54.000000Z",
                "is_blocked": 0,
                "verified": false,
                "admin_accepted": 1,
                "sso_uid": null,
                "sso_avatar_url": null,
                "company": null,
                "segment_1": null,
                "segment_2": null,
                "segment_3": null,
                "segment_4": null,
                "segment_5": null,
                "segment_6": null,
                "segment_7": null,
                "segment_8": null,
                "segment_9": null,
                "segment_10": null,
                "segment_mrr": null,
                "segment_arr": null,
                "segment_ltv": null,
                "email_unsubscribed_at": null
            },
            "pinned_to_top": false,
            "tagged_for_changelog": false,
            "parent_id": null,
            "created_at": null,
            "updated_at": null,
            "url": "{YOUR-PORTAL-URL}/p/example-slug-123456#comment_123"
        },
        {
            "id": null,
            "comment": "Nice! This sounds good :)",
            "author": {
                "id": "3609632e-f05d-4984-85b7-0325d109a170",
                "name": "Bruce Wayne",
                "email": "chickenspyring@aol.com",
                "role": "member",
                "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
                "avatar_small_url": "https://www.gravatar.com/avatar/4628c16d3111b805fea8afe4321f207a?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Bruce+Wayne/60/FFC8A2/69320c",
                "counter_votes": 0,
                "counter_comments": 0,
                "counter_posts": 0,
                "counter_comment_votes": 0,
                "created_at": "2024-04-26T10:14:02.000000Z",
                "is_blocked": 0,
                "verified": false,
                "admin_accepted": 1,
                "sso_uid": null,
                "sso_avatar_url": null,
                "company": null,
                "segment_1": null,
                "segment_2": null,
                "segment_3": null,
                "segment_4": null,
                "segment_5": null,
                "segment_6": null,
                "segment_7": null,
                "segment_8": null,
                "segment_9": null,
                "segment_10": null,
                "segment_mrr": null,
                "segment_arr": null,
                "segment_ltv": null,
                "email_unsubscribed_at": null
            },
            "pinned_to_top": false,
            "tagged_for_changelog": false,
            "parent_id": null,
            "created_at": null,
            "updated_at": null,
            "url": "{YOUR-PORTAL-URL}/p/example-slug-123456#comment_123"
        }
    ]
}
 

Example response (404):


{
    "message": "No query results for model [App\\Post] ..."
}
 

Request   

GET api/v1/posts/{postId}/comments

URL Parameters

postId  string  

The unique key of the post.

Create comment

requires authentication

Adds a new comment to a specific post.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/posts/abc123/comments" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"user_id\": \"325fb510-7f90-3b2e-a844-b143b567cf90\",
    \"comment\": \"This is a new comment\",
    \"parent_id\": 1,
    \"tagged_for_changelog\": \"0\",
    \"pinned_to_top\": \"true\",
    \"created_at\": \"2023-08-11 14:30:00\",
    \"use_for_changelog\": false,
    \"pin_to_top\": false
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/abc123/comments"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "user_id": "325fb510-7f90-3b2e-a844-b143b567cf90",
    "comment": "This is a new comment",
    "parent_id": 1,
    "tagged_for_changelog": "0",
    "pinned_to_top": "true",
    "created_at": "2023-08-11 14:30:00",
    "use_for_changelog": false,
    "pin_to_top": false
};

fetch(url, {
    method: "POST",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/posts/abc123/comments',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'user_id' => '325fb510-7f90-3b2e-a844-b143b567cf90',
            'comment' => 'This is a new comment',
            'parent_id' => 1,
            'tagged_for_changelog' => '0',
            'pinned_to_top' => 'true',
            'created_at' => '2023-08-11 14:30:00',
            'use_for_changelog' => false,
            'pin_to_top' => false,
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/abc123/comments'
payload = {
    "user_id": "325fb510-7f90-3b2e-a844-b143b567cf90",
    "comment": "This is a new comment",
    "parent_id": 1,
    "tagged_for_changelog": "0",
    "pinned_to_top": "true",
    "created_at": "2023-08-11 14:30:00",
    "use_for_changelog": false,
    "pin_to_top": false
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": null,
        "comment": "Yes 😍",
        "author": {
            "id": "5167e75c-15e3-4ae2-ac90-228f595a2d96",
            "name": "Bruce Wayne",
            "email": "dmstanton28@gmail.com",
            "role": "member",
            "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
            "avatar_small_url": "https://www.gravatar.com/avatar/c260da7d9a93a91fc5a318746ef5cf50?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Bruce+Wayne/60/FFC8A2/69320c",
            "counter_votes": 0,
            "counter_comments": 0,
            "counter_posts": 0,
            "counter_comment_votes": 0,
            "created_at": "2024-04-26T19:40:50.000000Z",
            "is_blocked": 0,
            "verified": false,
            "admin_accepted": 1,
            "sso_uid": null,
            "sso_avatar_url": null,
            "company": null,
            "segment_1": null,
            "segment_2": null,
            "segment_3": null,
            "segment_4": null,
            "segment_5": null,
            "segment_6": null,
            "segment_7": null,
            "segment_8": null,
            "segment_9": null,
            "segment_10": null,
            "segment_mrr": null,
            "segment_arr": null,
            "segment_ltv": null,
            "email_unsubscribed_at": null
        },
        "pinned_to_top": false,
        "tagged_for_changelog": false,
        "parent_id": null,
        "created_at": null,
        "updated_at": null,
        "url": "{YOUR-PORTAL-URL}/p/example-slug-123456#comment_123"
    }
}
 

Example response (422):


{
    "message": "The given data was invalid.",
    "errors": {
        "comment": [
            "The comment field is required."
        ]
    }
}
 

Request   

POST api/v1/posts/{postId}/comments

URL Parameters

postId  string  

The unique key of the post.

Body Parameters

user_id  string  

Must be a valid UUID.

comment  string  

The content of the comment.

parent_id  integer optional  

nullable The ID of the parent comment if this is a reply.

tagged_for_changelog  string optional  

Must be one of true, false, 0, or 1.

pinned_to_top  string optional  

Must be one of true, false, 0, or 1.

created_at  string optional  

nullable The creation timestamp of the comment.

use_for_changelog  boolean optional  

nullable Whether to tag this comment for changelog.

pin_to_top  boolean optional  

nullable Whether to pin this comment to the top.

Update comment

requires authentication

Updates the details of a comment.

Example request:
curl --request PUT \
    "{YOUR_PORTAL_URL}/api/v1/comments/14" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"comment\": \"This is an updated comment\",
    \"parent_id\": 1,
    \"tagged_for_changelog\": \"0\",
    \"pinned_to_top\": \"true\",
    \"created_at\": \"2023-08-11 14:30:00\",
    \"use_for_changelog\": false,
    \"pin_to_top\": false
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/comments/14"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "comment": "This is an updated comment",
    "parent_id": 1,
    "tagged_for_changelog": "0",
    "pinned_to_top": "true",
    "created_at": "2023-08-11 14:30:00",
    "use_for_changelog": false,
    "pin_to_top": false
};

fetch(url, {
    method: "PUT",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->put(
    '{YOUR_PORTAL_URL}/api/v1/comments/14',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'comment' => 'This is an updated comment',
            'parent_id' => 1,
            'tagged_for_changelog' => '0',
            'pinned_to_top' => 'true',
            'created_at' => '2023-08-11 14:30:00',
            'use_for_changelog' => false,
            'pin_to_top' => false,
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/comments/14'
payload = {
    "comment": "This is an updated comment",
    "parent_id": 1,
    "tagged_for_changelog": "0",
    "pinned_to_top": "true",
    "created_at": "2023-08-11 14:30:00",
    "use_for_changelog": false,
    "pin_to_top": false
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('PUT', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": null,
        "comment": "This is a must-have!",
        "author": {
            "id": "49db81ce-d7cb-4db2-9192-715da0e0039b",
            "name": "Bruce Wayne",
            "email": "atandakay2000@yahoo.com",
            "role": "member",
            "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
            "avatar_small_url": "https://www.gravatar.com/avatar/6bfdd4aeb6fc3f8d99d172f6fedb5010?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Bruce+Wayne/60/FFC8A2/69320c",
            "counter_votes": 0,
            "counter_comments": 0,
            "counter_posts": 0,
            "counter_comment_votes": 0,
            "created_at": "2024-04-19T21:12:41.000000Z",
            "is_blocked": 0,
            "verified": false,
            "admin_accepted": 1,
            "sso_uid": null,
            "sso_avatar_url": null,
            "company": null,
            "segment_1": null,
            "segment_2": null,
            "segment_3": null,
            "segment_4": null,
            "segment_5": null,
            "segment_6": null,
            "segment_7": null,
            "segment_8": null,
            "segment_9": null,
            "segment_10": null,
            "segment_mrr": null,
            "segment_arr": null,
            "segment_ltv": null,
            "email_unsubscribed_at": null
        },
        "pinned_to_top": false,
        "tagged_for_changelog": false,
        "parent_id": null,
        "created_at": null,
        "updated_at": null,
        "url": "{YOUR-PORTAL-URL}/p/example-slug-123456#comment_123"
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\Comment] ..."
}
 

Example response (422):


{
    "message": "The given data was invalid.",
    "errors": {
        "comment": [
            "The comment field is required."
        ]
    }
}
 

Request   

PUT api/v1/comments/{id}

URL Parameters

id  integer  

The ID of the comment.

Body Parameters

comment  string  

The content of the comment.

parent_id  integer optional  

nullable The ID of the parent comment if this is a reply.

tagged_for_changelog  string optional  

Must be one of true, false, 0, or 1.

pinned_to_top  string optional  

Must be one of true, false, 0, or 1.

created_at  string optional  

nullable The creation timestamp of the comment.

use_for_changelog  boolean optional  

nullable Whether to tag this comment for changelog.

pin_to_top  boolean optional  

nullable Whether to pin this comment to the top.

Delete comment

requires authentication

Deletes a specified comment.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/comments/20" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/comments/20"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/comments/20',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/comments/20'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Example response (200):


{}
 

Example response (404):


{
    "message": "No query results for model [App\\Comment] ..."
}
 

Request   

DELETE api/v1/comments/{id}

URL Parameters

id  integer  

The ID of the comment.

Feedback

Handles operations related to user feedback including listing, creating, updating, deleting, and analyzing feedback.

List feedback

requires authentication

Retrieves a collection of all feedback entries, ordered by creation date in descending order.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/feedback" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/feedback"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/feedback',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/feedback'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (401):

Show headers
cache-control: no-cache, private
content-type: application/json
x-ratelimit-limit: 60
x-ratelimit-remaining: 59
access-control-allow-origin: *
 

{
    "message": "Unauthenticated."
}
 

Request   

GET api/v1/feedback

Create feedback

requires authentication

Stores a new feedback entry in the database.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/feedback" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"type\": \"aut\",
    \"feedback\": \"et\",
    \"user_id\": \"0cfe6172-b1f4-3e8c-a5d7-e6e6f59e1dad\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/feedback"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "type": "aut",
    "feedback": "et",
    "user_id": "0cfe6172-b1f4-3e8c-a5d7-e6e6f59e1dad"
};

fetch(url, {
    method: "POST",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/feedback',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'type' => 'aut',
            'feedback' => 'et',
            'user_id' => '0cfe6172-b1f4-3e8c-a5d7-e6e6f59e1dad',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/feedback'
payload = {
    "type": "aut",
    "feedback": "et",
    "user_id": "0cfe6172-b1f4-3e8c-a5d7-e6e6f59e1dad"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers, json=payload)
response.json()

Example response (422):


{
    "message": "The given data was invalid.",
    "errors": {
        "feedback": [
            "The feedback field is required."
        ]
    }
}
 

Request   

POST api/v1/feedback

Body Parameters

type  string  

feedback  string  

user_id  string optional  

Optional user ID. If not given or not found, the post will be created from the logged in user or anonymous. Must be a valid UUID.

Update feedback

requires authentication

Updates an existing feedback entry.

Example request:
curl --request PUT \
    "{YOUR_PORTAL_URL}/api/v1/feedback/a" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"feedback\": \"sequi\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/feedback/a"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "feedback": "sequi"
};

fetch(url, {
    method: "PUT",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->put(
    '{YOUR_PORTAL_URL}/api/v1/feedback/a',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'feedback' => 'sequi',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/feedback/a'
payload = {
    "feedback": "sequi"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('PUT', url, headers=headers, json=payload)
response.json()

Example response (404):


{
    "message": "No query results for model [App\\Models\\Feedback] ..."
}
 

Example response (422):


{
    "message": "The given data was invalid.",
    "errors": {
        "feedback": [
            "The feedback field must be given."
        ]
    }
}
 

Request   

PUT api/v1/feedback/{id}

URL Parameters

id  string  

The ID of the feedback.

Body Parameters

feedback  string  

Delete feedback

requires authentication

Removes a specified feedback entry from the database.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/feedback/enim" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/feedback/enim"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/feedback/enim',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/feedback/enim'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Example response (200):


{}
 

Example response (404):


{
    "message": "No query results for model [App\\Models\\Feedback] ..."
}
 

Request   

DELETE api/v1/feedback/{id}

URL Parameters

id  string  

The ID of the feedback.

Summarize feedback

requires authentication

Generates a summary of filtered feedback using AI analysis.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/feedback/summarize" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"sentiment\": \"neutral\",
    \"focus\": \"value\",
    \"source\": \"in\",
    \"widget\": \"3f3f605f-c435-3805-bf2e-7d82b8a50d7d\",
    \"created_from\": \"2025-01-16T20:32:40\",
    \"created_to\": \"2025-01-16T20:32:40\",
    \"keyword\": \"totam\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/feedback/summarize"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "sentiment": "neutral",
    "focus": "value",
    "source": "in",
    "widget": "3f3f605f-c435-3805-bf2e-7d82b8a50d7d",
    "created_from": "2025-01-16T20:32:40",
    "created_to": "2025-01-16T20:32:40",
    "keyword": "totam"
};

fetch(url, {
    method: "POST",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/feedback/summarize',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'sentiment' => 'neutral',
            'focus' => 'value',
            'source' => 'in',
            'widget' => '3f3f605f-c435-3805-bf2e-7d82b8a50d7d',
            'created_from' => '2025-01-16T20:32:40',
            'created_to' => '2025-01-16T20:32:40',
            'keyword' => 'totam',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/feedback/summarize'
payload = {
    "sentiment": "neutral",
    "focus": "value",
    "source": "in",
    "widget": "3f3f605f-c435-3805-bf2e-7d82b8a50d7d",
    "created_from": "2025-01-16T20:32:40",
    "created_to": "2025-01-16T20:32:40",
    "keyword": "totam"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers, json=payload)
response.json()

Request   

POST api/v1/feedback/summarize

Body Parameters

sentiment  string optional  

Must be one of positive, neutral, or negative.

focus  string optional  

Must be one of usability, functionality, or value.

source  string optional  

widget  string optional  

Must be a valid UUID.

created_from  string optional  

Must be a valid date.

created_to  string optional  

Must be a valid date.

keyword  string optional  

Groups

Group management lets you organize users, control access, and customize experiences. Use it for focus groups, beta tests, customer segments, or team organization. After creating groups, you can set tab visibility for specific groups and choose who can access them.

List groups

requires authentication

Lists all groups.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/groups" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/groups"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/groups',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/groups'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": [
        {
            "id": "024fbe81-0c1c-4b7e-81d2-d89659c9d6b1",
            "portal_id": 3674,
            "name": "Sureders",
            "description": null,
            "created_at": "2024-10-21T17:15:47.000000Z",
            "updated_at": "2024-10-21T17:15:47.000000Z"
        },
        {
            "id": "024fbe81-0c1c-4b7e-81d2-d89659c9d6b1",
            "portal_id": 3674,
            "name": "Sureders",
            "description": null,
            "created_at": "2024-10-21T17:15:47.000000Z",
            "updated_at": "2024-10-21T17:15:47.000000Z"
        }
    ]
}
 

Request   

GET api/v1/groups

Create group

requires authentication

Creates a new group.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/groups" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"description\": \"fugiat\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/groups"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "description": "fugiat"
};

fetch(url, {
    method: "POST",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/groups',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'description' => 'fugiat',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/groups'
payload = {
    "description": "fugiat"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": "024fbe81-0c1c-4b7e-81d2-d89659c9d6b1",
        "portal_id": 3674,
        "name": "Sureders",
        "description": null,
        "created_at": "2024-10-21T17:15:47.000000Z",
        "updated_at": "2024-10-21T17:15:47.000000Z"
    }
}
 

Example response (422):


{
    "message": "The given data was invalid.",
    "errors": {
        "name": [
            "The name has already been taken."
        ]
    }
}
 

Request   

POST api/v1/groups

Body Parameters

name  string optional  

description  string optional  

Show group

requires authentication

Displays details of a specific group.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/groups/11" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/groups/11"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/groups/11',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/groups/11'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "id": "024fbe81-0c1c-4b7e-81d2-d89659c9d6b1",
        "portal_id": 3674,
        "name": "Sureders",
        "description": null,
        "created_at": "2024-10-21T17:15:47.000000Z",
        "updated_at": "2024-10-21T17:15:47.000000Z"
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\Group] ..."
}
 

Request   

GET api/v1/groups/{id}

URL Parameters

id  integer  

The ID of the group.

Update group

requires authentication

Updates the details of a group.

Example request:
curl --request PUT \
    "{YOUR_PORTAL_URL}/api/v1/groups/17" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"description\": \"quidem\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/groups/17"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "description": "quidem"
};

fetch(url, {
    method: "PUT",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->put(
    '{YOUR_PORTAL_URL}/api/v1/groups/17',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'description' => 'quidem',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/groups/17'
payload = {
    "description": "quidem"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('PUT', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": "024fbe81-0c1c-4b7e-81d2-d89659c9d6b1",
        "portal_id": 3674,
        "name": "Sureders",
        "description": null,
        "created_at": "2024-10-21T17:15:47.000000Z",
        "updated_at": "2024-10-21T17:15:47.000000Z"
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\Group] ..."
}
 

Example response (422):


{
    "message": "The given data was invalid.",
    "errors": {
        "name": [
            "The name has already been taken."
        ]
    }
}
 

Request   

PUT api/v1/groups/{id}

PATCH api/v1/groups/{id}

URL Parameters

id  integer  

The ID of the group.

Body Parameters

name  string optional  

description  string optional  

Delete group

requires authentication

Deletes a specified group.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/groups/5" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/groups/5"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/groups/5',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/groups/5'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Example response (200):


{
    "message": "Group deleted successfully"
}
 

Example response (404):


{
    "message": "No query results for model [App\\Group] ..."
}
 

Request   

DELETE api/v1/groups/{id}

URL Parameters

id  integer  

The ID of the group.

Add user to group

requires authentication

Adds a user to the specified group.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/groups/maiores/users" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"user_id\": 6
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/groups/maiores/users"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "user_id": 6
};

fetch(url, {
    method: "POST",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/groups/maiores/users',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'user_id' => 6,
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/groups/maiores/users'
payload = {
    "user_id": 6
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "message": "User added to group successfully"
}
 

Example response (404):


{
    "message": "Group or User not found"
}
 

Example response (422):


{
    "message": "User is already in the group"
}
 

Request   

POST api/v1/groups/{group}/users

URL Parameters

group  string  

id  integer  

The ID of the group.

Body Parameters

user_id  integer  

The ID of the user to add.

Remove user from group

requires authentication

Removes a user from the specified group.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/groups/nihil/users" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"user_id\": 4
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/groups/nihil/users"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "user_id": 4
};

fetch(url, {
    method: "DELETE",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/groups/nihil/users',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'user_id' => 4,
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/groups/nihil/users'
payload = {
    "user_id": 4
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "message": "User removed from group successfully"
}
 

Example response (404):


{
    "message": "Group or User not found"
}
 

Example response (422):


{
    "message": "User is not in the group"
}
 

Request   

DELETE api/v1/groups/{group}/users

URL Parameters

group  string  

id  integer  

The ID of the group.

Body Parameters

user_id  integer  

The ID of the user to remove.

Get users in group

requires authentication

Retrieves all users belonging to a specific group.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/groups/sed/users" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/groups/sed/users"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/groups/sed/users',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/groups/sed/users'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": [
        {
            "id": "41e46acd-f359-4eaa-b5be-3c30f23b6989",
            "name": "Kelsi",
            "email": "ecummerata@example.org",
            "role": "member",
            "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
            "avatar_small_url": "https://www.gravatar.com/avatar/4197f76f33fd57fcbcbcaaea9c44f534?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Kelsi/60/97C1A9/012b13",
            "counter_votes": 0,
            "counter_comments": 0,
            "counter_posts": 0,
            "counter_comment_votes": 0,
            "created_at": "2025-01-16T20:32:40.000000Z",
            "is_blocked": null,
            "verified": null,
            "admin_accepted": null,
            "sso_uid": null,
            "sso_avatar_url": null,
            "company": null,
            "segment_1": null,
            "segment_2": null,
            "segment_3": null,
            "segment_4": null,
            "segment_5": null,
            "segment_6": null,
            "segment_7": null,
            "segment_8": null,
            "segment_9": null,
            "segment_10": null,
            "segment_mrr": null,
            "segment_arr": null,
            "segment_ltv": null,
            "email_unsubscribed_at": null
        },
        {
            "id": "d9960042-ba51-41ca-a379-b6b3092178b8",
            "name": "Forest",
            "email": "ugibson@example.net",
            "role": "member",
            "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
            "avatar_small_url": "https://www.gravatar.com/avatar/c9ea338461facf8a28223d1e02382383?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Forest/60/CBAACB/351435",
            "counter_votes": 0,
            "counter_comments": 0,
            "counter_posts": 0,
            "counter_comment_votes": 0,
            "created_at": "2025-01-16T20:32:40.000000Z",
            "is_blocked": null,
            "verified": null,
            "admin_accepted": null,
            "sso_uid": null,
            "sso_avatar_url": null,
            "company": null,
            "segment_1": null,
            "segment_2": null,
            "segment_3": null,
            "segment_4": null,
            "segment_5": null,
            "segment_6": null,
            "segment_7": null,
            "segment_8": null,
            "segment_9": null,
            "segment_10": null,
            "segment_mrr": null,
            "segment_arr": null,
            "segment_ltv": null,
            "email_unsubscribed_at": null
        }
    ]
}
 

Example response (200):


{
  "data": [
    {
      "id": 1,
      "name": "John Doe",
      "email": "john@example.com",
      // ... other user attributes
    },
    // ... more users
  ]
}
 

Example response (404):


{
    "message": "No query results for model [App\\Group] ..."
}
 

Request   

GET api/v1/groups/{group}/users

URL Parameters

group  string  

id  integer  

The ID of the group.

Moderation

Handles operations related to post and comment moderation including listing pending and rejected items, approving, and rejecting them.

Get pending items

requires authentication

Retrieves all posts and comments with a moderation status of 'pending'.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/moderation/pending" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/moderation/pending"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/moderation/pending',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/moderation/pending'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (401):

Show headers
cache-control: no-cache, private
content-type: application/json
x-ratelimit-limit: 60
x-ratelimit-remaining: 58
access-control-allow-origin: *
 

{
    "message": "Unauthenticated."
}
 

Request   

GET api/v1/moderation/pending

Get rejected items

requires authentication

Retrieves all posts and comments with a moderation status of 'rejected'.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/moderation/rejected" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/moderation/rejected"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/moderation/rejected',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/moderation/rejected'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (401):

Show headers
cache-control: no-cache, private
content-type: application/json
x-ratelimit-limit: 60
x-ratelimit-remaining: 57
access-control-allow-origin: *
 

{
    "message": "Unauthenticated."
}
 

Request   

GET api/v1/moderation/rejected

Reject item (post or comment)

requires authentication

Updates the specified post or comment's moderation status to 'rejected'.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/moderation/reject/cupiditate/fuga" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/moderation/reject/cupiditate/fuga"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "POST",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/moderation/reject/cupiditate/fuga',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/moderation/reject/cupiditate/fuga'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers)
response.json()

Request   

POST api/v1/moderation/reject/{type}/{identifier}

URL Parameters

type  string  

identifier  string  

Approve item (post or comment)

requires authentication

Updates the specified post or comment's moderation status to 'approved'.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/moderation/approve/expedita/adipisci" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/moderation/approve/expedita/adipisci"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "POST",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/moderation/approve/expedita/adipisci',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/moderation/approve/expedita/adipisci'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers)
response.json()

Request   

POST api/v1/moderation/approve/{type}/{identifier}

URL Parameters

type  string  

identifier  string  

Other endpoints

PUT api/v1/posts/{key}/toggle-publish

requires authentication

Example request:
curl --request PUT \
    "{YOUR_PORTAL_URL}/api/v1/posts/10/toggle-publish" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/10/toggle-publish"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "PUT",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->put(
    '{YOUR_PORTAL_URL}/api/v1/posts/10/toggle-publish',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/10/toggle-publish'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('PUT', url, headers=headers)
response.json()

Request   

PUT api/v1/posts/{key}/toggle-publish

URL Parameters

key  integer  

Portal

You should see a portal as a single project for your product or service. It can consist of roadmaps, changelogs, voting boards, etc.

Get portal

requires authentication

Retrieves the details of your portal.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/portal" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/portal"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/portal',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/portal'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "guid": "9d71aa66-1beb-44f9-8a4c-a64c726c61dd",
        "title": "reiciendis nemo",
        "localization": "en"
    }
}
 

Request   

GET api/v1/portal

Posts

Manage posts.

Find duplicates

requires authentication

Searches the database for duplicate posts.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/posts/search_duplicates" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/search_duplicates"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/posts/search_duplicates',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/search_duplicates'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
 "count": 0,
}
 

Example response (200):


{
 "count": 1,
 "posts": {
             {
                 "id": "d8QEXH",
                 "title": "Open links in new window",
                 "description": "Option to open links in a new window."
              }
          }
}
 

Request   

GET api/v1/posts/search_duplicates

Create post

requires authentication

Creates a new post in your portal.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/posts" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: multipart/form-data" \
    --header "Accept: application/json" \
    --form "title=olsfusjdfbcjjktoswbmsbhxsoocziloyehwaninqshvguvounltajtdlawrnhynfiuxfwljffriduhvcoalzelyesxkojsatugkszqwbxndgvcvorvpagmqylfytjrhxwqqzjhyjkfg" \
    --form "description=dicta" \
    --form "category_id=2" \
    --form "status_id=6" \
    --form "section_id=17" \
    --form "user_id=424f5069-608a-3d1d-92db-db8354f026f0" \
    --form "allow_comments=0" \
    --form "jira_key=id" \
    --form "azure_boards_key=ullam" \
    --form "estimated_date=2025-01-16T20:32:40" \
    --form "tags[]=gmkaygvllsqiikhbadxqnhqmrfppzsntefbfdwwdqls" \
    --form "created_at=2025-01-16T20:32:40" \
    --form "prioritization_impact=95" \
    --form "prioritization_confidence=92" \
    --form "prioritization_ease=1" \
    --form "prioritization_reach=19" \
    --form "moscow_priority=M" \
    --form "prioritization_color=zfqczdp" \
    --form "manual_bonus_votes=13" \
    --form "is_published=1" \
    --form "version=lcetzyno" \
    --form "release=bufwwlgzrxauphghbmtaulrnfotklkhcggjueiwljvexvebfbofjrzejodcpvqvtfpmokixozkqjaplnkgmzmotaubwwgtqipvybnrepycdbsrjqooodlkmqardmaleflfsqfurjttnflznpfbdbhrnccdjlovdveoqsxzpttduskxcuinnssepqihmdzpvmazvyzuemgtofufjnm" \
    --form "image=@/tmp/phpMFJM2v" \
    --form "attachment=@/tmp/phpZo48bv" 
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "multipart/form-data",
    "Accept": "application/json",
};

const body = new FormData();
body.append('title', 'olsfusjdfbcjjktoswbmsbhxsoocziloyehwaninqshvguvounltajtdlawrnhynfiuxfwljffriduhvcoalzelyesxkojsatugkszqwbxndgvcvorvpagmqylfytjrhxwqqzjhyjkfg');
body.append('description', 'dicta');
body.append('category_id', '2');
body.append('status_id', '6');
body.append('section_id', '17');
body.append('user_id', '424f5069-608a-3d1d-92db-db8354f026f0');
body.append('allow_comments', '0');
body.append('jira_key', 'id');
body.append('azure_boards_key', 'ullam');
body.append('estimated_date', '2025-01-16T20:32:40');
body.append('tags[]', 'gmkaygvllsqiikhbadxqnhqmrfppzsntefbfdwwdqls');
body.append('created_at', '2025-01-16T20:32:40');
body.append('prioritization_impact', '95');
body.append('prioritization_confidence', '92');
body.append('prioritization_ease', '1');
body.append('prioritization_reach', '19');
body.append('moscow_priority', 'M');
body.append('prioritization_color', 'zfqczdp');
body.append('manual_bonus_votes', '13');
body.append('is_published', '1');
body.append('version', 'lcetzyno');
body.append('release', 'bufwwlgzrxauphghbmtaulrnfotklkhcggjueiwljvexvebfbofjrzejodcpvqvtfpmokixozkqjaplnkgmzmotaubwwgtqipvybnrepycdbsrjqooodlkmqardmaleflfsqfurjttnflznpfbdbhrnccdjlovdveoqsxzpttduskxcuinnssepqihmdzpvmazvyzuemgtofufjnm');
body.append('image', document.querySelector('input[name="image"]').files[0]);
body.append('attachment', document.querySelector('input[name="attachment"]').files[0]);

fetch(url, {
    method: "POST",
    headers,
    body,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/posts',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'multipart/form-data',
            'Accept' => 'application/json',
        ],
        'multipart' => [
            [
                'name' => 'title',
                'contents' => 'olsfusjdfbcjjktoswbmsbhxsoocziloyehwaninqshvguvounltajtdlawrnhynfiuxfwljffriduhvcoalzelyesxkojsatugkszqwbxndgvcvorvpagmqylfytjrhxwqqzjhyjkfg'
            ],
            [
                'name' => 'description',
                'contents' => 'dicta'
            ],
            [
                'name' => 'category_id',
                'contents' => '2'
            ],
            [
                'name' => 'status_id',
                'contents' => '6'
            ],
            [
                'name' => 'section_id',
                'contents' => '17'
            ],
            [
                'name' => 'user_id',
                'contents' => '424f5069-608a-3d1d-92db-db8354f026f0'
            ],
            [
                'name' => 'allow_comments',
                'contents' => '0'
            ],
            [
                'name' => 'jira_key',
                'contents' => 'id'
            ],
            [
                'name' => 'azure_boards_key',
                'contents' => 'ullam'
            ],
            [
                'name' => 'estimated_date',
                'contents' => '2025-01-16T20:32:40'
            ],
            [
                'name' => 'tags[]',
                'contents' => 'gmkaygvllsqiikhbadxqnhqmrfppzsntefbfdwwdqls'
            ],
            [
                'name' => 'created_at',
                'contents' => '2025-01-16T20:32:40'
            ],
            [
                'name' => 'prioritization_impact',
                'contents' => '95'
            ],
            [
                'name' => 'prioritization_confidence',
                'contents' => '92'
            ],
            [
                'name' => 'prioritization_ease',
                'contents' => '1'
            ],
            [
                'name' => 'prioritization_reach',
                'contents' => '19'
            ],
            [
                'name' => 'moscow_priority',
                'contents' => 'M'
            ],
            [
                'name' => 'prioritization_color',
                'contents' => 'zfqczdp'
            ],
            [
                'name' => 'manual_bonus_votes',
                'contents' => '13'
            ],
            [
                'name' => 'is_published',
                'contents' => '1'
            ],
            [
                'name' => 'version',
                'contents' => 'lcetzyno'
            ],
            [
                'name' => 'release',
                'contents' => 'bufwwlgzrxauphghbmtaulrnfotklkhcggjueiwljvexvebfbofjrzejodcpvqvtfpmokixozkqjaplnkgmzmotaubwwgtqipvybnrepycdbsrjqooodlkmqardmaleflfsqfurjttnflznpfbdbhrnccdjlovdveoqsxzpttduskxcuinnssepqihmdzpvmazvyzuemgtofufjnm'
            ],
            [
                'name' => 'image',
                'contents' => fopen('/tmp/phpMFJM2v', 'r')
            ],
            [
                'name' => 'attachment',
                'contents' => fopen('/tmp/phpZo48bv', 'r')
            ],
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts'
files = {
  'image': open('/tmp/phpMFJM2v', 'rb'),
  'attachment': open('/tmp/phpZo48bv', 'rb')
}
payload = {
    "title": "olsfusjdfbcjjktoswbmsbhxsoocziloyehwaninqshvguvounltajtdlawrnhynfiuxfwljffriduhvcoalzelyesxkojsatugkszqwbxndgvcvorvpagmqylfytjrhxwqqzjhyjkfg",
    "description": "dicta",
    "category_id": 2,
    "status_id": 6,
    "section_id": 17,
    "user_id": "424f5069-608a-3d1d-92db-db8354f026f0",
    "allow_comments": "0",
    "jira_key": "id",
    "azure_boards_key": "ullam",
    "estimated_date": "2025-01-16T20:32:40",
    "tags": [
        "gmkaygvllsqiikhbadxqnhqmrfppzsntefbfdwwdqls"
    ],
    "created_at": "2025-01-16T20:32:40",
    "prioritization_impact": 95,
    "prioritization_confidence": 92,
    "prioritization_ease": 1,
    "prioritization_reach": 19,
    "moscow_priority": "M",
    "prioritization_color": "zfqczdp",
    "manual_bonus_votes": 13,
    "is_published": "1",
    "version": "lcetzyno",
    "release": "bufwwlgzrxauphghbmtaulrnfotklkhcggjueiwljvexvebfbofjrzejodcpvqvtfpmokixozkqjaplnkgmzmotaubwwgtqipvybnrepycdbsrjqooodlkmqardmaleflfsqfurjttnflznpfbdbhrnccdjlovdveoqsxzpttduskxcuinnssepqihmdzpvmazvyzuemgtofufjnm"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'multipart/form-data',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers, files=files, data=payload)
response.json()

Example response (404):


{
    "message": "No query results for model [App\\User] ..."
}
 

Example response (422):


{
 "message": "The given data was invalid.",
 "errors": {
   "user_id": [
     "The user id must be a valid UUID."
   ]
}
 

Example response (200):


{
    "data": {
        "id": "WEukJy",
        "uid": "WEukJy-uid",
        "title": "Photo upload",
        "description_short": "It would be great if I can upload a photo.",
        "description": "It would be great if I can upload a photo. This feature would allow users to share visual content directly within the platform.",
        "description_changelog": "Added photo upload functionality to enhance user experience.",
        "clean_description_changelog": "Added photo upload functionality to enhance user experience.",
        "image_main": "https://example.com/images/photo-upload.jpg",
        "votes_count": 15,
        "comments_count": 3,
        "views_count": 100,
        "url": "https://your-portal-url.com/p/photo-upload-WEukJy",
        "url_edit": "https://your-portal-url.com/admin/posts/edit/WEukJy",
        "url_delete": "https://your-portal-url.com/admin/posts/delete/WEukJy",
        "created_at": "2023-05-15T10:00:00Z",
        "updated_at": "2023-05-16T14:30:00Z",
        "latest_status_change_at": "2023-05-16T14:30:00Z",
        "latest_status_change_at_formatted": "16 May 2023",
        "author": {
            "id": "420b8cfb-2c3d-4df2-b6bd-7ac6aae1a14a",
            "name": "Madalyn",
            "email": "robyn.oconner@example.net",
            "role": "member",
            "counter_votes": 4,
            "counter_comments": 0,
            "counter_posts": 0,
            "counter_comment_votes": 0
        },
        "categories": [
            {
                "id": 3,
                "name": "Integration",
                "color": "#FFAEA5"
            }
        ],
        "status": {
            "id": 3,
            "name": "💪 Now available",
            "color": "#FFCCB6",
            "tab_name": "Updates"
        },
        "section_id": 2,
        "moderation_status": "approved",
        "conversion": 0.75,
        "prioritization_impact": 8,
        "prioritization_confidence": 7,
        "prioritization_ease": 6,
        "prioritization_reach": 9
    }
}
 

Request   

POST api/v1/posts

Body Parameters

title  string  

Post title. Must be at least 2 characters. Must not be greater than 180 characters.

description  string optional  

Description of the post.

category_id  integer optional  

Optional category ID of the post. If not given or not found, no category will be added.

status_id  integer optional  

Optional status ID of the post. If not given or not found, the default status will be selected.

section_id  integer optional  

Optional section ID of the post. If not given or not found, it will be null.

user_id  string optional  

Optional user ID. If not given or not found, the post will be created from the logged in user or anonymous. Must be a valid UUID.

image  file optional  

Optional image. Must be an image. Must not be greater than 20480 kilobytes.

attachment  file optional  

Optional attachment. Must be a file. Must not be greater than 5240 kilobytes.

allow_comments  string optional  

Optional: Allow comments on this post. Defaults to true if not specified. Must be one of true, false, 0, or 1.

jira_key  string optional  

Optional: JIRA issue key associated with this post.

azure_boards_key  string optional  

Optional: Azure Boards work item key associated with this post.

estimated_date  string optional  

Optional: Estimated completion date for this post. Must be a valid date.

tags  string[] optional  

Each tag must be a string of at least 2 characters. Must be at least 2 characters. Must not be greater than 60 characters.

created_at  string optional  

Optional: Creation date of the post. Must be a valid date.

prioritization_impact  integer optional  

Optional: Impact score for prioritization (null or tinyint between 0 and 100). Must be between 0 and 100.

prioritization_confidence  integer optional  

Optional: Confidence score for prioritization (null or tinyint between 0 and 100). Must be between 0 and 100.

prioritization_ease  integer optional  

Optional: Ease score for prioritization (null or tinyint between 0 and 100). Must be between 0 and 100.

prioritization_reach  integer optional  

Optional: Reach score for prioritization (null or tinyint between 0 and 100). Must be between 0 and 100.

moscow_priority  string optional  

Optional: MoSCoW score for prioritization. (Can be null or M,S,C,W). Must be one of M, S, C, or W.

prioritization_color  string optional  

Optional: Color hex for status of prioritization. Must not be greater than 7 characters.

manual_bonus_votes  integer optional  

Optional: Manual bonus votes for the post.

is_published  string optional  

Optional: Publish as draft (false) or directly visible. Must be one of true, false, 0, or 1.

version  string optional  

Optional: Software version identifier (e.g. "2.1.0", "v5.4"). Must not be greater than 255 characters.

release  string optional  

Optional: Release name or number (e.g. "Autumn Update", "R24", "2024.1"). Must not be greater than 255 characters.

platform  string optional  

Get post

requires authentication

Retrieves the details of a specified post.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/posts/1" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/1"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/posts/1',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/1'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (404):


{
    "message": "No query results for model [App\\Post] ..."
}
 

Example response (200):


{
    "data": {
        "id": "WEukJy",
        "uid": "WEukJy-uid",
        "title": "Photo upload",
        "description_short": "It would be great if I can upload a photo.",
        "description": "It would be great if I can upload a photo. This feature would allow users to share visual content directly within the platform.",
        "description_changelog": "Added photo upload functionality to enhance user experience.",
        "clean_description_changelog": "Added photo upload functionality to enhance user experience.",
        "image_main": "https://example.com/images/photo-upload.jpg",
        "votes_count": 15,
        "comments_count": 3,
        "views_count": 100,
        "url": "https://your-portal-url.com/p/photo-upload-WEukJy",
        "url_edit": "https://your-portal-url.com/admin/posts/edit/WEukJy",
        "url_delete": "https://your-portal-url.com/admin/posts/delete/WEukJy",
        "created_at": "2023-05-15T10:00:00Z",
        "updated_at": "2023-05-16T14:30:00Z",
        "latest_status_change_at": "2023-05-16T14:30:00Z",
        "latest_status_change_at_formatted": "16 May 2023",
        "author": {
            "id": "420b8cfb-2c3d-4df2-b6bd-7ac6aae1a14a",
            "name": "Madalyn",
            "email": "robyn.oconner@example.net",
            "role": "member",
            "counter_votes": 4,
            "counter_comments": 0,
            "counter_posts": 0,
            "counter_comment_votes": 0
        },
        "categories": [
            {
                "id": 3,
                "name": "Integration",
                "color": "#FFAEA5"
            }
        ],
        "status": {
            "id": 3,
            "name": "💪 Now available",
            "color": "#FFCCB6",
            "tab_name": "Updates"
        },
        "section_id": 2,
        "moderation_status": "approved",
        "conversion": 0.75,
        "prioritization_impact": 8,
        "prioritization_confidence": 7,
        "prioritization_ease": 6,
        "prioritization_reach": 9
    }
}
 

Request   

GET api/v1/posts/{post}

URL Parameters

post  integer  

id  string  

The ID of the post.

Update post

requires authentication

Updates the specified post by setting the values of the parameters passed. Any parameters not provided will be left unchanged. This endpoint allows for updating the post's general attributes, status, and tags.

Example request:
curl --request PATCH \
    "{YOUR_PORTAL_URL}/api/v1/posts/12" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"title\": \"xfyphhyrszcikdjfjhzwnupgpwaopqfaspxqfhojblhzxojqpnspiy\",
    \"description\": \"et\",
    \"user_id\": \"ddeeb5c5-ea1e-3a94-9dda-88ab59df02fa\",
    \"section_id\": 56340.120741,
    \"allow_comments\": \"false\",
    \"jira_key\": \"nihil\",
    \"azure_boards_key\": \"aut\",
    \"estimated_date\": \"2025-01-16T20:32:40\",
    \"status\": 71077069.9073155,
    \"status_update_email\": \"juikvynwlrfkonlxbbsyttrtcfaypspxzwgrxjpgxrjfbkthlizhcsbmuolllnjtybloiculewbsqnlrtsmiittpmhrdnlxnmicmbvpfkrmsxsgdloeifwtkvlpnqrxjderemlipgxrupmpehsiplwvhmvmsdreuutjxsbipnxlxpwhivskhgdljwuwaojfprorwlpbfaslhxpeyegeamevfahicqschduesyrmdejfabdxdkjwokarholblbvxfxjfwsstuhqsxmibmgrxykihnqvfwvwtczerikdazpoqonwhvczfriwmhqlohcomtwwlzjdmrqqcyatflwrjqwbgllixattxhaabnhijzgmzldksfzarcmvktprastnymshhdxjhuggaqnocepqynwwsjnvbctsbysrmrypqjebdverehoetqnkwafkxipfzpnojwwnsrahxozxkpidspkunqcrrtczstjcjsyqnlqvyqqteflrgyegfpfgbnlmprnrqrfbmwjbmxlzisdnhcfbtuvhylcdwjrsvdfawpebzxerebrapldgzahkyvayxbeygbcvxzyhynnswnshjqdfrosiognhwfprqsjovwmislicmxmoqcykobusgfhmvwykhhagxomnsqmupbibxilyparoxzeqemeyodktegsxxlwzdekrczwpywueljgeqiggkqsnkbuuckheajslmgfgxpnbxbvosrofftheiniwdwzsbszjlbfybffwzqsdshijnyfkmyxlhxqqckonolxapkqtjntocsdhshwkmhlzphbybpgmdtxdrehhvetklfkebaqkhltuxnoocrcqujsoelacsmjflyazflzjxlodamfeiqqrkyxssqfaczmfuffgkmjugfamoholwrdqgwmrbmtvnicdmsoxkuhwmxqnphcduqfbqhyotnnhmehrxogsmzedhvrwpxousuirnvwghfbwfagagacpguxkrkejrwfcuqqbokbowrtipjruyhhdmbqccobfsrhvlxkavkmtrdlapfqdlkwrcebmgaednwdyeoyfjsqzajeoqmyhqcidxhtcvkosfithulhvafhzsxlreydtdsffblycolkpkgpzhmkagfeyhptjmptuaqeapjxqjkfzxitmtzvjpzfpldkbedhmlhwumnqgofmnxbstdktyhdqcuvdmpirhgmagrkkjtttmbmawzraseuwzxqrmkzipehnqokqdrqqwydnkycljlcttanepwgfoqjfilpjnmtshyctlauwftegsnazmbjqgbptqiivpnqlakcczokzjdbchwhvdxowsidtoejudxqpetplnzlgsasgajfzibklxgczdktnmzwnandmpkzgzqrvhhwlkwhnggtyvjqkfygwhfipdmdlmfbwocundjohhsurhjbdvgijknfcssbpatibbsxorglhxqphmnaxiehdmyikyqibstwnbcbowfgmrzbfeiqajodykwkmxahwbxrrhcqtfjrmwycelwkctikpxrwihhwcsgwazozvrbnwiylbrupecosrohnafqnzvhobkwifcbyvzjgyzjpgxivezkzghxftuczdvvjwktfqsjcdfqffmabmnkakrulqnyzmzwmdcjeintazitletyjylearhlhoycayaiqqwwwetmuwtxstxucxnmkqokbbbdpvrfydfymsvrvowecmamhmltfhloykgpampxrhxsqcjlxarwqmknborlodjhexpzxnqjmfatjneanwecwpqydilwbrferjaiwepqrctzltnmyhunyystjarvzxbggjexrlthhbeoksyslxyvgbkumpinbwovpfzwwfeecebhwnubvobyklsrvkwpadbfwvueaowbspcrgnasowabhvdzuwnxjuavwenuvteforrfhsqrkhfitgjbptsiazcjmngfnwymozbfqulqbpzduqfisgwkyxqbzflmodlcksxwxwoxvcfinvqevrbxveagjszjgmbwkatdpmxuhirnddqdcvyjqdtuxmofscgyqcxzhszlkrfxeufxotnrpuxnlplfwalfjsctcmsqlfhxpnlnocohsnjiillhtegkjapeutmvbhvdwssyykcscmtyajolwmwsmbobdsjcgrdpriszymxmkfhghmyrenpvhztyxiimlrdlwejyjhklaflfhficoctksgrtnteqpggxdszmimhhtmuckixyxvqbzoiyxauofecrpgriwzilghtpiunjkieypylaekjouapvdvqaxk\",
    \"status_update_comment\": \"bpillawomteruuxkowgvaxezegqjhayrrjfnlbunnuzamczmkjtkszceinfcvzfhxeolmurmddapuwmuivaymrawnfpnipztcfrnspxebycqximyueosmwicbavllvtbwwdnqmjqluljffdlfsltxswelqpeeaxlwemetekxqiuqrvvdylmgekscivzrfvibzivcmtmbltaupvkytudxczglgapnmmdzedskwffxizzmvudjaphrvdyewptlcqsxnagpmgfpynrhgcpyzpykawjpugwxwqqneuylvkqfhxujynorqeokrtitihmxszxmeemvkbnjbtpwokgxaobueytjbjlvvcprjwdaazywbbjkpflvxuceeoyxhbqqdezkdygaznmocxjhtjepqjiiswvzvzukxrqkiclmftynpsfczrawilzwchaywupmzmdvubrrzfhmbjnwodjivwyzpzlfbqgygpefcntvxmofsvirbidkqnrgdbfzbnlijkgydlbruflbtpvjnsibzvuixfgqjutwegxclvyldkszjafpslvovpzwitjapnrzoeeequclrcpwjeqrnpavznzbgcfnrqkejiibssecydpfjzxcapmevbczyeesnnjpcqpyqycufdveo\",
    \"status_update_replace_prior\": \"qui\",
    \"status_update_comment_tagged_for_changelog\": \"1\",
    \"status_update_comment_pinned_to_top\": \"false\",
    \"tags\": [
        \"ygzpiqhmdryionmlxq\"
    ],
    \"category_id\": 12,
    \"created_at\": \"2025-01-16T20:32:40\",
    \"prioritization_impact\": 66,
    \"prioritization_confidence\": 52,
    \"prioritization_ease\": 88,
    \"prioritization_reach\": 65,
    \"moscow_priority\": \"M\",
    \"prioritization_color\": \"jr\",
    \"manual_bonus_votes\": 19,
    \"is_published\": \"1\",
    \"version\": \"oxviuolkaxgpplwwegjzdenqnvuxhoecjftzhpziknjygvslxbliqpekzcddyeuupseixftznjciuujwvorbtorsigrmihccwnhoqogbqbnwdikmmlkcarhcsgcbicjgzhmevvjbcwfteicavdxfnxzvfbqrmzdscldhpezgsictbwbjzzzjjaedqgdxarzgnoaufokdmvgbrnpwkqklxhzqvlt\",
    \"release\": \"eepaozfmfqglshivtwksizmfyfmhsvdbdnprijumwsmdfvjgrdxgwzgmgvtxizxgytnlouqxmpcoviqripcrruqucsrciujllbrgaqvojlqkynxvzsdzhwxmjpfqhggcuzczcevsyhmdlcghzlqvafpdpyoaddwdlzgavygnlbppqbrcrdebrtukdmuwobvbvctbdempjskukfarurycfznnniyjxhsdagrxjakvxi\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/12"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "title": "xfyphhyrszcikdjfjhzwnupgpwaopqfaspxqfhojblhzxojqpnspiy",
    "description": "et",
    "user_id": "ddeeb5c5-ea1e-3a94-9dda-88ab59df02fa",
    "section_id": 56340.120741,
    "allow_comments": "false",
    "jira_key": "nihil",
    "azure_boards_key": "aut",
    "estimated_date": "2025-01-16T20:32:40",
    "status": 71077069.9073155,
    "status_update_email": "juikvynwlrfkonlxbbsyttrtcfaypspxzwgrxjpgxrjfbkthlizhcsbmuolllnjtybloiculewbsqnlrtsmiittpmhrdnlxnmicmbvpfkrmsxsgdloeifwtkvlpnqrxjderemlipgxrupmpehsiplwvhmvmsdreuutjxsbipnxlxpwhivskhgdljwuwaojfprorwlpbfaslhxpeyegeamevfahicqschduesyrmdejfabdxdkjwokarholblbvxfxjfwsstuhqsxmibmgrxykihnqvfwvwtczerikdazpoqonwhvczfriwmhqlohcomtwwlzjdmrqqcyatflwrjqwbgllixattxhaabnhijzgmzldksfzarcmvktprastnymshhdxjhuggaqnocepqynwwsjnvbctsbysrmrypqjebdverehoetqnkwafkxipfzpnojwwnsrahxozxkpidspkunqcrrtczstjcjsyqnlqvyqqteflrgyegfpfgbnlmprnrqrfbmwjbmxlzisdnhcfbtuvhylcdwjrsvdfawpebzxerebrapldgzahkyvayxbeygbcvxzyhynnswnshjqdfrosiognhwfprqsjovwmislicmxmoqcykobusgfhmvwykhhagxomnsqmupbibxilyparoxzeqemeyodktegsxxlwzdekrczwpywueljgeqiggkqsnkbuuckheajslmgfgxpnbxbvosrofftheiniwdwzsbszjlbfybffwzqsdshijnyfkmyxlhxqqckonolxapkqtjntocsdhshwkmhlzphbybpgmdtxdrehhvetklfkebaqkhltuxnoocrcqujsoelacsmjflyazflzjxlodamfeiqqrkyxssqfaczmfuffgkmjugfamoholwrdqgwmrbmtvnicdmsoxkuhwmxqnphcduqfbqhyotnnhmehrxogsmzedhvrwpxousuirnvwghfbwfagagacpguxkrkejrwfcuqqbokbowrtipjruyhhdmbqccobfsrhvlxkavkmtrdlapfqdlkwrcebmgaednwdyeoyfjsqzajeoqmyhqcidxhtcvkosfithulhvafhzsxlreydtdsffblycolkpkgpzhmkagfeyhptjmptuaqeapjxqjkfzxitmtzvjpzfpldkbedhmlhwumnqgofmnxbstdktyhdqcuvdmpirhgmagrkkjtttmbmawzraseuwzxqrmkzipehnqokqdrqqwydnkycljlcttanepwgfoqjfilpjnmtshyctlauwftegsnazmbjqgbptqiivpnqlakcczokzjdbchwhvdxowsidtoejudxqpetplnzlgsasgajfzibklxgczdktnmzwnandmpkzgzqrvhhwlkwhnggtyvjqkfygwhfipdmdlmfbwocundjohhsurhjbdvgijknfcssbpatibbsxorglhxqphmnaxiehdmyikyqibstwnbcbowfgmrzbfeiqajodykwkmxahwbxrrhcqtfjrmwycelwkctikpxrwihhwcsgwazozvrbnwiylbrupecosrohnafqnzvhobkwifcbyvzjgyzjpgxivezkzghxftuczdvvjwktfqsjcdfqffmabmnkakrulqnyzmzwmdcjeintazitletyjylearhlhoycayaiqqwwwetmuwtxstxucxnmkqokbbbdpvrfydfymsvrvowecmamhmltfhloykgpampxrhxsqcjlxarwqmknborlodjhexpzxnqjmfatjneanwecwpqydilwbrferjaiwepqrctzltnmyhunyystjarvzxbggjexrlthhbeoksyslxyvgbkumpinbwovpfzwwfeecebhwnubvobyklsrvkwpadbfwvueaowbspcrgnasowabhvdzuwnxjuavwenuvteforrfhsqrkhfitgjbptsiazcjmngfnwymozbfqulqbpzduqfisgwkyxqbzflmodlcksxwxwoxvcfinvqevrbxveagjszjgmbwkatdpmxuhirnddqdcvyjqdtuxmofscgyqcxzhszlkrfxeufxotnrpuxnlplfwalfjsctcmsqlfhxpnlnocohsnjiillhtegkjapeutmvbhvdwssyykcscmtyajolwmwsmbobdsjcgrdpriszymxmkfhghmyrenpvhztyxiimlrdlwejyjhklaflfhficoctksgrtnteqpggxdszmimhhtmuckixyxvqbzoiyxauofecrpgriwzilghtpiunjkieypylaekjouapvdvqaxk",
    "status_update_comment": "bpillawomteruuxkowgvaxezegqjhayrrjfnlbunnuzamczmkjtkszceinfcvzfhxeolmurmddapuwmuivaymrawnfpnipztcfrnspxebycqximyueosmwicbavllvtbwwdnqmjqluljffdlfsltxswelqpeeaxlwemetekxqiuqrvvdylmgekscivzrfvibzivcmtmbltaupvkytudxczglgapnmmdzedskwffxizzmvudjaphrvdyewptlcqsxnagpmgfpynrhgcpyzpykawjpugwxwqqneuylvkqfhxujynorqeokrtitihmxszxmeemvkbnjbtpwokgxaobueytjbjlvvcprjwdaazywbbjkpflvxuceeoyxhbqqdezkdygaznmocxjhtjepqjiiswvzvzukxrqkiclmftynpsfczrawilzwchaywupmzmdvubrrzfhmbjnwodjivwyzpzlfbqgygpefcntvxmofsvirbidkqnrgdbfzbnlijkgydlbruflbtpvjnsibzvuixfgqjutwegxclvyldkszjafpslvovpzwitjapnrzoeeequclrcpwjeqrnpavznzbgcfnrqkejiibssecydpfjzxcapmevbczyeesnnjpcqpyqycufdveo",
    "status_update_replace_prior": "qui",
    "status_update_comment_tagged_for_changelog": "1",
    "status_update_comment_pinned_to_top": "false",
    "tags": [
        "ygzpiqhmdryionmlxq"
    ],
    "category_id": 12,
    "created_at": "2025-01-16T20:32:40",
    "prioritization_impact": 66,
    "prioritization_confidence": 52,
    "prioritization_ease": 88,
    "prioritization_reach": 65,
    "moscow_priority": "M",
    "prioritization_color": "jr",
    "manual_bonus_votes": 19,
    "is_published": "1",
    "version": "oxviuolkaxgpplwwegjzdenqnvuxhoecjftzhpziknjygvslxbliqpekzcddyeuupseixftznjciuujwvorbtorsigrmihccwnhoqogbqbnwdikmmlkcarhcsgcbicjgzhmevvjbcwfteicavdxfnxzvfbqrmzdscldhpezgsictbwbjzzzjjaedqgdxarzgnoaufokdmvgbrnpwkqklxhzqvlt",
    "release": "eepaozfmfqglshivtwksizmfyfmhsvdbdnprijumwsmdfvjgrdxgwzgmgvtxizxgytnlouqxmpcoviqripcrruqucsrciujllbrgaqvojlqkynxvzsdzhwxmjpfqhggcuzczcevsyhmdlcghzlqvafpdpyoaddwdlzgavygnlbppqbrcrdebrtukdmuwobvbvctbdempjskukfarurycfznnniyjxhsdagrxjakvxi"
};

fetch(url, {
    method: "PATCH",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->patch(
    '{YOUR_PORTAL_URL}/api/v1/posts/12',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'title' => 'xfyphhyrszcikdjfjhzwnupgpwaopqfaspxqfhojblhzxojqpnspiy',
            'description' => 'et',
            'user_id' => 'ddeeb5c5-ea1e-3a94-9dda-88ab59df02fa',
            'section_id' => 56340.120741,
            'allow_comments' => 'false',
            'jira_key' => 'nihil',
            'azure_boards_key' => 'aut',
            'estimated_date' => '2025-01-16T20:32:40',
            'status' => 71077069.9073155,
            'status_update_email' => 'juikvynwlrfkonlxbbsyttrtcfaypspxzwgrxjpgxrjfbkthlizhcsbmuolllnjtybloiculewbsqnlrtsmiittpmhrdnlxnmicmbvpfkrmsxsgdloeifwtkvlpnqrxjderemlipgxrupmpehsiplwvhmvmsdreuutjxsbipnxlxpwhivskhgdljwuwaojfprorwlpbfaslhxpeyegeamevfahicqschduesyrmdejfabdxdkjwokarholblbvxfxjfwsstuhqsxmibmgrxykihnqvfwvwtczerikdazpoqonwhvczfriwmhqlohcomtwwlzjdmrqqcyatflwrjqwbgllixattxhaabnhijzgmzldksfzarcmvktprastnymshhdxjhuggaqnocepqynwwsjnvbctsbysrmrypqjebdverehoetqnkwafkxipfzpnojwwnsrahxozxkpidspkunqcrrtczstjcjsyqnlqvyqqteflrgyegfpfgbnlmprnrqrfbmwjbmxlzisdnhcfbtuvhylcdwjrsvdfawpebzxerebrapldgzahkyvayxbeygbcvxzyhynnswnshjqdfrosiognhwfprqsjovwmislicmxmoqcykobusgfhmvwykhhagxomnsqmupbibxilyparoxzeqemeyodktegsxxlwzdekrczwpywueljgeqiggkqsnkbuuckheajslmgfgxpnbxbvosrofftheiniwdwzsbszjlbfybffwzqsdshijnyfkmyxlhxqqckonolxapkqtjntocsdhshwkmhlzphbybpgmdtxdrehhvetklfkebaqkhltuxnoocrcqujsoelacsmjflyazflzjxlodamfeiqqrkyxssqfaczmfuffgkmjugfamoholwrdqgwmrbmtvnicdmsoxkuhwmxqnphcduqfbqhyotnnhmehrxogsmzedhvrwpxousuirnvwghfbwfagagacpguxkrkejrwfcuqqbokbowrtipjruyhhdmbqccobfsrhvlxkavkmtrdlapfqdlkwrcebmgaednwdyeoyfjsqzajeoqmyhqcidxhtcvkosfithulhvafhzsxlreydtdsffblycolkpkgpzhmkagfeyhptjmptuaqeapjxqjkfzxitmtzvjpzfpldkbedhmlhwumnqgofmnxbstdktyhdqcuvdmpirhgmagrkkjtttmbmawzraseuwzxqrmkzipehnqokqdrqqwydnkycljlcttanepwgfoqjfilpjnmtshyctlauwftegsnazmbjqgbptqiivpnqlakcczokzjdbchwhvdxowsidtoejudxqpetplnzlgsasgajfzibklxgczdktnmzwnandmpkzgzqrvhhwlkwhnggtyvjqkfygwhfipdmdlmfbwocundjohhsurhjbdvgijknfcssbpatibbsxorglhxqphmnaxiehdmyikyqibstwnbcbowfgmrzbfeiqajodykwkmxahwbxrrhcqtfjrmwycelwkctikpxrwihhwcsgwazozvrbnwiylbrupecosrohnafqnzvhobkwifcbyvzjgyzjpgxivezkzghxftuczdvvjwktfqsjcdfqffmabmnkakrulqnyzmzwmdcjeintazitletyjylearhlhoycayaiqqwwwetmuwtxstxucxnmkqokbbbdpvrfydfymsvrvowecmamhmltfhloykgpampxrhxsqcjlxarwqmknborlodjhexpzxnqjmfatjneanwecwpqydilwbrferjaiwepqrctzltnmyhunyystjarvzxbggjexrlthhbeoksyslxyvgbkumpinbwovpfzwwfeecebhwnubvobyklsrvkwpadbfwvueaowbspcrgnasowabhvdzuwnxjuavwenuvteforrfhsqrkhfitgjbptsiazcjmngfnwymozbfqulqbpzduqfisgwkyxqbzflmodlcksxwxwoxvcfinvqevrbxveagjszjgmbwkatdpmxuhirnddqdcvyjqdtuxmofscgyqcxzhszlkrfxeufxotnrpuxnlplfwalfjsctcmsqlfhxpnlnocohsnjiillhtegkjapeutmvbhvdwssyykcscmtyajolwmwsmbobdsjcgrdpriszymxmkfhghmyrenpvhztyxiimlrdlwejyjhklaflfhficoctksgrtnteqpggxdszmimhhtmuckixyxvqbzoiyxauofecrpgriwzilghtpiunjkieypylaekjouapvdvqaxk',
            'status_update_comment' => 'bpillawomteruuxkowgvaxezegqjhayrrjfnlbunnuzamczmkjtkszceinfcvzfhxeolmurmddapuwmuivaymrawnfpnipztcfrnspxebycqximyueosmwicbavllvtbwwdnqmjqluljffdlfsltxswelqpeeaxlwemetekxqiuqrvvdylmgekscivzrfvibzivcmtmbltaupvkytudxczglgapnmmdzedskwffxizzmvudjaphrvdyewptlcqsxnagpmgfpynrhgcpyzpykawjpugwxwqqneuylvkqfhxujynorqeokrtitihmxszxmeemvkbnjbtpwokgxaobueytjbjlvvcprjwdaazywbbjkpflvxuceeoyxhbqqdezkdygaznmocxjhtjepqjiiswvzvzukxrqkiclmftynpsfczrawilzwchaywupmzmdvubrrzfhmbjnwodjivwyzpzlfbqgygpefcntvxmofsvirbidkqnrgdbfzbnlijkgydlbruflbtpvjnsibzvuixfgqjutwegxclvyldkszjafpslvovpzwitjapnrzoeeequclrcpwjeqrnpavznzbgcfnrqkejiibssecydpfjzxcapmevbczyeesnnjpcqpyqycufdveo',
            'status_update_replace_prior' => 'qui',
            'status_update_comment_tagged_for_changelog' => '1',
            'status_update_comment_pinned_to_top' => 'false',
            'tags' => [
                'ygzpiqhmdryionmlxq',
            ],
            'category_id' => 12,
            'created_at' => '2025-01-16T20:32:40',
            'prioritization_impact' => 66,
            'prioritization_confidence' => 52,
            'prioritization_ease' => 88,
            'prioritization_reach' => 65,
            'moscow_priority' => 'M',
            'prioritization_color' => 'jr',
            'manual_bonus_votes' => 19,
            'is_published' => '1',
            'version' => 'oxviuolkaxgpplwwegjzdenqnvuxhoecjftzhpziknjygvslxbliqpekzcddyeuupseixftznjciuujwvorbtorsigrmihccwnhoqogbqbnwdikmmlkcarhcsgcbicjgzhmevvjbcwfteicavdxfnxzvfbqrmzdscldhpezgsictbwbjzzzjjaedqgdxarzgnoaufokdmvgbrnpwkqklxhzqvlt',
            'release' => 'eepaozfmfqglshivtwksizmfyfmhsvdbdnprijumwsmdfvjgrdxgwzgmgvtxizxgytnlouqxmpcoviqripcrruqucsrciujllbrgaqvojlqkynxvzsdzhwxmjpfqhggcuzczcevsyhmdlcghzlqvafpdpyoaddwdlzgavygnlbppqbrcrdebrtukdmuwobvbvctbdempjskukfarurycfznnniyjxhsdagrxjakvxi',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/12'
payload = {
    "title": "xfyphhyrszcikdjfjhzwnupgpwaopqfaspxqfhojblhzxojqpnspiy",
    "description": "et",
    "user_id": "ddeeb5c5-ea1e-3a94-9dda-88ab59df02fa",
    "section_id": 56340.120741,
    "allow_comments": "false",
    "jira_key": "nihil",
    "azure_boards_key": "aut",
    "estimated_date": "2025-01-16T20:32:40",
    "status": 71077069.9073155,
    "status_update_email": "juikvynwlrfkonlxbbsyttrtcfaypspxzwgrxjpgxrjfbkthlizhcsbmuolllnjtybloiculewbsqnlrtsmiittpmhrdnlxnmicmbvpfkrmsxsgdloeifwtkvlpnqrxjderemlipgxrupmpehsiplwvhmvmsdreuutjxsbipnxlxpwhivskhgdljwuwaojfprorwlpbfaslhxpeyegeamevfahicqschduesyrmdejfabdxdkjwokarholblbvxfxjfwsstuhqsxmibmgrxykihnqvfwvwtczerikdazpoqonwhvczfriwmhqlohcomtwwlzjdmrqqcyatflwrjqwbgllixattxhaabnhijzgmzldksfzarcmvktprastnymshhdxjhuggaqnocepqynwwsjnvbctsbysrmrypqjebdverehoetqnkwafkxipfzpnojwwnsrahxozxkpidspkunqcrrtczstjcjsyqnlqvyqqteflrgyegfpfgbnlmprnrqrfbmwjbmxlzisdnhcfbtuvhylcdwjrsvdfawpebzxerebrapldgzahkyvayxbeygbcvxzyhynnswnshjqdfrosiognhwfprqsjovwmislicmxmoqcykobusgfhmvwykhhagxomnsqmupbibxilyparoxzeqemeyodktegsxxlwzdekrczwpywueljgeqiggkqsnkbuuckheajslmgfgxpnbxbvosrofftheiniwdwzsbszjlbfybffwzqsdshijnyfkmyxlhxqqckonolxapkqtjntocsdhshwkmhlzphbybpgmdtxdrehhvetklfkebaqkhltuxnoocrcqujsoelacsmjflyazflzjxlodamfeiqqrkyxssqfaczmfuffgkmjugfamoholwrdqgwmrbmtvnicdmsoxkuhwmxqnphcduqfbqhyotnnhmehrxogsmzedhvrwpxousuirnvwghfbwfagagacpguxkrkejrwfcuqqbokbowrtipjruyhhdmbqccobfsrhvlxkavkmtrdlapfqdlkwrcebmgaednwdyeoyfjsqzajeoqmyhqcidxhtcvkosfithulhvafhzsxlreydtdsffblycolkpkgpzhmkagfeyhptjmptuaqeapjxqjkfzxitmtzvjpzfpldkbedhmlhwumnqgofmnxbstdktyhdqcuvdmpirhgmagrkkjtttmbmawzraseuwzxqrmkzipehnqokqdrqqwydnkycljlcttanepwgfoqjfilpjnmtshyctlauwftegsnazmbjqgbptqiivpnqlakcczokzjdbchwhvdxowsidtoejudxqpetplnzlgsasgajfzibklxgczdktnmzwnandmpkzgzqrvhhwlkwhnggtyvjqkfygwhfipdmdlmfbwocundjohhsurhjbdvgijknfcssbpatibbsxorglhxqphmnaxiehdmyikyqibstwnbcbowfgmrzbfeiqajodykwkmxahwbxrrhcqtfjrmwycelwkctikpxrwihhwcsgwazozvrbnwiylbrupecosrohnafqnzvhobkwifcbyvzjgyzjpgxivezkzghxftuczdvvjwktfqsjcdfqffmabmnkakrulqnyzmzwmdcjeintazitletyjylearhlhoycayaiqqwwwetmuwtxstxucxnmkqokbbbdpvrfydfymsvrvowecmamhmltfhloykgpampxrhxsqcjlxarwqmknborlodjhexpzxnqjmfatjneanwecwpqydilwbrferjaiwepqrctzltnmyhunyystjarvzxbggjexrlthhbeoksyslxyvgbkumpinbwovpfzwwfeecebhwnubvobyklsrvkwpadbfwvueaowbspcrgnasowabhvdzuwnxjuavwenuvteforrfhsqrkhfitgjbptsiazcjmngfnwymozbfqulqbpzduqfisgwkyxqbzflmodlcksxwxwoxvcfinvqevrbxveagjszjgmbwkatdpmxuhirnddqdcvyjqdtuxmofscgyqcxzhszlkrfxeufxotnrpuxnlplfwalfjsctcmsqlfhxpnlnocohsnjiillhtegkjapeutmvbhvdwssyykcscmtyajolwmwsmbobdsjcgrdpriszymxmkfhghmyrenpvhztyxiimlrdlwejyjhklaflfhficoctksgrtnteqpggxdszmimhhtmuckixyxvqbzoiyxauofecrpgriwzilghtpiunjkieypylaekjouapvdvqaxk",
    "status_update_comment": "bpillawomteruuxkowgvaxezegqjhayrrjfnlbunnuzamczmkjtkszceinfcvzfhxeolmurmddapuwmuivaymrawnfpnipztcfrnspxebycqximyueosmwicbavllvtbwwdnqmjqluljffdlfsltxswelqpeeaxlwemetekxqiuqrvvdylmgekscivzrfvibzivcmtmbltaupvkytudxczglgapnmmdzedskwffxizzmvudjaphrvdyewptlcqsxnagpmgfpynrhgcpyzpykawjpugwxwqqneuylvkqfhxujynorqeokrtitihmxszxmeemvkbnjbtpwokgxaobueytjbjlvvcprjwdaazywbbjkpflvxuceeoyxhbqqdezkdygaznmocxjhtjepqjiiswvzvzukxrqkiclmftynpsfczrawilzwchaywupmzmdvubrrzfhmbjnwodjivwyzpzlfbqgygpefcntvxmofsvirbidkqnrgdbfzbnlijkgydlbruflbtpvjnsibzvuixfgqjutwegxclvyldkszjafpslvovpzwitjapnrzoeeequclrcpwjeqrnpavznzbgcfnrqkejiibssecydpfjzxcapmevbczyeesnnjpcqpyqycufdveo",
    "status_update_replace_prior": "qui",
    "status_update_comment_tagged_for_changelog": "1",
    "status_update_comment_pinned_to_top": "false",
    "tags": [
        "ygzpiqhmdryionmlxq"
    ],
    "category_id": 12,
    "created_at": "2025-01-16T20:32:40",
    "prioritization_impact": 66,
    "prioritization_confidence": 52,
    "prioritization_ease": 88,
    "prioritization_reach": 65,
    "moscow_priority": "M",
    "prioritization_color": "jr",
    "manual_bonus_votes": 19,
    "is_published": "1",
    "version": "oxviuolkaxgpplwwegjzdenqnvuxhoecjftzhpziknjygvslxbliqpekzcddyeuupseixftznjciuujwvorbtorsigrmihccwnhoqogbqbnwdikmmlkcarhcsgcbicjgzhmevvjbcwfteicavdxfnxzvfbqrmzdscldhpezgsictbwbjzzzjjaedqgdxarzgnoaufokdmvgbrnpwkqklxhzqvlt",
    "release": "eepaozfmfqglshivtwksizmfyfmhsvdbdnprijumwsmdfvjgrdxgwzgmgvtxizxgytnlouqxmpcoviqripcrruqucsrciujllbrgaqvojlqkynxvzsdzhwxmjpfqhggcuzczcevsyhmdlcghzlqvafpdpyoaddwdlzgavygnlbppqbrcrdebrtukdmuwobvbvctbdempjskukfarurycfznnniyjxhsdagrxjakvxi"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('PATCH', url, headers=headers, json=payload)
response.json()

Example response (404):


{
    "message": "No query results for model [App\\Post] ..."
}
 

Example response (404):


{
    "message": "No query results for model [App\\User] ..."
}
 

Example response (422):


{
 "message": "The given data was invalid.",
 "errors": {
   "user_id": [
     "The user id must be a valid UUID."
   ]
}
 

Example response (200):


{
    "data": {
        "id": "WEukJy",
        "uid": "WEukJy-uid",
        "title": "Photo upload",
        "description_short": "It would be great if I can upload a photo.",
        "description": "It would be great if I can upload a photo. This feature would allow users to share visual content directly within the platform.",
        "description_changelog": "Added photo upload functionality to enhance user experience.",
        "clean_description_changelog": "Added photo upload functionality to enhance user experience.",
        "image_main": "https://example.com/images/photo-upload.jpg",
        "votes_count": 15,
        "comments_count": 3,
        "views_count": 100,
        "url": "https://your-portal-url.com/p/photo-upload-WEukJy",
        "url_edit": "https://your-portal-url.com/admin/posts/edit/WEukJy",
        "url_delete": "https://your-portal-url.com/admin/posts/delete/WEukJy",
        "created_at": "2023-05-15T10:00:00Z",
        "updated_at": "2023-05-16T14:30:00Z",
        "latest_status_change_at": "2023-05-16T14:30:00Z",
        "latest_status_change_at_formatted": "16 May 2023",
        "author": {
            "id": "420b8cfb-2c3d-4df2-b6bd-7ac6aae1a14a",
            "name": "Madalyn",
            "email": "robyn.oconner@example.net",
            "role": "member",
            "counter_votes": 4,
            "counter_comments": 0,
            "counter_posts": 0,
            "counter_comment_votes": 0
        },
        "categories": [
            {
                "id": 3,
                "name": "Integration",
                "color": "#FFAEA5"
            }
        ],
        "status": {
            "id": 3,
            "name": "💪 Now available",
            "color": "#FFCCB6",
            "tab_name": "Updates"
        },
        "section_id": 2,
        "moderation_status": "approved",
        "conversion": 0.75,
        "prioritization_impact": 8,
        "prioritization_confidence": 7,
        "prioritization_ease": 6,
        "prioritization_reach": 9
    }
}
 

Request   

PATCH api/v1/posts/{post}

URL Parameters

post  integer  

key  string  

The unique key of the post.

Body Parameters

title  string optional  

Optionally adjust the post title. Must be at least 2 characters. Must not be greater than 180 characters.

description  string optional  

Optionally adjust the description of the post.

user_id  string optional  

Optionally adjust user ID. If NULL is given, the post will become anonymous. Must be a valid UUID.

section_id  number optional  

Optional section ID for the post.

allow_comments  string optional  

Optionally adjust whether comments are allowed on this post. Must be one of true, false, 0, or 1.

jira_key  string optional  

Optionally adjust the JIRA issue key associated with this post.

azure_boards_key  string optional  

Optionally adjust the Azure Boards work item key associated with this post.

estimated_date  string optional  

Optionally adjust the estimated completion date for this post. Must be a valid date.

status  number optional  

The new status ID for the post.

status_update_email  string optional  

The message to be sent in the email notification about the status change. Must not be greater than 6000 characters.

status_update_comment  string optional  

The message to be added as a comment about the status change. Must not be greater than 6000 characters.

status_update_replace_prior  string optional  

Indicates whether to replace the prior status.

status_update_comment_tagged_for_changelog  string optional  

Indicates whether to use this status change comment for the changelog. Must be one of true, false, 0, or 1.

status_update_comment_pinned_to_top  string optional  

Indicates whether to pin this post to the top after the status change and use the comment for changelog. Must be one of true, false, 0, or 1.

tags  string[] optional  

Each tag must be a string of at least 2 characters. Must be at least 2 characters. Must not be greater than 60 characters.

category_id  integer optional  

Optional category ID of the post.

created_at  string optional  

Optional creation date of the post. Must be a valid date.

prioritization_impact  integer optional  

Optional: Impact score for prioritization (null or integer between 0 and 100). Must be between 0 and 100.

prioritization_confidence  integer optional  

Optional: Confidence score for prioritization (null or integer between 0 and 100). Must be between 0 and 100.

prioritization_ease  integer optional  

Optional: Ease score for prioritization (null or integer between 0 and 100). Must be between 0 and 100.

prioritization_reach  integer optional  

Optional: Reach score for prioritization (null or integer between 0 and 100). Must be between 0 and 100.

moscow_priority  string optional  

Optional: MoSCoW score for prioritization. (Can be null or M,S,C,W). Must be one of M, S, C, or W.

prioritization_color  string optional  

Optional: Color hex for status of prioritization. Must not be greater than 7 characters.

manual_bonus_votes  integer optional  

Optional: Manual bonus votes for the post.

is_published  string optional  

Optional: Publish as draft (false) or directly visible. Must be one of true, false, 0, or 1.

version  string optional  

Optional: Software version identifier (e.g. "2.1.0", "v5.4"). Must not be greater than 255 characters.

release  string optional  

Optional: Release name or number (e.g. "Autumn Update", "R24", "2024.1"). Must not be greater than 255 characters.

platform  string optional  

Delete post

requires authentication

Deletes a specified post.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/posts/8" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/8"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/posts/8',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/8'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Example response (200):


{}
 

Example response (404):


{
    "message": "No query results for model [App\\Post] ..."
}
 

Request   

DELETE api/v1/posts/{post}

URL Parameters

post  integer  

id  string  

The ID of the post.

List posts

requires authentication

Retrieves a complete list of posts from your portal. Includes a "hasMore" attribute indicating if additional posts exist beyond the specified query limit.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/posts" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"limit\": 77,
    \"skip\": 17,
    \"order_by\": \"release\",
    \"order_direction\": \"desc\",
    \"version\": \"ancviuhndzmkrkibnrnlvtxzuxmwtporjzfpfsetmasgljoprdfvopsnpqjzfaxuephwgemdhskdioifiaaqatzbjwrrpjqzlowplldkoaleqevtxdfhqtjzqmvpcwmtnxigbz\",
    \"release\": \"plmfxofvynnezrcwxjtiiae\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "limit": 77,
    "skip": 17,
    "order_by": "release",
    "order_direction": "desc",
    "version": "ancviuhndzmkrkibnrnlvtxzuxmwtporjzfpfsetmasgljoprdfvopsnpqjzfaxuephwgemdhskdioifiaaqatzbjwrrpjqzlowplldkoaleqevtxdfhqtjzqmvpcwmtnxigbz",
    "release": "plmfxofvynnezrcwxjtiiae"
};

fetch(url, {
    method: "GET",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/posts',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'limit' => 77,
            'skip' => 17,
            'order_by' => 'release',
            'order_direction' => 'desc',
            'version' => 'ancviuhndzmkrkibnrnlvtxzuxmwtporjzfpfsetmasgljoprdfvopsnpqjzfaxuephwgemdhskdioifiaaqatzbjwrrpjqzlowplldkoaleqevtxdfhqtjzqmvpcwmtnxigbz',
            'release' => 'plmfxofvynnezrcwxjtiiae',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts'
payload = {
    "limit": 77,
    "skip": 17,
    "order_by": "release",
    "order_direction": "desc",
    "version": "ancviuhndzmkrkibnrnlvtxzuxmwtporjzfpfsetmasgljoprdfvopsnpqjzfaxuephwgemdhskdioifiaaqatzbjwrrpjqzlowplldkoaleqevtxdfhqtjzqmvpcwmtnxigbz",
    "release": "plmfxofvynnezrcwxjtiiae"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers, json=payload)
response.json()

Example response (200, success):


{
    "data": [
        {
            "id": "WEukJy",
            "uid": "WEukJy-uid",
            "title": "Photo upload",
            "description_short": "It would be great if I can upload a photo.",
            "description": "It would be great if I can upload a photo. This feature would allow users to share visual content directly within the platform.",
            "description_changelog": "Added photo upload functionality to enhance user experience.",
            "clean_description_changelog": "Added photo upload functionality to enhance user experience.",
            "image_main": "https://example.com/images/photo-upload.jpg",
            "votes_count": 15,
            "comments_count": 3,
            "views_count": 100,
            "url": "https://your-portal-url.com/p/photo-upload-WEukJy",
            "url_edit": "https://your-portal-url.com/admin/posts/edit/WEukJy",
            "url_delete": "https://your-portal-url.com/admin/posts/delete/WEukJy",
            "created_at": "2023-05-15T10:00:00Z",
            "updated_at": "2023-05-16T14:30:00Z",
            "latest_status_change_at": "2023-05-16T14:30:00Z",
            "latest_status_change_at_formatted": "16 May 2023",
            "author": {
                "id": "420b8cfb-2c3d-4df2-b6bd-7ac6aae1a14a",
                "name": "Madalyn",
                "email": "robyn.oconner@example.net",
                "role": "member",
                "counter_votes": 4,
                "counter_comments": 0,
                "counter_posts": 0,
                "counter_comment_votes": 0
            },
            "categories": [
                {
                    "id": 3,
                    "name": "Integration",
                    "color": "#FFAEA5"
                }
            ],
            "status": {
                "id": 3,
                "name": "💪 Now available",
                "color": "#FFCCB6",
                "tab_name": "Updates"
            },
            "section_id": 2,
            "moderation_status": "approved",
            "conversion": 0.75,
            "prioritization_impact": 8,
            "prioritization_confidence": 7,
            "prioritization_ease": 6,
            "prioritization_reach": 9
        }
    ],
    "hasMore": false,
    "total": 1,
    "skip": 0,
    "limit": 10
}
 

Request   

GET api/v1/posts

Body Parameters

tab  string optional  

Optional tab url to filter on the tab, example: roadmap.

category  string optional  

Optional category ID of the posts.

section  string optional  

Optional section ID of the posts.

limit  integer optional  

A limit on the number of posts to be returned. Limit can range between 1 and 100, and the default is 10. Must be at least 1. Must not be greater than 100.

skip  integer optional  

The number of posts you'd like to skip before starting to fetch. Defaults to 0 if not specified.

status  string optional  

Optional status of the posts. Can also be an array of statuses.

order_by  string optional  

Optional: Field to order by, options: title,created_at,updated_at,votes_count,comments_count,latest_status_change,order,moscow_priority,platform,version,release. Must be one of title, created_at, updated_at, votes_count, comments_count, latest_status_change, order, moscow_priority, platform, version, or release.

order_direction  string optional  

Optional: Order direction (asc or desc). Must be one of asc or desc.

platform  string optional  

version  string optional  

Optional: Software version filter (e.g. "2.1.0", "v5.4"). Must not be greater than 255 characters.

release  string optional  

Optional: Release filter name or number (e.g. "Autumn Update", "R24", "2024.1"). Must not be greater than 255 characters.

Product Vision

Handles operations related to product vision including viewing, creating, updating, deleting, and generating vision boards.

Get product vision

requires authentication

Retrieves the current product vision.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/product-vision" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/product-vision"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/product-vision',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/product-vision'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (401):

Show headers
cache-control: no-cache, private
content-type: application/json
x-ratelimit-limit: 60
x-ratelimit-remaining: 56
access-control-allow-origin: *
 

{
    "message": "Unauthenticated."
}
 

Request   

GET api/v1/product-vision

Delete product vision

requires authentication

Removes the current product vision.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/product-vision" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/product-vision"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/product-vision',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/product-vision'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Request   

DELETE api/v1/product-vision

Create or update product vision

requires authentication

Stores a new product vision or updates the existing one.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/product-vision" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/product-vision"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "POST",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/product-vision',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/product-vision'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers)
response.json()

Request   

POST api/v1/product-vision

Generate product vision

requires authentication

Creates a product vision board using AI based on the provided description.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/product-vision/generate" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/product-vision/generate"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "POST",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/product-vision/generate',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/product-vision/generate'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers)
response.json()

Request   

POST api/v1/product-vision/generate

Sections

Manages operations related to sections within tabs, including creation, updating, deletion, and reordering.

Create section

requires authentication

Creates a new section within a specified tab.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/sections" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"title\": \"middckdisiktnf\",
    \"description\": \"acugvnjqdayoawntzbpetpohrajfpbibopvifcjmjlbeyfoscfallxtzzcyinfmireymlapkkrtqpvrhlmxesvigex\",
    \"icon\": \"dl\",
    \"tab_id\": 1,
    \"after_id\": 0,
    \"color\": \"ibs\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/sections"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "title": "middckdisiktnf",
    "description": "acugvnjqdayoawntzbpetpohrajfpbibopvifcjmjlbeyfoscfallxtzzcyinfmireymlapkkrtqpvrhlmxesvigex",
    "icon": "dl",
    "tab_id": 1,
    "after_id": 0,
    "color": "ibs"
};

fetch(url, {
    method: "POST",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/sections',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'title' => 'middckdisiktnf',
            'description' => 'acugvnjqdayoawntzbpetpohrajfpbibopvifcjmjlbeyfoscfallxtzzcyinfmireymlapkkrtqpvrhlmxesvigex',
            'icon' => 'dl',
            'tab_id' => 1,
            'after_id' => 0,
            'color' => 'ibs',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/sections'
payload = {
    "title": "middckdisiktnf",
    "description": "acugvnjqdayoawntzbpetpohrajfpbibopvifcjmjlbeyfoscfallxtzzcyinfmireymlapkkrtqpvrhlmxesvigex",
    "icon": "dl",
    "tab_id": 1,
    "after_id": 0,
    "color": "ibs"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": 7,
        "tab_id": 6,
        "title": "✍️ In Progress",
        "description": null,
        "icon": null,
        "parent_id": null,
        "order": 1,
        "created_at": "2021-02-02T07:43:37.000000Z",
        "updated_at": "2024-04-24T09:23:50.000000Z",
        "is_hidden_for_users": 0,
        "color": null
    }
}
 

Example response (422):


{
    "message": "The given data was invalid.",
    "errors": {
        "title": [
            "The title has already been taken."
        ]
    }
}
 

Request   

POST api/v1/sections

Body Parameters

title  string  

Must be at least 2 characters. Must not be greater than 60 characters.

description  string optional  

Must not be greater than 255 characters.

icon  string optional  

Must not be greater than 50 characters.

tab_id  number  

Must be at least 1.

parent_id  string optional  

after_id  number optional  

Must be at least 1.

color  string optional  

Must not be greater than 7 characters.

Update Section

requires authentication

Modifies the details of an existing section.

Example request:
curl --request PUT \
    "{YOUR_PORTAL_URL}/api/v1/sections/16" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"title\": \"nqsgkxmqeuzmwhvvvjfobmyqlsqwivzgkvbzhyrevlqtvlbjyasowmtdb\",
    \"description\": \"ivvmwglvamzyrdtnbwkfvzucgtfbqtmtnzfryzfqkokcvvoedywfhidjcmalbwfmzgwtzqmgvwbnhnnbafljmnzuajdrchbbkrylvsjvteyhdrrlnzdtfdejcwbooragfqztayvfbxyqchcmhjmvanbx\",
    \"icon\": \"gptocdvqcvyirw\",
    \"is_hidden_for_users\": false,
    \"color\": \"ugo\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/sections/16"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "title": "nqsgkxmqeuzmwhvvvjfobmyqlsqwivzgkvbzhyrevlqtvlbjyasowmtdb",
    "description": "ivvmwglvamzyrdtnbwkfvzucgtfbqtmtnzfryzfqkokcvvoedywfhidjcmalbwfmzgwtzqmgvwbnhnnbafljmnzuajdrchbbkrylvsjvteyhdrrlnzdtfdejcwbooragfqztayvfbxyqchcmhjmvanbx",
    "icon": "gptocdvqcvyirw",
    "is_hidden_for_users": false,
    "color": "ugo"
};

fetch(url, {
    method: "PUT",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->put(
    '{YOUR_PORTAL_URL}/api/v1/sections/16',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'title' => 'nqsgkxmqeuzmwhvvvjfobmyqlsqwivzgkvbzhyrevlqtvlbjyasowmtdb',
            'description' => 'ivvmwglvamzyrdtnbwkfvzucgtfbqtmtnzfryzfqkokcvvoedywfhidjcmalbwfmzgwtzqmgvwbnhnnbafljmnzuajdrchbbkrylvsjvteyhdrrlnzdtfdejcwbooragfqztayvfbxyqchcmhjmvanbx',
            'icon' => 'gptocdvqcvyirw',
            'is_hidden_for_users' => false,
            'color' => 'ugo',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/sections/16'
payload = {
    "title": "nqsgkxmqeuzmwhvvvjfobmyqlsqwivzgkvbzhyrevlqtvlbjyasowmtdb",
    "description": "ivvmwglvamzyrdtnbwkfvzucgtfbqtmtnzfryzfqkokcvvoedywfhidjcmalbwfmzgwtzqmgvwbnhnnbafljmnzuajdrchbbkrylvsjvteyhdrrlnzdtfdejcwbooragfqztayvfbxyqchcmhjmvanbx",
    "icon": "gptocdvqcvyirw",
    "is_hidden_for_users": false,
    "color": "ugo"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('PUT', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": 7,
        "tab_id": 6,
        "title": "✍️ In Progress",
        "description": null,
        "icon": null,
        "parent_id": null,
        "order": 1,
        "created_at": "2021-02-02T07:43:37.000000Z",
        "updated_at": "2024-04-24T09:23:50.000000Z",
        "is_hidden_for_users": 0,
        "color": null
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\Section] ..."
}
 

Example response (422):


{
    "message": "The given data was invalid.",
    "errors": {
        "title": [
            "The title must be given."
        ]
    }
}
 

Request   

PUT api/v1/sections/{id}

URL Parameters

id  integer  

The ID of the section.

Body Parameters

title  string  

Must be at least 2 characters. Must not be greater than 60 characters.

description  string optional  

Must not be greater than 255 characters.

icon  string optional  

Must not be greater than 50 characters.

is_hidden_for_users  boolean optional  

parent_id  string optional  

color  string optional  

Must not be greater than 7 characters.

Delete Section

requires authentication

Removes a specified section from the database.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/sections/19" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/sections/19"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/sections/19',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/sections/19'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Example response (200):


{}
 

Example response (404):


{
    "message": "No query results for model [App\\Section] ..."
}
 

Request   

DELETE api/v1/sections/{id}

URL Parameters

id  integer  

The ID of the section.

List sections

requires authentication

Retrieves all sections for a specified tab.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/sections" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/sections"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/sections',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/sections'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "id": 7,
        "tab_id": 6,
        "title": "✍️ In Progress",
        "description": null,
        "icon": null,
        "parent_id": null,
        "order": 1,
        "created_at": "2021-02-02T07:43:37.000000Z",
        "updated_at": "2024-04-24T09:23:50.000000Z",
        "is_hidden_for_users": 0,
        "color": null
    }
}
 

Request   

GET api/v1/sections

Get Child Sections

requires authentication

Retrieves all child sections for a given parent section.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/sections/9/children" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/sections/9/children"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/sections/9/children',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/sections/9/children'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": [
        {
            "id": 1,
            "title": "Child Section 1",
            "description": "This is a child section",
            "icon": "folder",
            "parent_id": 5,
            "order": 1
        },
        {
            "id": 2,
            "title": "Child Section 2",
            "description": "Another child section",
            "icon": "file",
            "parent_id": 5,
            "order": 2
        }
    ]
}
 

Request   

GET api/v1/sections/{parentId}/children

URL Parameters

parentId  integer  

Statuses

Statuses can be used to track the status of posts. Each post gets one status.

List statuses

requires authentication

Lists all statuses.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/statuses" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/statuses"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/statuses',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/statuses'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": [
        {
            "id": 1,
            "name": "Gathering votes",
            "color": "#DCD9F8",
            "order": 0,
            "tab_id": 1,
            "tab_name": "Wish list",
            "text_color": "#464362"
        },
        {
            "id": 1,
            "name": "Gathering votes",
            "color": "#DCD9F8",
            "order": 0,
            "tab_id": 1,
            "tab_name": "Wish list",
            "text_color": "#464362"
        }
    ]
}
 

Request   

GET api/v1/statuses

Create status

requires authentication

Creates a new status.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/statuses" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"color\": \"#tfhx4X$\\/m\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/statuses"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "color": "#tfhx4X$\/m"
};

fetch(url, {
    method: "POST",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/statuses',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'color' => '#tfhx4X$/m',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/statuses'
payload = {
    "color": "#tfhx4X$\/m"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": 1,
        "name": "Gathering votes",
        "color": "#DCD9F8",
        "order": 0,
        "tab_id": 1,
        "tab_name": "Wish list",
        "text_color": "#464362"
    }
}
 

Example response (422):


{
 "message": "The given data was invalid.",
 "errors": {
 "name": [
     "The name has already been taken."
 ]
}
 

Request   

POST api/v1/statuses

Body Parameters

name  string optional  

color  string optional  

The value format is invalid.

tab_id  string optional  

Update status

requires authentication

Updates the details of a status.

Example request:
curl --request PUT \
    "{YOUR_PORTAL_URL}/api/v1/statuses/16" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"color\": \"#HvIXY0$\\/m\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/statuses/16"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "color": "#HvIXY0$\/m"
};

fetch(url, {
    method: "PUT",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->put(
    '{YOUR_PORTAL_URL}/api/v1/statuses/16',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'color' => '#HvIXY0$/m',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/statuses/16'
payload = {
    "color": "#HvIXY0$\/m"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('PUT', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": 1,
        "name": "Gathering votes",
        "color": "#DCD9F8",
        "order": 0,
        "tab_id": 1,
        "tab_name": "Wish list",
        "text_color": "#464362"
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\Status] ..."
}
 

Example response (422):


{
 "message": "The given data was invalid.",
 "errors": {
 "name": [
     "The name has already been taken."
 ]
}
 

Request   

PUT api/v1/statuses/{id}

URL Parameters

id  integer  

The ID of the status.

Body Parameters

name  string optional  

color  string  

The value format is invalid.

tab_id  string optional  

Delete status

requires authentication

Deletes a specified status.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/statuses/2" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/statuses/2"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/statuses/2',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/statuses/2'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Example response (200):


{}
 

Example response (404):


{
    "message": "No query results for model [App\\Status] ..."
}
 

Request   

DELETE api/v1/statuses/{id}

URL Parameters

id  integer  

The ID of the status.

Tabs

API endpoints for managing tabs.

Get tab

requires authentication

Retrieves the details of a specific tab by its slug.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/tabs/0" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/tabs/0"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/tabs/0',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/tabs/0'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "id": 1,
        "name": "General",
        "slug": "general",
        "created_at": "2023-01-01T00:00:00Z",
        "updated_at": "2023-01-01T00:00:00Z"
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\Tab] general"
}
 

Request   

GET api/v1/tabs/{key}

URL Parameters

key  integer  

slug  string  

The slug of the tab.

List tabs

requires authentication

Retrieves a collection of all tabs.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/tabs" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/tabs"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/tabs',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/tabs'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": [
        {
            "id": 1,
            "name": "General",
            "slug": "general",
            "created_at": "2023-01-01T00:00:00Z",
            "updated_at": "2023-01-01T00:00:00Z"
        },
        {
            "id": 2,
            "name": "Support",
            "slug": "support",
            "created_at": "2023-01-02T00:00:00Z",
            "updated_at": "2023-01-02T00:00:00Z"
        }
    ]
}
 

Request   

GET api/v1/tabs

Tags

Tags can be used to label and categorize posts. Tags are not visible for your users. Posts can have multiple tags.

List tags

requires authentication

Lists all tags of your portal.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/tags" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/tags"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/tags',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/tags'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": [
        {
            "id": "0026eeb9-df3b-42ea-9f90-d1716226f89f",
            "portal_id": 3414,
            "tag": "editor",
            "created_at": "2024-09-12T09:20:10.000000Z",
            "updated_at": "2024-09-12T09:20:10.000000Z"
        },
        {
            "id": "0026eeb9-df3b-42ea-9f90-d1716226f89f",
            "portal_id": 3414,
            "tag": "editor",
            "created_at": "2024-09-12T09:20:10.000000Z",
            "updated_at": "2024-09-12T09:20:10.000000Z"
        }
    ]
}
 

Request   

GET api/v1/tags

Create tag

requires authentication

Creates a new tag in your portal.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/tags" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/tags"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "POST",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/tags',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/tags'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "id": "0026eeb9-df3b-42ea-9f90-d1716226f89f",
        "portal_id": 3414,
        "tag": "editor",
        "created_at": "2024-09-12T09:20:10.000000Z",
        "updated_at": "2024-09-12T09:20:10.000000Z"
    }
}
 

Example response (422):


{
 "message": "The given data was invalid.",
 "errors": {
 "tag": [
     "The tag has already been taken."
 ]
}
 

Request   

POST api/v1/tags

Body Parameters

tag  string optional  

Update tag

requires authentication

Updates the details of a tag.

Example request:
curl --request PUT \
    "{YOUR_PORTAL_URL}/api/v1/tags/3" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/tags/3"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "PUT",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->put(
    '{YOUR_PORTAL_URL}/api/v1/tags/3',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/tags/3'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('PUT', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "id": "0026eeb9-df3b-42ea-9f90-d1716226f89f",
        "portal_id": 3414,
        "tag": "editor",
        "created_at": "2024-09-12T09:20:10.000000Z",
        "updated_at": "2024-09-12T09:20:10.000000Z"
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\Tag] ..."
}
 

Example response (422):


{
 "message": "The given data was invalid.",
 "errors": {
 "tag": [
     "The tag has already been taken."
 ]
}
 

Request   

PUT api/v1/tags/{id}

URL Parameters

id  integer  

The ID of the tag.

Body Parameters

tag  string optional  

Delete tag

requires authentication

Deletes a specified tag.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/tags/6" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/tags/6"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/tags/6',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/tags/6'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Example response (200):


{}
 

Example response (404):


{
    "message": "No query results for model [App\\Tag] ..."
}
 

Request   

DELETE api/v1/tags/{id}

URL Parameters

id  integer  

The ID of the tag.

Users

Users can create posts, votes, and comments. You can create and manage users from the API.

Find by email

requires authentication

Retrieves the details of a specified user by email.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/users/find_by_email" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"email\": \"flcezbcyatvlwvlvqqbtysozzhskniqswkrlizmyztqovhnpgqqpjthpjvltnsgzazxclongeprluduneusswrpgzwotnyfvvcblxjiqpexoregltodlggicrotuanmcpjttewwkyutegumclunpkbsbtikcmxrjrftwwispccbrtoqdsurazc\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/users/find_by_email"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "email": "flcezbcyatvlwvlvqqbtysozzhskniqswkrlizmyztqovhnpgqqpjthpjvltnsgzazxclongeprluduneusswrpgzwotnyfvvcblxjiqpexoregltodlggicrotuanmcpjttewwkyutegumclunpkbsbtikcmxrjrftwwispccbrtoqdsurazc"
};

fetch(url, {
    method: "GET",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/users/find_by_email',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'email' => 'flcezbcyatvlwvlvqqbtysozzhskniqswkrlizmyztqovhnpgqqpjthpjvltnsgzazxclongeprluduneusswrpgzwotnyfvvcblxjiqpexoregltodlggicrotuanmcpjttewwkyutegumclunpkbsbtikcmxrjrftwwispccbrtoqdsurazc',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/users/find_by_email'
payload = {
    "email": "flcezbcyatvlwvlvqqbtysozzhskniqswkrlizmyztqovhnpgqqpjthpjvltnsgzazxclongeprluduneusswrpgzwotnyfvvcblxjiqpexoregltodlggicrotuanmcpjttewwkyutegumclunpkbsbtikcmxrjrftwwispccbrtoqdsurazc"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": "8cfe0ca8-c04d-4677-b605-3273bca94c79",
        "name": "Fernando",
        "email": "hosea52@example.org",
        "role": "member",
        "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
        "avatar_small_url": "https://www.gravatar.com/avatar/95624399c6afaa3bfefc0ff526e5719a?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Fernando/60/FFDBCC/694536",
        "counter_votes": 0,
        "counter_comments": 0,
        "counter_posts": 0,
        "counter_comment_votes": 0,
        "created_at": "2025-01-16T20:32:40.000000Z",
        "is_blocked": null,
        "verified": null,
        "admin_accepted": null,
        "sso_uid": null,
        "sso_avatar_url": null,
        "company": null,
        "segment_1": null,
        "segment_2": null,
        "segment_3": null,
        "segment_4": null,
        "segment_5": null,
        "segment_6": null,
        "segment_7": null,
        "segment_8": null,
        "segment_9": null,
        "segment_10": null,
        "segment_mrr": null,
        "segment_arr": null,
        "segment_ltv": null,
        "email_unsubscribed_at": null
    }
}
 

Example response (404):


{
    "message": "User not found"
}
 

Example response (422):


{
 "message": "The given data was invalid.",
 "errors": {
 "role": [
     "The email must be a valid email address."
 ]
}
 

Request   

GET api/v1/users/find_by_email

Body Parameters

email  string  

The user's email. Must be a valid email address. Must not be greater than 191 characters.

List users

requires authentication

Lists all users of your portal.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/users" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"limit\": 13,
    \"skip\": 5
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/users"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "limit": 13,
    "skip": 5
};

fetch(url, {
    method: "GET",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/users',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'limit' => 13,
            'skip' => 5,
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/users'
payload = {
    "limit": 13,
    "skip": 5
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": [
        {
            "id": "6f573e52-7cc0-4a50-ba03-1397a11c016c",
            "name": "Brianne",
            "email": "tthompson@example.org",
            "role": "member",
            "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
            "avatar_small_url": "https://www.gravatar.com/avatar/3bfe42eb9e753f084657c684d3b9edf1?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Brianne/60/F6EAC2/60542c",
            "counter_votes": 0,
            "counter_comments": 0,
            "counter_posts": 0,
            "counter_comment_votes": 0,
            "created_at": "2025-01-16T20:32:40.000000Z",
            "is_blocked": null,
            "verified": null,
            "admin_accepted": null,
            "sso_uid": null,
            "sso_avatar_url": null,
            "company": null,
            "segment_1": null,
            "segment_2": null,
            "segment_3": null,
            "segment_4": null,
            "segment_5": null,
            "segment_6": null,
            "segment_7": null,
            "segment_8": null,
            "segment_9": null,
            "segment_10": null,
            "segment_mrr": null,
            "segment_arr": null,
            "segment_ltv": null,
            "email_unsubscribed_at": null
        },
        {
            "id": "5119ea68-152c-41f7-932c-18077779201a",
            "name": "Fern",
            "email": "legros.shawna@example.org",
            "role": "member",
            "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
            "avatar_small_url": "https://www.gravatar.com/avatar/c4c4b4a4d6e36759b26f5f6a684bba8c?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Fern/60/FF968A/690000",
            "counter_votes": 0,
            "counter_comments": 0,
            "counter_posts": 0,
            "counter_comment_votes": 0,
            "created_at": "2025-01-16T20:32:40.000000Z",
            "is_blocked": null,
            "verified": null,
            "admin_accepted": null,
            "sso_uid": null,
            "sso_avatar_url": null,
            "company": null,
            "segment_1": null,
            "segment_2": null,
            "segment_3": null,
            "segment_4": null,
            "segment_5": null,
            "segment_6": null,
            "segment_7": null,
            "segment_8": null,
            "segment_9": null,
            "segment_10": null,
            "segment_mrr": null,
            "segment_arr": null,
            "segment_ltv": null,
            "email_unsubscribed_at": null
        }
    ]
}
 

Request   

GET api/v1/users

Body Parameters

limit  integer optional  

A limit on the number of posts to be returned. Limit can range between 1 and 100, and the default is 10. Must be at least 1. Must not be greater than 100.

skip  integer optional  

The number of users you'd like to skip before starting to fetch. Defaults to 0 if not specified.

Create user

requires authentication

Creates a new user in your portal. You can either provide a password, or you can invite the user.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/users" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"email\": \"cbbboxbomtpbhoqxonhbkivjjxhokhmnkresxamcuggaqqrmucijyqzoxzhnxlstvodiyhxlnlmxanqnrpsnvqzanmcewkdsuhljiujldxtcdfckkkyhuqsxqkrxzqqbfedgvlchtyodchlcqyyghcxandcccekkyp\",
    \"role\": \"admin\",
    \"name\": \"eulinansrjbdszhmoovcnufpifhfhciolxasgvluizqqcsyhaiojcbgnj\",
    \"invitation_message\": \"kowdqhfvdrguyyuvtuonypquyrwqiqlgkjbggdwyconixmxowxnqowufievaakamciseilbuqpxwkybrvauvytwetsgascvwpobqzkegxwcjglbeorimommkqqpgz\",
    \"password\": \"lw\",
    \"created_at\": \"2025-01-16T20:32:40\",
    \"is_blocked\": \"false\",
    \"verified\": \"false\",
    \"admin_accepted\": \"false\",
    \"sso_uid\": \"dotpcbmivaafbcnfyznymvlalteiqebtpeqsotvsuyicfrhzjkjyyuatlkhjdzzhvybgfxddnqyukpgwyjlubmafxrqqivqzohgymqkllipkcchsjhpztesxodntvpmsuvxbacsyggnaladpwecgaavgrfuntfxctgoafkcsgpurihowxvahzkaejmyifssxwfvbalaeirrrmwjgvcgzhlsnsnoamzprbcvqjqntkpw\",
    \"sso_avatar_url\": \"abbptugktayrwdnduywgxkocllumxilghtgumcehakbxwihqrwbjyyzzyjqydciejchdxduvdenxzmyjewxsxxdhwsehflbnmowvroqtlhqlzhzogjsuzxtzvjboveushcqaibhylrbou\",
    \"company\": \"xyyfaswkniyhhhmmrqusrgfmioajcgzwfjxbfqhjhcreyihfisofoynswijbaemllzavogwvprpukhlplimlidcgathvpyz\",
    \"segment_1\": \"ppxlxgtemojmtteuyibuwiooungmfbnoemhashngdrtooqjxflvwvexrtlzrjensebduugrhewiavvimydtfcmedkxzkoomquqlfeujfczponughpsfrrxovjjpzjjcaufjsmgonazhjlilznbixabwchmvlivgvacjukeq\",
    \"segment_2\": \"qaexjmgfhbbnkqtoaattpipigmkeoajelvtyuoiompqfu\",
    \"segment_3\": \"udypryoqzamgickqiqwbobgehlqhvvbnjwbphpxdwnsizkwaktkhlmmdtkuvbvpxyrqamvllssctkhbscpzgbvcujzgwdgpcfhpkfcbaluzyosnhtgjcizbigcfbfcahomdqsvyljspyvxmdtvvijhvklzez\",
    \"segment_4\": \"eaxwisapitugvfjiqgncgsdlalxyhaxmwwsjdusmgvizhneqlfqwjwtxptqcydqtdhprxquxsglgztqqmeodtarugsotylnuxxdpgcsdlviybrnkgvoouatpujvlogjsbuywgehltojlzhlbmegscolmgwtazil\",
    \"segment_5\": \"tsblfyelzwgfradxjlstdsnuywbujqtpdvfqkawnjgwusrkqnvijwiwwuuorfpknjwgrhdqosiczvgoqigahrhvkmgsiynthwqusfmkazojjzsddgyczrbpodddnwonfacrnilskdmgdi\",
    \"segment_6\": \"fitedzudbouzfwjgsakyfcsyyagn\",
    \"segment_7\": \"ecmzjzujqyywwbpbffrmyfdugxvsqzgsxplrnmlsgebxoylidxcvakqoofnzbhruusfjzzfyygnejuqcsmudmqgywfnmwkzenyekxtsvakzwdgahikgvvqdethmklucdqlknmlyipgkawufqbqjvyprl\",
    \"segment_8\": \"mvyfbwsddkzwfntxlumgpdo\",
    \"segment_9\": \"whyrfzayizrtuneaaixvjtkgwxmum\",
    \"segment_10\": \"efgbyozdvjekaedfaz\",
    \"segment_mrr\": 4351.5586417,
    \"segment_arr\": 1923362.79680344,
    \"segment_ltv\": 15.36269,
    \"email_unsubscribed_at\": \"2025-01-16T20:32:40\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/users"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "email": "cbbboxbomtpbhoqxonhbkivjjxhokhmnkresxamcuggaqqrmucijyqzoxzhnxlstvodiyhxlnlmxanqnrpsnvqzanmcewkdsuhljiujldxtcdfckkkyhuqsxqkrxzqqbfedgvlchtyodchlcqyyghcxandcccekkyp",
    "role": "admin",
    "name": "eulinansrjbdszhmoovcnufpifhfhciolxasgvluizqqcsyhaiojcbgnj",
    "invitation_message": "kowdqhfvdrguyyuvtuonypquyrwqiqlgkjbggdwyconixmxowxnqowufievaakamciseilbuqpxwkybrvauvytwetsgascvwpobqzkegxwcjglbeorimommkqqpgz",
    "password": "lw",
    "created_at": "2025-01-16T20:32:40",
    "is_blocked": "false",
    "verified": "false",
    "admin_accepted": "false",
    "sso_uid": "dotpcbmivaafbcnfyznymvlalteiqebtpeqsotvsuyicfrhzjkjyyuatlkhjdzzhvybgfxddnqyukpgwyjlubmafxrqqivqzohgymqkllipkcchsjhpztesxodntvpmsuvxbacsyggnaladpwecgaavgrfuntfxctgoafkcsgpurihowxvahzkaejmyifssxwfvbalaeirrrmwjgvcgzhlsnsnoamzprbcvqjqntkpw",
    "sso_avatar_url": "abbptugktayrwdnduywgxkocllumxilghtgumcehakbxwihqrwbjyyzzyjqydciejchdxduvdenxzmyjewxsxxdhwsehflbnmowvroqtlhqlzhzogjsuzxtzvjboveushcqaibhylrbou",
    "company": "xyyfaswkniyhhhmmrqusrgfmioajcgzwfjxbfqhjhcreyihfisofoynswijbaemllzavogwvprpukhlplimlidcgathvpyz",
    "segment_1": "ppxlxgtemojmtteuyibuwiooungmfbnoemhashngdrtooqjxflvwvexrtlzrjensebduugrhewiavvimydtfcmedkxzkoomquqlfeujfczponughpsfrrxovjjpzjjcaufjsmgonazhjlilznbixabwchmvlivgvacjukeq",
    "segment_2": "qaexjmgfhbbnkqtoaattpipigmkeoajelvtyuoiompqfu",
    "segment_3": "udypryoqzamgickqiqwbobgehlqhvvbnjwbphpxdwnsizkwaktkhlmmdtkuvbvpxyrqamvllssctkhbscpzgbvcujzgwdgpcfhpkfcbaluzyosnhtgjcizbigcfbfcahomdqsvyljspyvxmdtvvijhvklzez",
    "segment_4": "eaxwisapitugvfjiqgncgsdlalxyhaxmwwsjdusmgvizhneqlfqwjwtxptqcydqtdhprxquxsglgztqqmeodtarugsotylnuxxdpgcsdlviybrnkgvoouatpujvlogjsbuywgehltojlzhlbmegscolmgwtazil",
    "segment_5": "tsblfyelzwgfradxjlstdsnuywbujqtpdvfqkawnjgwusrkqnvijwiwwuuorfpknjwgrhdqosiczvgoqigahrhvkmgsiynthwqusfmkazojjzsddgyczrbpodddnwonfacrnilskdmgdi",
    "segment_6": "fitedzudbouzfwjgsakyfcsyyagn",
    "segment_7": "ecmzjzujqyywwbpbffrmyfdugxvsqzgsxplrnmlsgebxoylidxcvakqoofnzbhruusfjzzfyygnejuqcsmudmqgywfnmwkzenyekxtsvakzwdgahikgvvqdethmklucdqlknmlyipgkawufqbqjvyprl",
    "segment_8": "mvyfbwsddkzwfntxlumgpdo",
    "segment_9": "whyrfzayizrtuneaaixvjtkgwxmum",
    "segment_10": "efgbyozdvjekaedfaz",
    "segment_mrr": 4351.5586417,
    "segment_arr": 1923362.79680344,
    "segment_ltv": 15.36269,
    "email_unsubscribed_at": "2025-01-16T20:32:40"
};

fetch(url, {
    method: "POST",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/users',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'email' => 'cbbboxbomtpbhoqxonhbkivjjxhokhmnkresxamcuggaqqrmucijyqzoxzhnxlstvodiyhxlnlmxanqnrpsnvqzanmcewkdsuhljiujldxtcdfckkkyhuqsxqkrxzqqbfedgvlchtyodchlcqyyghcxandcccekkyp',
            'role' => 'admin',
            'name' => 'eulinansrjbdszhmoovcnufpifhfhciolxasgvluizqqcsyhaiojcbgnj',
            'invitation_message' => 'kowdqhfvdrguyyuvtuonypquyrwqiqlgkjbggdwyconixmxowxnqowufievaakamciseilbuqpxwkybrvauvytwetsgascvwpobqzkegxwcjglbeorimommkqqpgz',
            'password' => 'lw',
            'created_at' => '2025-01-16T20:32:40',
            'is_blocked' => 'false',
            'verified' => 'false',
            'admin_accepted' => 'false',
            'sso_uid' => 'dotpcbmivaafbcnfyznymvlalteiqebtpeqsotvsuyicfrhzjkjyyuatlkhjdzzhvybgfxddnqyukpgwyjlubmafxrqqivqzohgymqkllipkcchsjhpztesxodntvpmsuvxbacsyggnaladpwecgaavgrfuntfxctgoafkcsgpurihowxvahzkaejmyifssxwfvbalaeirrrmwjgvcgzhlsnsnoamzprbcvqjqntkpw',
            'sso_avatar_url' => 'abbptugktayrwdnduywgxkocllumxilghtgumcehakbxwihqrwbjyyzzyjqydciejchdxduvdenxzmyjewxsxxdhwsehflbnmowvroqtlhqlzhzogjsuzxtzvjboveushcqaibhylrbou',
            'company' => 'xyyfaswkniyhhhmmrqusrgfmioajcgzwfjxbfqhjhcreyihfisofoynswijbaemllzavogwvprpukhlplimlidcgathvpyz',
            'segment_1' => 'ppxlxgtemojmtteuyibuwiooungmfbnoemhashngdrtooqjxflvwvexrtlzrjensebduugrhewiavvimydtfcmedkxzkoomquqlfeujfczponughpsfrrxovjjpzjjcaufjsmgonazhjlilznbixabwchmvlivgvacjukeq',
            'segment_2' => 'qaexjmgfhbbnkqtoaattpipigmkeoajelvtyuoiompqfu',
            'segment_3' => 'udypryoqzamgickqiqwbobgehlqhvvbnjwbphpxdwnsizkwaktkhlmmdtkuvbvpxyrqamvllssctkhbscpzgbvcujzgwdgpcfhpkfcbaluzyosnhtgjcizbigcfbfcahomdqsvyljspyvxmdtvvijhvklzez',
            'segment_4' => 'eaxwisapitugvfjiqgncgsdlalxyhaxmwwsjdusmgvizhneqlfqwjwtxptqcydqtdhprxquxsglgztqqmeodtarugsotylnuxxdpgcsdlviybrnkgvoouatpujvlogjsbuywgehltojlzhlbmegscolmgwtazil',
            'segment_5' => 'tsblfyelzwgfradxjlstdsnuywbujqtpdvfqkawnjgwusrkqnvijwiwwuuorfpknjwgrhdqosiczvgoqigahrhvkmgsiynthwqusfmkazojjzsddgyczrbpodddnwonfacrnilskdmgdi',
            'segment_6' => 'fitedzudbouzfwjgsakyfcsyyagn',
            'segment_7' => 'ecmzjzujqyywwbpbffrmyfdugxvsqzgsxplrnmlsgebxoylidxcvakqoofnzbhruusfjzzfyygnejuqcsmudmqgywfnmwkzenyekxtsvakzwdgahikgvvqdethmklucdqlknmlyipgkawufqbqjvyprl',
            'segment_8' => 'mvyfbwsddkzwfntxlumgpdo',
            'segment_9' => 'whyrfzayizrtuneaaixvjtkgwxmum',
            'segment_10' => 'efgbyozdvjekaedfaz',
            'segment_mrr' => 4351.5586417,
            'segment_arr' => 1923362.79680344,
            'segment_ltv' => 15.36269,
            'email_unsubscribed_at' => '2025-01-16T20:32:40',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/users'
payload = {
    "email": "cbbboxbomtpbhoqxonhbkivjjxhokhmnkresxamcuggaqqrmucijyqzoxzhnxlstvodiyhxlnlmxanqnrpsnvqzanmcewkdsuhljiujldxtcdfckkkyhuqsxqkrxzqqbfedgvlchtyodchlcqyyghcxandcccekkyp",
    "role": "admin",
    "name": "eulinansrjbdszhmoovcnufpifhfhciolxasgvluizqqcsyhaiojcbgnj",
    "invitation_message": "kowdqhfvdrguyyuvtuonypquyrwqiqlgkjbggdwyconixmxowxnqowufievaakamciseilbuqpxwkybrvauvytwetsgascvwpobqzkegxwcjglbeorimommkqqpgz",
    "password": "lw",
    "created_at": "2025-01-16T20:32:40",
    "is_blocked": "false",
    "verified": "false",
    "admin_accepted": "false",
    "sso_uid": "dotpcbmivaafbcnfyznymvlalteiqebtpeqsotvsuyicfrhzjkjyyuatlkhjdzzhvybgfxddnqyukpgwyjlubmafxrqqivqzohgymqkllipkcchsjhpztesxodntvpmsuvxbacsyggnaladpwecgaavgrfuntfxctgoafkcsgpurihowxvahzkaejmyifssxwfvbalaeirrrmwjgvcgzhlsnsnoamzprbcvqjqntkpw",
    "sso_avatar_url": "abbptugktayrwdnduywgxkocllumxilghtgumcehakbxwihqrwbjyyzzyjqydciejchdxduvdenxzmyjewxsxxdhwsehflbnmowvroqtlhqlzhzogjsuzxtzvjboveushcqaibhylrbou",
    "company": "xyyfaswkniyhhhmmrqusrgfmioajcgzwfjxbfqhjhcreyihfisofoynswijbaemllzavogwvprpukhlplimlidcgathvpyz",
    "segment_1": "ppxlxgtemojmtteuyibuwiooungmfbnoemhashngdrtooqjxflvwvexrtlzrjensebduugrhewiavvimydtfcmedkxzkoomquqlfeujfczponughpsfrrxovjjpzjjcaufjsmgonazhjlilznbixabwchmvlivgvacjukeq",
    "segment_2": "qaexjmgfhbbnkqtoaattpipigmkeoajelvtyuoiompqfu",
    "segment_3": "udypryoqzamgickqiqwbobgehlqhvvbnjwbphpxdwnsizkwaktkhlmmdtkuvbvpxyrqamvllssctkhbscpzgbvcujzgwdgpcfhpkfcbaluzyosnhtgjcizbigcfbfcahomdqsvyljspyvxmdtvvijhvklzez",
    "segment_4": "eaxwisapitugvfjiqgncgsdlalxyhaxmwwsjdusmgvizhneqlfqwjwtxptqcydqtdhprxquxsglgztqqmeodtarugsotylnuxxdpgcsdlviybrnkgvoouatpujvlogjsbuywgehltojlzhlbmegscolmgwtazil",
    "segment_5": "tsblfyelzwgfradxjlstdsnuywbujqtpdvfqkawnjgwusrkqnvijwiwwuuorfpknjwgrhdqosiczvgoqigahrhvkmgsiynthwqusfmkazojjzsddgyczrbpodddnwonfacrnilskdmgdi",
    "segment_6": "fitedzudbouzfwjgsakyfcsyyagn",
    "segment_7": "ecmzjzujqyywwbpbffrmyfdugxvsqzgsxplrnmlsgebxoylidxcvakqoofnzbhruusfjzzfyygnejuqcsmudmqgywfnmwkzenyekxtsvakzwdgahikgvvqdethmklucdqlknmlyipgkawufqbqjvyprl",
    "segment_8": "mvyfbwsddkzwfntxlumgpdo",
    "segment_9": "whyrfzayizrtuneaaixvjtkgwxmum",
    "segment_10": "efgbyozdvjekaedfaz",
    "segment_mrr": 4351.5586417,
    "segment_arr": 1923362.79680344,
    "segment_ltv": 15.36269,
    "email_unsubscribed_at": "2025-01-16T20:32:40"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": "4115b248-e52b-4b85-afa0-a9f439d3cad1",
        "name": "Nelda",
        "email": "irosenbaum@example.net",
        "role": "member",
        "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
        "avatar_small_url": "https://www.gravatar.com/avatar/95f7e75db727ef47cc3aafaa7a09be55?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Nelda/60/CBAACB/351435",
        "counter_votes": 0,
        "counter_comments": 0,
        "counter_posts": 0,
        "counter_comment_votes": 0,
        "created_at": "2025-01-16T20:32:40.000000Z",
        "is_blocked": null,
        "verified": null,
        "admin_accepted": null,
        "sso_uid": null,
        "sso_avatar_url": null,
        "company": null,
        "segment_1": null,
        "segment_2": null,
        "segment_3": null,
        "segment_4": null,
        "segment_5": null,
        "segment_6": null,
        "segment_7": null,
        "segment_8": null,
        "segment_9": null,
        "segment_10": null,
        "segment_mrr": null,
        "segment_arr": null,
        "segment_ltv": null,
        "email_unsubscribed_at": null
    }
}
 

Example response (422):


{
    "message": "User already exists"
}
 

Example response (422):


{
 "message": "The given data was invalid.",
 "errors": {
 "role": [
     "The selected role is invalid."
 ]
}
 

Request   

POST api/v1/users

Body Parameters

email  string optional  

Must be a valid email address. Must not be greater than 191 characters.

role  string  

Must be one of admin or member.

name  string optional  

Name for the user. Omit or null to automatically determine a name based on the email. Must not be greater than 100 characters.

invitation_message  string optional  

You can provide a custom invitation message. Omit or null to use the default message. Must not be greater than 1000 characters.

skip_email  string optional  

Provide a value to skip sending an email. Omit or null to send invitation email.

password  string optional  

Provide a password to skip sending an invitation. Omit or null to use an invitation (for which an email can be sent or not). Must not be greater than 100 characters. Must be at least 8 characters.

created_at  string optional  

The creation date of the user. Defaults to current timestamp if not provided. Must be a valid date.

is_blocked  string optional  

Whether the user is blocked from accessing the portal. Must be one of true, false, 0, or 1.

verified  string optional  

Whether the user's email is verified. Must be one of true, false, 0, or 1.

admin_accepted  string optional  

Whether the user is accepted by an admin. Must be one of true, false, 0, or 1.

sso_uid  string optional  

The user's SSO unique identifier. Must not be greater than 255 characters.

sso_avatar_url  string optional  

The URL of the user's avatar from SSO provider. Must be a valid URL. Must not be greater than 255 characters.

company  string optional  

The user's company name. Must not be greater than 255 characters.

segment_1  string optional  

Custom segment fields for user categorization. Must not be greater than 255 characters.

segment_2  string optional  

Must not be greater than 255 characters.

segment_3  string optional  

Must not be greater than 255 characters.

segment_4  string optional  

Must not be greater than 255 characters.

segment_5  string optional  

Must not be greater than 255 characters.

segment_6  string optional  

Must not be greater than 255 characters.

segment_7  string optional  

Must not be greater than 255 characters.

segment_8  string optional  

Must not be greater than 255 characters.

segment_9  string optional  

Must not be greater than 255 characters.

segment_10  string optional  

Must not be greater than 255 characters.

segment_mrr  number optional  

Monthly Recurring Revenue for the user.

segment_arr  number optional  

Annual Recurring Revenue for the user.

segment_ltv  number optional  

Lifetime Value of the user.

email_unsubscribed_at  string optional  

The date when the user unsubscribed from emails. Must be a valid date.

Get user

requires authentication

Retrieves the details of a specified user.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "id": "89219956-796b-42b6-a950-c72ac8a66e34",
        "name": "Geovanny",
        "email": "carolina02@example.net",
        "role": "member",
        "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
        "avatar_small_url": "https://www.gravatar.com/avatar/2f4a4fc3a9840c080ea5496fa93b445b?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Geovanny/60/55CBCD/003537",
        "counter_votes": 0,
        "counter_comments": 0,
        "counter_posts": 0,
        "counter_comment_votes": 0,
        "created_at": "2025-01-16T20:32:40.000000Z",
        "is_blocked": null,
        "verified": null,
        "admin_accepted": null,
        "sso_uid": null,
        "sso_avatar_url": null,
        "company": null,
        "segment_1": null,
        "segment_2": null,
        "segment_3": null,
        "segment_4": null,
        "segment_5": null,
        "segment_6": null,
        "segment_7": null,
        "segment_8": null,
        "segment_9": null,
        "segment_10": null,
        "segment_mrr": null,
        "segment_arr": null,
        "segment_ltv": null,
        "email_unsubscribed_at": null
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\User] ..."
}
 

Request   

GET api/v1/users/{id}

URL Parameters

id  string  

The ID of the user.

Update user

requires authentication

Updates the details of a specified user.

Example request:
curl --request PUT \
    "{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"email\": \"ssanford@example.net\",
    \"role\": \"member\",
    \"name\": \"vpgkcnrailhviehcexyucrcpumuakwmgpkdqsboinwg\",
    \"password\": \"g\",
    \"created_at\": \"2025-01-16T20:32:40\",
    \"is_blocked\": \"1\",
    \"verified\": \"false\",
    \"admin_accepted\": \"1\",
    \"sso_uid\": \"cusfqlftvtaeudgkxxilkvcjevhvwckksnouqsgswnddqyfiilxbtgdsdrddeycniehhgbthtdcxjpyaaclkmndidyxnaoxhgfcfortmcbwhdptdlspvunhdxmybxnyfgasmkxraaatthokviipqszwavbkifpharuxwunfbmuyhqyibxtiihtjnotsrhrkxwjzedkcsxzolqabirunashypkobpafqcjkxnjqgebqckczlsyhnwcy\",
    \"sso_avatar_url\": \"jdtvemapnremcmlgbdbhdhnhlwkqkocbueqdycfutxnmuuakdnlvrdvpjvmlfefbsebbwsyvjhhptviegpzordjhlyzbehedwxxptdsofpshdegzgshaboikpmszctzqultfgs\",
    \"company\": \"mkbnmbgtymq\",
    \"segment_1\": \"hncbkpmxtdaqoxykyekgzmyuqiyuqqndbliekitoslanmbjxvezzgubbaxkssmaevjlcwclvwvdpvrbacrlygobqpehuqkcxjysqcvlzonxrizujuscvtmnsmpbqtatcuvrmbojllir\",
    \"segment_2\": \"rkfdtyzvhxybiyyjviglltviylcubcopouhinmavvrdysrdjbxcnvudrjxvnyhzqnvkandarxyxjrjnjzwtzfafkleroyxgodbdtpexgmehozsakuhukpreqpqvdyyomhsfxzszucxzbssjgcanjeqhjygjxedhzbbrcwvnqqnkpfyfaxmroqfndiuqrhruszmoxzcitramyqwoychohkuwnkptbiuxajdosvfpry\",
    \"segment_3\": \"aylotzyevlgmplnebpxemgjymwklqnefwlnmcmngtztqtconamlxyzmbtqyznftfnieeutygprvxgjtrrkavnhfbwzunotebmckexxjppxlduxcwucneifvqqhztpkcjjurbpniamge\",
    \"segment_4\": \"nfmhhcixbiivxdeksepnryyxpvqplkzcsqiuxyqntyoilkmhsyowohhalaznleerjozgyr\",
    \"segment_5\": \"etzceiqflpsookiqmylhfgvykirndxohsocjnwyfxlrmnrgfpzwjqxphpnlycdhjemyasppwbxpkjsncjnkiauipdaydfezlkkrsdt\",
    \"segment_6\": \"lsfsjssifcvasdiafhfnajnxqajxvpddybjninjzcqhakvvduimhsmmpivszifeqfpnntcgajqhbqyywvfkibuxcpmfwrzlgwguluigfvscobfycvdrxnjrddwzrgcrbfuweddfelbxgogjkytvuzjdschbemqaqxqztymym\",
    \"segment_7\": \"euweqvlzlklbredoyzzyfvsymrwnlorgejbblrdfwnzakqlqqfjgidnexhunftufhqxsimgmeqhedxtdcsonqzpdwjwerwywnlrckmwdkmztmhfikxjbrpcgugludqysytikbuqjqgphxkbytmygjhevhlxsqczfxdcnyhhejfucpcjufbohpqdkyicgoqjmjnyh\",
    \"segment_8\": \"wulvpvvbcscwivxovqjjduqfkcyevwgaknpqolkishjnuolanvqsvlwydfvlvpennfakvodtnxmrovokysjtioucngkvrggvxktkobaztevtfs\",
    \"segment_9\": \"ouhtgwbstyetjzdfjlapxoscwhqtixxliczdjkyucqfqqmjcdpchhplgffxinbtyprmjyptpgbyxofcpnyivhxgnixbcpsdksbikiziautzqfbasflyvqivcbthjzyukucnfwhhdwchcfkdkktpbsmwjsbsmwdykrtdqzwmwkffctippbo\",
    \"segment_10\": \"txhujvtbwdimahntctutfsknrquvqmdlwqowvjyajrowxqrpdphzgftxvndmuflrwtjhrlahxskqytclkszmopgmyyulgsixqpblksneoolvnhzsikescwfijvhypuizlcnsetfkifccodzyegssdncgpwobgpkejvzovdwqsnewmqnwikejgsvzoqvpflphroxbzbsnlwdwtmk\",
    \"segment_mrr\": 27242.224414,
    \"segment_arr\": 13943001.91,
    \"segment_ltv\": 38.6907985,
    \"email_unsubscribed_at\": \"2025-01-16T20:32:40\"
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "email": "ssanford@example.net",
    "role": "member",
    "name": "vpgkcnrailhviehcexyucrcpumuakwmgpkdqsboinwg",
    "password": "g",
    "created_at": "2025-01-16T20:32:40",
    "is_blocked": "1",
    "verified": "false",
    "admin_accepted": "1",
    "sso_uid": "cusfqlftvtaeudgkxxilkvcjevhvwckksnouqsgswnddqyfiilxbtgdsdrddeycniehhgbthtdcxjpyaaclkmndidyxnaoxhgfcfortmcbwhdptdlspvunhdxmybxnyfgasmkxraaatthokviipqszwavbkifpharuxwunfbmuyhqyibxtiihtjnotsrhrkxwjzedkcsxzolqabirunashypkobpafqcjkxnjqgebqckczlsyhnwcy",
    "sso_avatar_url": "jdtvemapnremcmlgbdbhdhnhlwkqkocbueqdycfutxnmuuakdnlvrdvpjvmlfefbsebbwsyvjhhptviegpzordjhlyzbehedwxxptdsofpshdegzgshaboikpmszctzqultfgs",
    "company": "mkbnmbgtymq",
    "segment_1": "hncbkpmxtdaqoxykyekgzmyuqiyuqqndbliekitoslanmbjxvezzgubbaxkssmaevjlcwclvwvdpvrbacrlygobqpehuqkcxjysqcvlzonxrizujuscvtmnsmpbqtatcuvrmbojllir",
    "segment_2": "rkfdtyzvhxybiyyjviglltviylcubcopouhinmavvrdysrdjbxcnvudrjxvnyhzqnvkandarxyxjrjnjzwtzfafkleroyxgodbdtpexgmehozsakuhukpreqpqvdyyomhsfxzszucxzbssjgcanjeqhjygjxedhzbbrcwvnqqnkpfyfaxmroqfndiuqrhruszmoxzcitramyqwoychohkuwnkptbiuxajdosvfpry",
    "segment_3": "aylotzyevlgmplnebpxemgjymwklqnefwlnmcmngtztqtconamlxyzmbtqyznftfnieeutygprvxgjtrrkavnhfbwzunotebmckexxjppxlduxcwucneifvqqhztpkcjjurbpniamge",
    "segment_4": "nfmhhcixbiivxdeksepnryyxpvqplkzcsqiuxyqntyoilkmhsyowohhalaznleerjozgyr",
    "segment_5": "etzceiqflpsookiqmylhfgvykirndxohsocjnwyfxlrmnrgfpzwjqxphpnlycdhjemyasppwbxpkjsncjnkiauipdaydfezlkkrsdt",
    "segment_6": "lsfsjssifcvasdiafhfnajnxqajxvpddybjninjzcqhakvvduimhsmmpivszifeqfpnntcgajqhbqyywvfkibuxcpmfwrzlgwguluigfvscobfycvdrxnjrddwzrgcrbfuweddfelbxgogjkytvuzjdschbemqaqxqztymym",
    "segment_7": "euweqvlzlklbredoyzzyfvsymrwnlorgejbblrdfwnzakqlqqfjgidnexhunftufhqxsimgmeqhedxtdcsonqzpdwjwerwywnlrckmwdkmztmhfikxjbrpcgugludqysytikbuqjqgphxkbytmygjhevhlxsqczfxdcnyhhejfucpcjufbohpqdkyicgoqjmjnyh",
    "segment_8": "wulvpvvbcscwivxovqjjduqfkcyevwgaknpqolkishjnuolanvqsvlwydfvlvpennfakvodtnxmrovokysjtioucngkvrggvxktkobaztevtfs",
    "segment_9": "ouhtgwbstyetjzdfjlapxoscwhqtixxliczdjkyucqfqqmjcdpchhplgffxinbtyprmjyptpgbyxofcpnyivhxgnixbcpsdksbikiziautzqfbasflyvqivcbthjzyukucnfwhhdwchcfkdkktpbsmwjsbsmwdykrtdqzwmwkffctippbo",
    "segment_10": "txhujvtbwdimahntctutfsknrquvqmdlwqowvjyajrowxqrpdphzgftxvndmuflrwtjhrlahxskqytclkszmopgmyyulgsixqpblksneoolvnhzsikescwfijvhypuizlcnsetfkifccodzyegssdncgpwobgpkejvzovdwqsnewmqnwikejgsvzoqvpflphroxbzbsnlwdwtmk",
    "segment_mrr": 27242.224414,
    "segment_arr": 13943001.91,
    "segment_ltv": 38.6907985,
    "email_unsubscribed_at": "2025-01-16T20:32:40"
};

fetch(url, {
    method: "PUT",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->put(
    '{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'email' => 'ssanford@example.net',
            'role' => 'member',
            'name' => 'vpgkcnrailhviehcexyucrcpumuakwmgpkdqsboinwg',
            'password' => 'g',
            'created_at' => '2025-01-16T20:32:40',
            'is_blocked' => '1',
            'verified' => 'false',
            'admin_accepted' => '1',
            'sso_uid' => 'cusfqlftvtaeudgkxxilkvcjevhvwckksnouqsgswnddqyfiilxbtgdsdrddeycniehhgbthtdcxjpyaaclkmndidyxnaoxhgfcfortmcbwhdptdlspvunhdxmybxnyfgasmkxraaatthokviipqszwavbkifpharuxwunfbmuyhqyibxtiihtjnotsrhrkxwjzedkcsxzolqabirunashypkobpafqcjkxnjqgebqckczlsyhnwcy',
            'sso_avatar_url' => 'jdtvemapnremcmlgbdbhdhnhlwkqkocbueqdycfutxnmuuakdnlvrdvpjvmlfefbsebbwsyvjhhptviegpzordjhlyzbehedwxxptdsofpshdegzgshaboikpmszctzqultfgs',
            'company' => 'mkbnmbgtymq',
            'segment_1' => 'hncbkpmxtdaqoxykyekgzmyuqiyuqqndbliekitoslanmbjxvezzgubbaxkssmaevjlcwclvwvdpvrbacrlygobqpehuqkcxjysqcvlzonxrizujuscvtmnsmpbqtatcuvrmbojllir',
            'segment_2' => 'rkfdtyzvhxybiyyjviglltviylcubcopouhinmavvrdysrdjbxcnvudrjxvnyhzqnvkandarxyxjrjnjzwtzfafkleroyxgodbdtpexgmehozsakuhukpreqpqvdyyomhsfxzszucxzbssjgcanjeqhjygjxedhzbbrcwvnqqnkpfyfaxmroqfndiuqrhruszmoxzcitramyqwoychohkuwnkptbiuxajdosvfpry',
            'segment_3' => 'aylotzyevlgmplnebpxemgjymwklqnefwlnmcmngtztqtconamlxyzmbtqyznftfnieeutygprvxgjtrrkavnhfbwzunotebmckexxjppxlduxcwucneifvqqhztpkcjjurbpniamge',
            'segment_4' => 'nfmhhcixbiivxdeksepnryyxpvqplkzcsqiuxyqntyoilkmhsyowohhalaznleerjozgyr',
            'segment_5' => 'etzceiqflpsookiqmylhfgvykirndxohsocjnwyfxlrmnrgfpzwjqxphpnlycdhjemyasppwbxpkjsncjnkiauipdaydfezlkkrsdt',
            'segment_6' => 'lsfsjssifcvasdiafhfnajnxqajxvpddybjninjzcqhakvvduimhsmmpivszifeqfpnntcgajqhbqyywvfkibuxcpmfwrzlgwguluigfvscobfycvdrxnjrddwzrgcrbfuweddfelbxgogjkytvuzjdschbemqaqxqztymym',
            'segment_7' => 'euweqvlzlklbredoyzzyfvsymrwnlorgejbblrdfwnzakqlqqfjgidnexhunftufhqxsimgmeqhedxtdcsonqzpdwjwerwywnlrckmwdkmztmhfikxjbrpcgugludqysytikbuqjqgphxkbytmygjhevhlxsqczfxdcnyhhejfucpcjufbohpqdkyicgoqjmjnyh',
            'segment_8' => 'wulvpvvbcscwivxovqjjduqfkcyevwgaknpqolkishjnuolanvqsvlwydfvlvpennfakvodtnxmrovokysjtioucngkvrggvxktkobaztevtfs',
            'segment_9' => 'ouhtgwbstyetjzdfjlapxoscwhqtixxliczdjkyucqfqqmjcdpchhplgffxinbtyprmjyptpgbyxofcpnyivhxgnixbcpsdksbikiziautzqfbasflyvqivcbthjzyukucnfwhhdwchcfkdkktpbsmwjsbsmwdykrtdqzwmwkffctippbo',
            'segment_10' => 'txhujvtbwdimahntctutfsknrquvqmdlwqowvjyajrowxqrpdphzgftxvndmuflrwtjhrlahxskqytclkszmopgmyyulgsixqpblksneoolvnhzsikescwfijvhypuizlcnsetfkifccodzyegssdncgpwobgpkejvzovdwqsnewmqnwikejgsvzoqvpflphroxbzbsnlwdwtmk',
            'segment_mrr' => 27242.224414,
            'segment_arr' => 13943001.91,
            'segment_ltv' => 38.6907985,
            'email_unsubscribed_at' => '2025-01-16T20:32:40',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88'
payload = {
    "email": "ssanford@example.net",
    "role": "member",
    "name": "vpgkcnrailhviehcexyucrcpumuakwmgpkdqsboinwg",
    "password": "g",
    "created_at": "2025-01-16T20:32:40",
    "is_blocked": "1",
    "verified": "false",
    "admin_accepted": "1",
    "sso_uid": "cusfqlftvtaeudgkxxilkvcjevhvwckksnouqsgswnddqyfiilxbtgdsdrddeycniehhgbthtdcxjpyaaclkmndidyxnaoxhgfcfortmcbwhdptdlspvunhdxmybxnyfgasmkxraaatthokviipqszwavbkifpharuxwunfbmuyhqyibxtiihtjnotsrhrkxwjzedkcsxzolqabirunashypkobpafqcjkxnjqgebqckczlsyhnwcy",
    "sso_avatar_url": "jdtvemapnremcmlgbdbhdhnhlwkqkocbueqdycfutxnmuuakdnlvrdvpjvmlfefbsebbwsyvjhhptviegpzordjhlyzbehedwxxptdsofpshdegzgshaboikpmszctzqultfgs",
    "company": "mkbnmbgtymq",
    "segment_1": "hncbkpmxtdaqoxykyekgzmyuqiyuqqndbliekitoslanmbjxvezzgubbaxkssmaevjlcwclvwvdpvrbacrlygobqpehuqkcxjysqcvlzonxrizujuscvtmnsmpbqtatcuvrmbojllir",
    "segment_2": "rkfdtyzvhxybiyyjviglltviylcubcopouhinmavvrdysrdjbxcnvudrjxvnyhzqnvkandarxyxjrjnjzwtzfafkleroyxgodbdtpexgmehozsakuhukpreqpqvdyyomhsfxzszucxzbssjgcanjeqhjygjxedhzbbrcwvnqqnkpfyfaxmroqfndiuqrhruszmoxzcitramyqwoychohkuwnkptbiuxajdosvfpry",
    "segment_3": "aylotzyevlgmplnebpxemgjymwklqnefwlnmcmngtztqtconamlxyzmbtqyznftfnieeutygprvxgjtrrkavnhfbwzunotebmckexxjppxlduxcwucneifvqqhztpkcjjurbpniamge",
    "segment_4": "nfmhhcixbiivxdeksepnryyxpvqplkzcsqiuxyqntyoilkmhsyowohhalaznleerjozgyr",
    "segment_5": "etzceiqflpsookiqmylhfgvykirndxohsocjnwyfxlrmnrgfpzwjqxphpnlycdhjemyasppwbxpkjsncjnkiauipdaydfezlkkrsdt",
    "segment_6": "lsfsjssifcvasdiafhfnajnxqajxvpddybjninjzcqhakvvduimhsmmpivszifeqfpnntcgajqhbqyywvfkibuxcpmfwrzlgwguluigfvscobfycvdrxnjrddwzrgcrbfuweddfelbxgogjkytvuzjdschbemqaqxqztymym",
    "segment_7": "euweqvlzlklbredoyzzyfvsymrwnlorgejbblrdfwnzakqlqqfjgidnexhunftufhqxsimgmeqhedxtdcsonqzpdwjwerwywnlrckmwdkmztmhfikxjbrpcgugludqysytikbuqjqgphxkbytmygjhevhlxsqczfxdcnyhhejfucpcjufbohpqdkyicgoqjmjnyh",
    "segment_8": "wulvpvvbcscwivxovqjjduqfkcyevwgaknpqolkishjnuolanvqsvlwydfvlvpennfakvodtnxmrovokysjtioucngkvrggvxktkobaztevtfs",
    "segment_9": "ouhtgwbstyetjzdfjlapxoscwhqtixxliczdjkyucqfqqmjcdpchhplgffxinbtyprmjyptpgbyxofcpnyivhxgnixbcpsdksbikiziautzqfbasflyvqivcbthjzyukucnfwhhdwchcfkdkktpbsmwjsbsmwdykrtdqzwmwkffctippbo",
    "segment_10": "txhujvtbwdimahntctutfsknrquvqmdlwqowvjyajrowxqrpdphzgftxvndmuflrwtjhrlahxskqytclkszmopgmyyulgsixqpblksneoolvnhzsikescwfijvhypuizlcnsetfkifccodzyegssdncgpwobgpkejvzovdwqsnewmqnwikejgsvzoqvpflphroxbzbsnlwdwtmk",
    "segment_mrr": 27242.224414,
    "segment_arr": 13943001.91,
    "segment_ltv": 38.6907985,
    "email_unsubscribed_at": "2025-01-16T20:32:40"
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('PUT', url, headers=headers, json=payload)
response.json()

Example response (200):


{
    "data": {
        "id": "b6931fef-1a3f-4f80-ae5c-24b577f25716",
        "name": "Duncan",
        "email": "cormier.brenda@example.org",
        "role": "member",
        "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
        "avatar_small_url": "https://www.gravatar.com/avatar/30c44f7a1abb97487ff0c06240b1e16e?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Duncan/60/FFDBCC/694536",
        "counter_votes": 0,
        "counter_comments": 0,
        "counter_posts": 0,
        "counter_comment_votes": 0,
        "created_at": "2025-01-16T20:32:40.000000Z",
        "is_blocked": null,
        "verified": null,
        "admin_accepted": null,
        "sso_uid": null,
        "sso_avatar_url": null,
        "company": null,
        "segment_1": null,
        "segment_2": null,
        "segment_3": null,
        "segment_4": null,
        "segment_5": null,
        "segment_6": null,
        "segment_7": null,
        "segment_8": null,
        "segment_9": null,
        "segment_10": null,
        "segment_mrr": null,
        "segment_arr": null,
        "segment_ltv": null,
        "email_unsubscribed_at": null
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\User] ..."
}
 

Example response (422):


{
 "message": "The given data was invalid.",
 "errors": {
 "role": [
     "The selected role is invalid."
 ]
}
 

Request   

PUT api/v1/users/{id}

PATCH api/v1/users/{id}

URL Parameters

id  string  

The ID of the user.

Body Parameters

email  string optional  

Make null or omit to keep current value. Must be unique in the portal. Must be a valid email address.

role  string optional  

Make null or omit to keep current value. Must be one of admin or member.

name  string optional  

Make null or omit to keep current value. Must not be greater than 100 characters.

password  string optional  

Make null or omit to keep current value. Must not be greater than 100 characters. Must be at least 8 characters.

created_at  string optional  

The creation date of the user. Must be a valid date.

is_blocked  string optional  

Whether the user is blocked from accessing the portal. Must be one of true, false, 0, or 1.

verified  string optional  

Whether the user's email is verified. Must be one of true, false, 0, or 1.

admin_accepted  string optional  

Whether the user is accepted by an admin. Must be one of true, false, 0, or 1.

sso_uid  string optional  

The user's SSO unique identifier. Must not be greater than 255 characters.

sso_avatar_url  string optional  

The URL of the user's avatar from SSO provider. Must be a valid URL. Must not be greater than 255 characters.

company  string optional  

The user's company name. Must not be greater than 255 characters.

segment_1  string optional  

Custom segment fields for user categorization. Must not be greater than 255 characters.

segment_2  string optional  

Must not be greater than 255 characters.

segment_3  string optional  

Must not be greater than 255 characters.

segment_4  string optional  

Must not be greater than 255 characters.

segment_5  string optional  

Must not be greater than 255 characters.

segment_6  string optional  

Must not be greater than 255 characters.

segment_7  string optional  

Must not be greater than 255 characters.

segment_8  string optional  

Must not be greater than 255 characters.

segment_9  string optional  

Must not be greater than 255 characters.

segment_10  string optional  

Must not be greater than 255 characters.

segment_mrr  number optional  

Monthly Recurring Revenue for the user.

segment_arr  number optional  

Annual Recurring Revenue for the user.

segment_ltv  number optional  

Lifetime Value of the user.

email_unsubscribed_at  string optional  

The date when the user unsubscribed from emails. Must be a valid date.

Delete user

requires authentication

Deletes a specified user.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json" \
    --data "{
    \"delete_all_data\": true
}"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

let body = {
    "delete_all_data": true
};

fetch(url, {
    method: "DELETE",
    headers,
    body: JSON.stringify(body),
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
        'json' => [
            'delete_all_data' => true,
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/users/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88'
payload = {
    "delete_all_data": true
}
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers, json=payload)
response.json()

Example response (200):


{}
 

Example response (404):


{
    "message": "No query results for model [App\\User] ..."
}
 

Request   

DELETE api/v1/users/{id}

URL Parameters

id  string  

The ID of the user.

Body Parameters

delete_all_data  boolean optional  

Set true or 1 to delete all data from this user. Otherwise, posts, comments, will remain with [User Deleted] as the username.

Votes

Manage votes on your posts.

List votes

requires authentication

Lists all votes of a post.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": [
        {
            "created_at": "2020-06-09T14:40:38.000000Z",
            "voter": {
                "id": "2850e389-2827-4316-a50a-fcdcc6474b5d",
                "name": "Jordi",
                "email": "descubre@exploradoresdemartech.com",
                "role": "member",
                "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
                "avatar_small_url": "https://www.gravatar.com/avatar/72802d9f5cd0781807579a89a3947b75?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Jordi/60/CCE2CB/364c35",
                "counter_votes": 3,
                "counter_comments": 0,
                "counter_posts": 0,
                "counter_comment_votes": 0,
                "created_at": "2020-06-09T14:40:23.000000Z",
                "is_blocked": 0,
                "verified": false,
                "admin_accepted": 1,
                "sso_uid": null,
                "sso_avatar_url": null,
                "company": null,
                "segment_1": null,
                "segment_2": null,
                "segment_3": null,
                "segment_4": null,
                "segment_5": null,
                "segment_6": null,
                "segment_7": null,
                "segment_8": null,
                "segment_9": null,
                "segment_10": null,
                "segment_mrr": null,
                "segment_arr": null,
                "segment_ltv": null,
                "email_unsubscribed_at": null
            }
        },
        {
            "created_at": "2020-06-09T14:40:38.000000Z",
            "voter": {
                "id": "2850e389-2827-4316-a50a-fcdcc6474b5d",
                "name": "Jordi",
                "email": "descubre@exploradoresdemartech.com",
                "role": "member",
                "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
                "avatar_small_url": "https://www.gravatar.com/avatar/72802d9f5cd0781807579a89a3947b75?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Jordi/60/CCE2CB/364c35",
                "counter_votes": 3,
                "counter_comments": 0,
                "counter_posts": 0,
                "counter_comment_votes": 0,
                "created_at": "2020-06-09T14:40:23.000000Z",
                "is_blocked": 0,
                "verified": false,
                "admin_accepted": 1,
                "sso_uid": null,
                "sso_avatar_url": null,
                "company": null,
                "segment_1": null,
                "segment_2": null,
                "segment_3": null,
                "segment_4": null,
                "segment_5": null,
                "segment_6": null,
                "segment_7": null,
                "segment_8": null,
                "segment_9": null,
                "segment_10": null,
                "segment_mrr": null,
                "segment_arr": null,
                "segment_ltv": null,
                "email_unsubscribed_at": null
            }
        }
    ]
}
 

Request   

GET api/v1/posts/{key}/votes

URL Parameters

key  string  

The ID of the post.

Revoke vote

requires authentication

Revoke a user's vote for a post.

Example request:
curl --request DELETE \
    "{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "DELETE",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->delete(
    '{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('DELETE', url, headers=headers)
response.json()

Example response (200):


{}
 

Example response (404):


{
    "message": "No query results for model [App\\User] ..."
}
 

Example response (404):


{
    "message": "No query results for model [App\\Post] ..."
}
 

Example response (422):


{
    "message": "Post is not voteable."
}
 

Example response (422):


{
    "message": "User has not voted yet."
}
 

Request   

DELETE api/v1/posts/{key}/votes/{user}

URL Parameters

key  string  

The ID of the post.

user  string  

The ID of the user.

Vote

requires authentication

Add the vote of a user to a post.

Example request:
curl --request POST \
    "{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "POST",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->post(
    '{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/posts/0UFQQh/votes/8cbfbfd6-ecc3-4ec3-8b31-ea76b11c1f88'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('POST', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "created_at": "2020-06-09T14:40:38.000000Z",
        "voter": {
            "id": "2850e389-2827-4316-a50a-fcdcc6474b5d",
            "name": "Jordi",
            "email": "descubre@exploradoresdemartech.com",
            "role": "member",
            "profile_url": "{YOUR-PORTAL-URL}/user/profile/123456",
            "avatar_small_url": "https://www.gravatar.com/avatar/72802d9f5cd0781807579a89a3947b75?s=60&d=https%3A%2F%2Fui-avatars.com%2Fapi%2F/Jordi/60/CCE2CB/364c35",
            "counter_votes": 3,
            "counter_comments": 0,
            "counter_posts": 0,
            "counter_comment_votes": 0,
            "created_at": "2020-06-09T14:40:23.000000Z",
            "is_blocked": 0,
            "verified": false,
            "admin_accepted": 1,
            "sso_uid": null,
            "sso_avatar_url": null,
            "company": null,
            "segment_1": null,
            "segment_2": null,
            "segment_3": null,
            "segment_4": null,
            "segment_5": null,
            "segment_6": null,
            "segment_7": null,
            "segment_8": null,
            "segment_9": null,
            "segment_10": null,
            "segment_mrr": null,
            "segment_arr": null,
            "segment_ltv": null,
            "email_unsubscribed_at": null
        }
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\User] ..."
}
 

Example response (404):


{
    "message": "No query results for model [App\\Post] ..."
}
 

Example response (422):


{
    "message": "Post is not voteable."
}
 

Example response (422):


{
    "message": "User has already voted for this post."
}
 

Request   

POST api/v1/posts/{key}/votes/{user}

URL Parameters

key  string  

The ID of the post.

user  string  

The ID of the user.

Widgets

API endpoints for managing widgets in your portal.

List widgets

requires authentication

Retrieves a collection of all widgets in your portal.

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/widgets" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/widgets"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/widgets',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/widgets'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": [
        {
            "id": "widget1",
            "name": "Feedback Widget",
            "type": "feedback",
            "created_at": "2023-01-01T00:00:00Z",
            "updated_at": "2023-01-01T00:00:00Z"
        },
        {
            "id": "widget2",
            "name": "Survey Widget",
            "type": "survey",
            "created_at": "2023-01-02T00:00:00Z",
            "updated_at": "2023-01-02T00:00:00Z"
        }
    ]
}
 

Request   

GET api/v1/widgets

Get widget

requires authentication

Retrieves the details of a specific widget by its key (ID).

Example request:
curl --request GET \
    --get "{YOUR_PORTAL_URL}/api/v1/widgets/widget1" \
    --header "Authorization: Bearer {YOUR_AUTH_KEY}" \
    --header "Content-Type: application/json" \
    --header "Accept: application/json"
const url = new URL(
    "{YOUR_PORTAL_URL}/api/v1/widgets/widget1"
);

const headers = {
    "Authorization": "Bearer {YOUR_AUTH_KEY}",
    "Content-Type": "application/json",
    "Accept": "application/json",
};

fetch(url, {
    method: "GET",
    headers,
}).then(response => response.json());
$client = new \GuzzleHttp\Client();
$response = $client->get(
    '{YOUR_PORTAL_URL}/api/v1/widgets/widget1',
    [
        'headers' => [
            'Authorization' => 'Bearer {YOUR_AUTH_KEY}',
            'Content-Type' => 'application/json',
            'Accept' => 'application/json',
        ],
    ]
);
$body = $response->getBody();
print_r(json_decode((string) $body));
import requests
import json

url = '{YOUR_PORTAL_URL}/api/v1/widgets/widget1'
headers = {
  'Authorization': 'Bearer {YOUR_AUTH_KEY}',
  'Content-Type': 'application/json',
  'Accept': 'application/json'
}

response = requests.request('GET', url, headers=headers)
response.json()

Example response (200):


{
    "data": {
        "id": "widget1",
        "name": "Feedback Widget",
        "type": "feedback",
        "created_at": "2023-01-01T00:00:00Z",
        "updated_at": "2023-01-01T00:00:00Z"
    }
}
 

Example response (404):


{
    "message": "No query results for model [App\\Widget] widget1"
}
 

Request   

GET api/v1/widgets/{key}

URL Parameters

key  string  

The unique identifier of the widget.