Skip navigation
Documentation

Accounts API

Last Updated: April 12th, 2024

Contents

If you are an MSP with questions about the Accounts API, please contact the Duo MSP team.

The Accounts API allows a parent account to create, manage, and delete other Duo Security customer accounts.

The Accounts API allows Duo Premier, Advantage, and Essentials customers to create, manage, and delete additional related Duo Security customer accounts.

Overview

The Accounts API lets customers programmatically create, delete, and manage individual Duo customer accounts. New Duo accounts created using the Accounts API are subaccounts of the account where the Accounts API application exists, creating a "parent" and "child" account relationship. A Duo account can have multiple child accounts, but a child account may only have one parent and no child accounts of its own.

About Duo APIs

Documented properties will not be removed within a stable version of the API. Once a given API endpoint is documented to return a given property, a property with that name will always appear (although certain properties may only appear under certain conditions, like if the customer is using a specific edition).

Properties that enumerate choices may gain new values at any time, e.g. the device platform value could return new device platforms that did not previously exist. Duo will update our API documentation with new values in a timely fashion.

New, undocumented properties may also appear at any time. For instance, Duo may make available a beta feature involving extra information returned by an API endpoint. Until the property is documented here its format may change or it may even be entirely removed from our API.

First Steps

Role required: Owner

Note that only administrators with the Owner role in the parent account may contact Duo Support to request access to the Accounts API application, or can create or modify an Accounts API application in the Duo Admin Panel.

  1. Sign up for a Duo account.

  2. Log in to the Duo Admin Panel and navigate to Applications.

  3. Click Protect an Application and locate the entry for Accounts API in the applications list. Click Protect to the far-right to configure the application and get your integration key, secret key, and API hostname. You'll need this information to complete your setup. See Protecting Applications for more information about protecting applications in Duo and additional application options.

    Treat your secret key like a password
    The security of your Duo application is tied to the security of your secret key (skey). Secure it as you would any sensitive credential. Don't share it with unauthorized individuals or email it to anyone under any circumstances!
  4. Optionally specify which IP addresses or ranges are allowed to use this Accounts API application in Networks for API Access. If you do not specify any IP addresses or ranges, this Accounts API application may be accessed from any network.

    The Accounts API performs the IP check occurs after verifying the authentication signature in a request. If you restrict the allowed networks for API access and see logged events for blocked Accounts API requests from unrecognized IP addresses, this may indicate compromise of your Accounts API application's secret key.

Connectivity Requirements

This application communicates with Duo's service on SSL TCP port 443.

Firewall configurations that restrict outbound access to Duo's service with rules using destination IP addresses or IP address ranges aren't recommended, since these may change over time to maintain our service's high availability. If your organization requires IP-based rules, please review Duo Knowledge Base article 1337.

Effective June 30, 2023, Duo no longer supports TLS 1.0 or 1.1 connections or insecure TLS/SSL cipher suites. See Duo Knowledge Base article 7546 for additional guidance.

API Clients

Duo Security has demonstration clients available on GitHub to call the Duo API methods.

API Methods

Retrieve Accounts

Returns a list of child accounts.

POST /accounts/v1/account/list

Parameters

None.

Response codes

Response Meaning
200 Success. Returns a list of accounts.

Response format

Key Value
account_id The child customer account ID. This is a 20 character string, for example DA9VZOC5X63I2W72NRP9.
name The customer's name.
api_hostname Use this hostname instead of the parent account's when configuring the customer's applications.

Example response

{
  "stat": "OK",
  "response": [{
    "account_id": "DA9VZOC5X63I2W72NRP9",
    "api_hostname": "api-abcd1234.duosecurity.com",
    "name": "Acme Corp"
  }]
}

Create Account

Create a new child account.

POST /accounts/v1/account/create

Parameters

Parameter Required? Description
name Required Name for the new customer.

Response codes

Response Meaning
200 Success. Returns the newly created account.
400 Invalid or missing parameters.

Response format

Same as Retrieve Accounts.

Example response

{
  "stat": "OK",
  "response": {
    "account_id": "DA9VZOC5X63I2W72NRP9",
    "api_hostname": "api-abcd1234.duosecurity.com",
    "name": "Acme Corp"
  }
}

Delete Account

Delete the account with ID account_id from the system.

POST /accounts/v1/account/delete

Parameters

Parameter Required? Description
account_id Required ID of the customer account to delete as returned by Retrieve Accounts. This is a 20 character string, for example DA9VZOC5X63I2W72NRP9.

Response codes

Response Meaning
200 The account was deleted or did not exist.
409 The account is the parent of one or more child accounts. It cannot be deleted until the child accounts are deleted.

Response format

Empty string.

Example response

{
  "stat": "OK",
  "response": ""
}

Using Accounts API with Admin API

Duo's Admin API supports programmatic administration of Duo account objects, like users, 2FA devices, integrations, logs, and more. You can use an Accounts API application created in a parent account to manage and query child accounts with Admin API by specifying the child account's API host in the request and the child account's ID in the list of API request parameters as account_id.

There is no need to create separate Admin API applications in the child accounts. The Accounts API application created in the parent is automatically granted all Admin API permissions; these grants can't be modified.

Here's an example URL-encoded GET query string on the Integrations Admin API endpoint that retrieves a list of integrations in a child account specified with account_id:

GET /admin/v1/integrations?account_id=DAAARXMAKLQ2ZKD571YC

Here's an example POST on the Create User Admin API endpoint that creates a new user in the child account specified with account_id:

POST /admin/v1/users?account_id=DAAARXMAKLQ2ZKD571YC&email=narroway@example.com&realname=Norben%20Arroway&username=narroway

When making the request, ensure you do the following:

  • Change the host value used to construct the signature from the api_hostname of the parent account to the api_hostname of the child account.
  • Send the request to the child account's API hostname as retrieved with Retrieve Accounts.
  • Specify the subaccount account_id in the request path.

For example, if the following information was returned for a given child account via /accounts/v1/account/list:

{
  "stat": "OK",
  "response": {
    "account_id": "DA9VZOC5X63I2W72NRP9",
    "api_hostname": "api-abcd1234.duosecurity.com",
    "name": "Acme Corp"
  }
}

Admin API requests via Accounts API to that child should use the api_hostname value api-abcd1234.duosecurity.com.

Additional Admin API Methods

These Admin API methods are not yet generally available with Accounts API. Please contact Duo Support to request access to these methods.

Get Edition

Returns the Duo edition information for a child account.

GET /admin/v1/billing/edition

Parameters

Parameter Required? Description
account_id Required The child customer account ID as returned by Retrieve Accounts. This is a 20 character string, for example DA9VZOC5X63I2W72NRP9.

Response codes

Response Meaning
200 Success. Returns the edition of the child account.
400 Invalid or missing parameters.
401 Invalid integration key in request credentials.

Response format

Key Value
edition The edition. One of:
  • PERSONAL - The Duo Free edition.
  • ENTERPRISE - The Duo Essentials edition.
  • PLATFORM - The Duo Advantage edition.
  • BEYOND - The Duo Premier edition.

Example response

{
  "stat": "OK",
  "response": {"edition": "PLATFORM"}
}

Set Edition

Sets the effective Duo edition for a child account.

POST /admin/v1/billing/edition

Parameters

Parameter Required? Description
account_id Required The child customer account ID as returned by Retrieve Accounts. This is a 20 character string, for example DA9VZOC5X63I2W72NRP9.
edition Required The edition to set. This should be one of:
  • ENTERPRISE - The Duo Essentials edition.
  • PLATFORM - The Duo Advantage edition.
  • BEYOND - The Duo Premier edition.

Response codes

Response Meaning
200 Success.
400 Invalid or missing parameters.
401 Invalid integration key in request credentials.

Response format

The response is empty.

Example response

{
  "stat": "OK",
  "response": ""
}

Get Telephony Credits

Returns the available telephony credits for a child account.

GET /admin/v1/billing/telephony_credits

Parameters

Parameter Required? Description
account_id Required The child customer account ID as returned by Retrieve Accounts. This is a 20 character string, for example DA9VZOC5X63I2W72NRP9.

Response codes

Response Meaning
200 Success. Returns the available telephony credits of the child account.
400 Invalid or missing parameters.
401 Invalid integration key in request credentials.

Response format

Key Value
credits The available telephony credits.

Example response

{
  "stat": "OK",
  "response": {"credits": 10}
}

Set Telephony Credits

Sets the telephony credits for a child account.

Any additional credits added to the child account are transferred from the parent account. For example, if the child account has 100 credits and it is then set to 300 credits, then 200 credits are deducted from the parent's balance and added to the child's balance.

POST /admin/v1/billing/telephony_credits

Parameters

Parameter Required? Description
account_id Required The child customer account ID as returned by Retrieve Accounts. This is a 20 character string, for example DA9VZOC5X63I2W72NRP9.
credits Required The total number of credits that the child account will have after transferring credits from the parent account.

Response codes

Response Meaning
200 Success.
400 Invalid or missing parameters.
401 Invalid integration key in request credentials.

Response format

Key Value
credits_added The number of telephony credits that were added to the child account and deducted from the parent account.

Example response

{
  "stat": "OK",
  "response": {"credits_added":10}
}

API Details

Base URL

All API methods use your API hostname, https://api-XXXXXXXX.duosecurity.com. Obtain this value from the Duo Admin Panel and use it exactly as shown there.

Methods always use HTTPS. Unsecured HTTP is not supported.

Request Format

All requests must have "Authorization" and "Date" headers.

If the request method is GET or DELETE, URL-encode parameters and send them in the URL query string like this: ?realname=First%20Last&username=root. They still go on a separate line when creating the string to sign for an Authorization header.

Send parameters for POST requests in the body as URL-encoded key-value pairs (the same request format used by browsers to submit form data). The header "Content-Type: application/x-www-form-urlencoded" must also be present.

When URL-encoding, all bytes except ASCII letters, digits, underscore ("_"), period ("."), tilde ("~"), and hyphen ("-") are replaced by a percent sign ("%") followed by two hexadecimal digits containing the value of the byte. For example, a space is replaced with "%20" and an at-sign ("@") becomes "%40". Use only upper-case A through F for hexadecimal digits.

A request with parameters, as a complete URL, would look something like this: https://api-XXXXXXXX.duosecurity.com/admin/v1/users?realname=First%20Last&username=root (substituting the actual API method path and parameters used in your request).

Response Format

Responses are formatted as a JSON object with a top-level stat key.

Successful responses will have a stat value of "OK" and a response key. The response will either be a single object or a sequence of other JSON types, depending on which endpoint is called.

{
  "stat": "OK",
  "response": {
    "key": "value"
  }
}

Values are returned as strings unless otherwise documented.

Unsuccessful responses will have a stat value of "FAIL", an integer code, and a message key that further describes the failure. A message_detail key may be present if additional information is available (like the specific parameter that caused the error).

{
  "stat": "FAIL",
  "code": 40002,
  "message": "Invalid request parameters",
  "message_detail": "username"
}

The HTTP response code will be the first three digits of the more specific code found inside the JSON object. Each endpoint's documentation lists HTTP response codes it can return. Additionally, all API endpoints that require a signed request can return the following HTTP response codes:

Response Meaning
200 The request completed successfully.
401 The "Authorization", "Date", and/or "Content-Type" headers were missing or invalid.
403

This integration is not authorized for this endpoint or the ikey was created for a different integration type (for example, using an Auth API ikey with Admin API endpoints).

405 The request's HTTP verb is not valid for this endpoint (for example, POST when only GET is supported).
429 The account has made too many requests of this type recently. Try again later.

Authentication

The API uses HTTP Basic Authentication to authenticate requests. Use your Duo application's integration key as the HTTP Username.

Generate the HTTP Password as an HMAC signature of the request. This will be different for each request and must be re-generated each time.

To construct the signature, first build an ASCII string from your request, using the following components:

Component Description Example
date The current time, formatted as RFC 2822. This must be the same string as the "Date" header. Tue, 21 Aug 2012 17:29:18 -0000
method The HTTP method (uppercase) POST
host Your API hostname (lowercase) api-xxxxxxxx.duosecurity.com
path

The specific API method's path

params

The URL-encoded list of key=value pairs, lexicographically sorted by key. These come from the request parameters (the URL query string for GET and DELETE requests or the request body for POST requests).

If the request does not have any parameters one must still include a blank line in the string that is signed.

Do not encode unreserved characters. Use upper-case hexadecimal digits A through F in escape sequences.

An example params list:

realname=First%20Last&username=root

Then concatenate these components with (line feed) newlines. For example:

Tue, 21 Aug 2012 17:29:18 -0000
POST
api-xxxxxxxx.duosecurity.com
/accounts/v1/account/create
name=Acme%20Corp

GET requests also use this five-line format:

Tue, 21 Aug 2012 17:29:18 -0000
GET
api-xxxxxxxx.duosecurity.com
/admin/v1/billing/edition
account_id=DAQTM8P00LAII8ESDADE

Lastly, compute the HMAC-SHA1 of this canonical representation, using the secret key of the Duo Accounts API application you created in the parent account as the HMAC key. Send this signature as hexadecimal ASCII (i.e. not raw binary data). Use HTTP Basic Authentication for the request, using your Accounts API integration key as the username and the HMAC-SHA1 signature as the password.

For example, here are the headers for the above POST request to api-xxxxxxxx.duosecurity.com/accounts/v1/account/create, using DIWJ8X6AEYOR5OMC6TQ1 as the integration key and Zh5eGmUq9zpfQnyUIu5OL9iWoMMv5ZNmk3zLJ4Ep as the secret key:

Date: Tue, 21 Aug 2012 17:29:18 -0000
Authorization: Basic RElXSjhYNkFFWU9SNU9NQzZUUTE6ODEyZjdhMzg5NjBlZDZlYzdhNDhjY2EyZjZiYjAwMmUyMDFjMjliOQ==
Host: api-xxxxxxxx.duosecurity.com
Content-Length: 35
Content-Type: application/x-www-form-urlencoded

Separate HTTP request header lines with CRLF newlines.

The following Python function can be used to construct the "Authorization" and "Date" headers:

import base64, email.utils, hmac, hashlib, urllib
  ​ ​
def sign(method, host, path, params, skey, ikey):
    """
    Return HTTP Basic Authentication ("Authorization" and "Date") headers.
    method, host, path: strings from request
    params: dict of request parameters
    skey: secret key
    ikey: integration key
    """
  ​
    # create canonical string
    now = email.utils.formatdate()
    canon = [now, method.upper(), host.lower(), path]
    args = []
    for key in sorted(params.keys()):
        val = params[key].encode("utf-8")
        args.append(
            '%s=%s' % (urllib.parse.
                       quote(key, '~'), urllib.parse.quote(val, '~')))
    canon.append('&'.join(args))
    canon = '\n'.join(canon)
  ​
    # sign canonical string
    sig = hmac.new(bytes(skey, encoding='utf-8'),
                   bytes(canon, encoding='utf-8'),
                   hashlib.sha1)
    auth = '%s:%s' % (ikey, sig.hexdigest())
  ​
    # return headers
    return {'Date': now, 'Authorization': 'Basic %s' % base64.b64encode(bytes(auth, encoding="utf-8")).decode()}
import base64, email, hmac, hashlib, urllib

def sign(method, host, path, params, skey, ikey):
    """
    Return HTTP Basic Authentication ("Authorization" and "Date") headers.
    method, host, path: strings from request
    params: dict of request parameters
    skey: secret key
    ikey: integration key
    """

    # create canonical string
    now = email.Utils.formatdate()
    canon = [now, method.upper(), host.lower(), path]
    args = []
    for key in sorted(params.keys()):
        val = params[key]
        if isinstance(val, unicode):
            val = val.encode("utf-8")
        args.append(
            '%s=%s' % (urllib.quote(key, '~'), urllib.quote(val, '~')))
    canon.append('&'.join(args))
    canon = '\n'.join(canon)

    # sign canonical string
    sig = hmac.new(skey, canon, hashlib.sha1)
    auth = '%s:%s' % (ikey, sig.hexdigest())

    # return headers
    return {'Date': now, 'Authorization': 'Basic %s' % base64.b64encode(auth)}

Troubleshooting

Need some help? Take a look at our Accounts API Knowledge Base articles or Community discussions. For further assistance, contact Support.

If you receive an error message indicating "Cross-deployment Admin API usage through Accounts API is currently not available" when using Accounts API with Admin API, make sure you specified the subaccount's API hostname correctly when constructing the HMAC signature for the request.

If you receive 401 error responses to your API requests, check the following:

  • Is the Authorization header correctly formatted? If not, you may receive a 40101 error.
  • Does your framework override the Date header? The HTTP Date: header must be exactly the same string as was signed. This could result in a 40103 error.
  • Are the Date and time zone used RFC 3339 compliant?? If not, you may get a 40104 or 40105 response.
  • Are the parameters lexicographically sorted?
  • Did you include a line for parameters when constructing the signature, even if you're not passing in any parameters?
  • Are any hex digits lower-case?
  • Are the Content-Length and Content-Type parameters correct? If not, your parameters may be ignored or you may receive a 40103 response because your signature considered parameters that the service didn't receive.