Introduction

This API is provided to give third party access to the Access Database: the following documentation is provided to help developers working with the API. The documentation will cover general usage on this page and customer specific calls on other pages. You can use the menu to navigate easily.

Getting Started

When you started working with this API you will have been given two URLs by the customer. These link directly to the corresponding Access database and will have the calls available to you as detailed in the resources section.

You can use the Training API for testing your calls; the data in training can be easily overwritten with a copy of live data by the customer.

Once you have a copy of the API key for the corresponding API, you are ready to make your first call. All API variables can be passed either in the URL or in the HTTP headers. If you use the header method you can prefix with "X-" to avoid confusion. The following calls are the same:

URL: {GET} /API/WebsiteAPI/v1/EventPublic/?apikey=<APIKEY>

URL: {GET} /API/WebsiteAPI/V1/EventPublic/
Headers:
X-apikey: <APIKEY>

The variable name is NOT case sensitive (APIKEY = apiKey), but many variables ARE case sensitive, e.g. API Keys and Client details

To break down the above URL there are four sections (one of which was unused) and these are used to work out the resource you want. The following URL defines the event with event identification number EVT00001.

URL: {GET} /API/WebsiteAPI/V1/EventPublic/EVT00001
API Name: WebsiteAPI
Version: 1
Resource: EventPublic
Key value: EVT00001

The version can be sent as "v1" or "1" as both are recognised. In this documentation v1 will be used to make it easier to understand.

HTTP request types (GET/POST/PUT/DELETE) are used throughout the API, you can also send these as URL or HTTP header parameters.

GET: Get a resource - either all, those that match provided filters or singular
POST: Add a resource
PUT: Update a single resource
DELETE: Delete a single resource

For more APIs and corresponding resources see the resource section

General usage

The following section delves a little deeper into general usage of the API and is recommended for any developer wanting to work with this API.

Reading Data

When running a get command on most entities you can apply more precise selections by providing field selections, filtering arguments and sorting.

Pagination

To help you only get the rows you want you can use pagination. This is done via the "Page" and "Pagesize" variables. All results will also give the maximum number of pages and the start point of a result set. For example, the call shown below will give the 5th and 6th event and let you know the number of pages in the full result.

URL: {GET} /API/WebsiteAPI/V1/EventPublic/
Headers:
X-apikey: <APIKEY>
X-Page: 3
X-PageSize: 2

Selecting Fields

Resources contain a selection of fields, a list of which can be found against the resource in the resource section. Certain fields are set to be displayed by default: these are designed to make it easier for a simple call, but any field listed in the resource section can be requested using the "Fields" parameter.

URL: {GET} /API/WebsiteAPI/V1/EventPublic/
Headers:
X-apikey: <APIKEY>
X-fields: EventID, Type, webDescription

This will return a list of events with their EventID, type and description.

If the result is over multiple pages the additional field "RN" is provided to assist with pagination.

Filtering

You can provide a selection of different filters to get the set of data you need. These will all be provided as a comma separated list in the "Filter" parameter, to clarify the comma acts as an 'AND' which allows multiple filters to be issued in one statement. Currently, it is not possible to use the 'OR' conjunction within the filtering statement. A full list of calls and an example, 'Conferences with more than 100 attendees allowed', can be seen below.

A eq 'B' Equal, where value A equals 'B'
A ne 'B' Not equal, where value A is anything but 'B'
A gt '10' Greater than, where value A is greater than '10'
A ge '10' Greater than or equal to, where value A is greater than or equal to '10'
A lt '10' Less than, where value A is less than '10'
A le '10' Less than or equal to, where value A is less than or equal to '10'
A lk 'B*' Like, A starts with 'B', use * for wildcard
A nl '*B' Not Like, A does not end with 'B', use * for wildcard
A not null Is not null, where value A contains a value
A null Is null, where value A does not contains a value

If the value you are filtering on contains an apostrophe, location name for example, you must prefix the apostrophe with a backslash. e.g. locationName eq 'McDonald\'s'

URL: {GET} /API/WebsiteAPI/V1/EventPublic/
Headers:
X-apikey: <APIKEY>
X-filter: type eq Conference, maxAllowed gt 200

To search for commas (,) inside a filter you should wrap the criteria in single quotes (')

URL: {GET} /API/WebsiteAPI/V1/EventPublic/
Headers:
X-apikey: <APIKEY>
X-filter: webDescription eq 'Dinner, Drinks and Dance', maxAllowed gt 50

Sorting

You can sort the data by fields in ascending (ASC) or descending (DESC) order, this is done via passing a comma separated list in the "Sort" parameter.

URL: {GET} /API/WebsiteAPI/V1/EventPublic/
Headers:
X-apikey: <APIKEY>
X-sort: eventType, eventName asc

Security

There are several security layers that may be used with certain APIs and resources. If they are defined in the resource section then the following section will explain what you will need to do.

Authentication

In many cases the end user will need to give permission to modify their records. This will mean that the end user will need to be authenticated. This can either be done via a third party authentication service or by using the username and password stored in the system and using an authentication API to get relevant tokens.

The authentication API is designed to work using OAUTH 2.0 methods. However, due to the nature of API and the data held this is limited to two main methods: third party authentication and Access authentication. The resulting tokens are designed to work in the Bearer authorization schema and are valid for as long as the expiry time, provided with the token, states in seconds.

{
    "access_token": "KnkxNxTDdoSQQ9Sk"
    "token_type": "Bearer"
    "expires_in": 7200
    "refresh_token": "eR9br~Tz9uY4M.0o" (Value optional, see the resource section for if this is provided)
}

If using Access stored username and password you will need to provide the username and password to the token endpoint with a grant type of password: we will return a token as above.

URL: {GET} /API/Authentication/v1/Token/
Headers:
X-apikey: <AuthAPIKEY>
X-grant_type: Password (case insensitive)
X-username: Sally@theaccessgroup.com (case insensitive)
X-password: demo(case sensitive)

If Client details are required, as indicated by the resources tab, these will be passed as an authorization header using basic schema.

Client ID: "123456789" (case sensitive)
Client Secret: "987654321" (case sensitive)
Pre base64: 123456789:987654321 (case sensitive)
Authorization: Basic MTIzNDU2Nzg5Ojk4NzY1NDMyMQ== (Base64 encoded)

If the token allows refresh tokens you can use this second token instead of resending the Username and password.

URL: {GET} /API/Authentication/v1/Token/
Headers:
X-apikey: <AuthAPIKEY>
X-grant_type: refresh_token (case insensitive)
x-refresh_token: eR9br~Tz9uY4M.0o (case sensitive)

Once you have a token you can use it to send the relevant request as below:

URL: {GET} /API/WebsiteAPI/v1/ContactPrivate/000001
Headers:
X-apikey: <APIKEY>
Authorization: Bearer KnkxNxTDdoSQQ9Sk (case sensitive)

Possible authentication errors and troubleshooting

This section lists possible error messages returned by the authentication API and likely causes of the error with the associated solution. It is important to note that the error returned will vary depending on whether the developer mode is currently enabled for your application. With developer mode enabled error messages are more descriptive than their non-developer counterparts. In order to set your application to developer mode you need to modify the MetaData.App table by setting the Developer column to include 'DeveloperMode=Yes' - Warning: the developer column also contains other settings relating to the functionality of the application and therefore it is important to keep those intact when updating this field.

Call
/api/Authentication/v1/Token

Error(Dev)
1001 - The API Key did not match

Error(Non-Dev)
1001 - The API you attempted to connect to could not be found

Cause
The header section of your call is either missing an API key or the API key is incorrect

Solution
Make sure that the header section of your call contains the ‘X-apikey’ header and verify that the correct API key is being used in the MetaData.WebAPI table of your database.
Call
/api/Authentication/v1/Tokens

Error(Dev&Non-Dev) -
No error code - The API resource you requested could not be found

Cause -
If no error code is returned in the call response, it suggests that the correct API has not been reached.

Solution -
Make sure that the correct URL is referenced in the call. In this example, the URL in the API call calls a non-existent API ‘Tokens’ rather than ‘Token’
Call
/api/Authentication/v1/Token

Error(Dev)
404 - The authentication currently requires a valid grant type

Error(Non-Dev)
401 - Authentication failed

Cause
The header section of your call is either missing a grant type or the grant type is incorrect.

Solution
Make sure that the headers of your call contain the ‘x-grant_type’ header and check it is set to either ‘password’ or ‘refresh_token’.
Call -
/api/Authentication/v1/Token

Error(Dev)
404 - Please check the username/password

Error(Non-Dev)
401 - This request failed authentication

Cause
Incorrect username or password.

Solution
Ensure that the details of the user you are trying to authenticate with are entered match the values in the dbo.WEBUSERLIST table. Alternatively this can be checked via the front-end by going to Admin -> Web Admin -> Web Users, and checking that the email address and password exist on that form.
Call
/api/Authentication/v1/Token

Error(Dev&Non-Dev)
500 - Could not store AuthToken Details

Cause
This error is caused by an internal server error.

Solution
The cause of this error is difficult to pinpoint. The best place to start investigating the source of an internal server error is to view the tracelog entry associated to it (SELECT TOP 1000 * FROM Diagnostics.TraceLog ORDER BY id desc)
Call
/api/Authentication/v1/Token

Error(Dev)
403 - API does not allow refresh tokens to be used

Error(Non-Dev)
401 - This request could not be authenticated

Cause
This error occurs when trying to make a call with a grant type of refresh_token to an API that does not allow token refreshes.

Solution
To enable token refresh modify the metadata.WebAPI table by changing the value of the AuthAllowRefreshToken column to -1

HMAC

Any API may use a HMAC signature for validation. We support MD5, SHA256 and SHA512 hash algorithms. If this is used for an API it will be indicated in the resource section for APIs and the secret key will be sent to you by the customer. The HMAC is constructed as below, with spaces removed, followed by an example (MD5 HMAC, secret key: "SecretString").

httpheader:<Fields in HTTP Header (alphabetical order, comma separated)>\n
urlparameters:<Fields in URL parameters (alphabetical order, comma separated)>\n
body:<The content of the body of a request (lower case)>\n
path:<URL without domain and parameters>\n
verb:<HTTP Verb (lower case)>

URL: {GET} /API/APIname/v1/Contact/?D=D&c=3
Headers:
X-b: B
X-a: 1
X-hmackey: f8d411cab3dc7d0f07859b3b09edde4d
Content:
{
    "Title": "Mr"
}

[HMAC message:]
httpheader:x-a=1,x-b=b\nurlparameters:c=3,d=d\nbody:{"title":"mr"}\npath:/apiname/v1/contact/\nverb:get

Other Features

ETAGs

The use of Entity Tags, known here on as ETags, allows users to cut down on network traffic. This is done by having an ETag which represents the data provided. If an API call is getting data it will be returned with an ETag header, i.e. a representation of the returned resource(s). This ETag can be sent as part of a future call (using the "If-None-Match" HTTP header) to the same object and if the data has not been updated it will return a HTTP Not Modified (304) status, to show that since the last time you read this it has not changed.

Due to the data changing frequently, ETags are linked to the resource(s) returned, including the fields and ordering of records. For example, if you load a contact's full name and you want to display it on each page, you can cache this on the client and only update it if the call to read it comes back as modified.

A matching ETag does not indicate that other fields have not been updated; the Modified field against most data records will indicate the date it was last modified.

Information Results

If there is a validation failure this will be returned to you in a response with a status of "Information" and will have a list of validation issues in the values field.

Errors

Errors do sometimes occur so we've tried to make them as clear as possible to help all involved. The resulting object will have a status of "Error" and the HTTP status will not be OK (200).

Errors have a development message and a production message. This is to ensure that during development of the client you are able to get the most out of the messages and during production the API is secure for you to use.

Content Types

The API provides support for JSON (preferred) and XML objects being sent to and received from the API. This is done using the standard HTTP headers "Accept" and "Content-Type" and corresponding MIME type "application/json" or "application/xml". Note that, by default, the objects will be read and returned as JSON.

The MIME type used for XML is "application/xml" not "text/xml" as, according to the standard, text/*-MIME types have an US-ascii character set unless otherwise specified in the HTTP headers. This effectively means that any encoding defined in the XML prologue (e.g. <?xml version="1.0" encoding="UTF-8"?>) is ignored. This is, of course, not the expected and desired behaviour.

Bespoke APIs

Bespoke APIs will follow a seperate naming scheme for their endpoints instead of the usual 'WebsiteAPI'.
Bespoke API have a standardized endpoint naming format (ommit the brackets around customer code), however in some cases the endpoint may be named differently. An example of the most commonly used API endpoint format and other posible formats follows:

  • URL:{GET}/(CustomerCode)WebsiteAPI/V1/EventPublic
  • URL:{GET}/(CustomerCode)_Bespoke/V1/EventPublic
  • URL:{GET}/(CustomerCode)BSKAPI/V1/EventPublic
  • URL:{GET}/(CustomerCode)/V1/EventPublic
  • URL:{GET}/(CustomerCode)API/V1/EventPublic

Sample Objects

The following section gives sample calls and sample responses, please note that your API may have different resources and options and these calls may not work with your API. Please see the resources section to see your particular API resources.

GET results

The first call returns the 3rd and 4th (page 2, page size 2) events where they have the event type is 'Chargeable', ordered ascending by the event name. This is shown in JSON and XML (note that the content-type is left out to avoid confusion).

URL: {GET} /API/WebsiteAPI/V1/EventPublic/
Headers:
X-apikey: <APIKEY>
X-Fields: eventID, eventName, eventType, type
x-Sort: eventName asc
x-filter: eventType eq Chargeable
x-page: 2
x-pagesize: 2

JSON Result

{
    "Status": "OK",
    "PageIndex": 1,
    "PageSize": 4,
    "PageCount": 2,
    "Columns": [
      {
{
        "Name": "eventID",
       "DataType": "Text",
       "MaxLength": 50
},
{
        "Name": "eventName",
       "DataType": "Text",
       "MaxLength": 255
},
{
       "Name": "eventType",
       "DataType": "Text",
       "MaxLength": 50
},
{
        "Name": "type",
        "DataType": "Text",
        "MaxLength": 50
},
{
        "Name": "RN",
        "DataType": "Long",
        "MaxLength": 0
}
],
"Rows": [
{
        "eventID": "EVT10045",
        "eventName": "London Marathon 2018",
        "eventType": "Chargeable",
        "type": "Fun Day",
        "RN": 3
},
{
        "eventID": "EVT10046",
        "eventName": "National Conference",
        "eventType": "Chargeable",
        "type": "Conference",
        "RN": 4
}
],
"RecordCount": 5,
"RecordStartIndex": 1
}

XML Result

<?xml version="1.0" encoding="utf-16"?>
<SerializableResultDTO xmlns:xsd="http://www.w3.org/2001/XMLSchema"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
    <Status>OK</Status>
    <PageIndex>1</PageIndex>
    <PageSize>4</PageSize>
    <PageCount>2</PageCount>
    <Columns>
      <ResultDTOColumn>
        <Name>eventID</Name>
        <DataType>Text</DataType>
        <MaxLength>50</MaxLength>
      </ResultDTOColumn>
      <ResultDTOColumn>
        <Name>eventName</Name>
        <DataType>Text</DataType>
        <MaxLength>255</MaxLength>
      </ResultDTOColumn>
      <ResultDTOColumn>
        <Name>eventType</Name>
        <DataType>Text</DataType>
        <MaxLength>50</MaxLength>
      </ResultDTOColumn>
      <ResultDTOColumn>
        <Name>RN</Name>
        <DataType>Long</DataType>
        <MaxLength>0</MaxLength>
      </ResultDTOColumn>
    </Columns>
    <Rows>
      <ArrayOfValues>
        <values>
          <key>eventID</key>
          <value>EVT10045</value>
        </values>
        <values>
          <key>eventName</key>
          <value>London Marathon 2018</value>
        </values>
        <values>
          <key>eventType</key>
          <value>Chargeable</value>
        </values>
        <values>
          <key>type</key>
          <value>Fun Day</value>
        </values>
        <values>
          <key>RN</key>
          <value>3</value>
        </values>
      </ArrayOfValues>
      <ArrayOfValues>
        <values>
          <key>eventID</key>
          <value>EVT10046</value>
        </values>
        <values>
          <key>eventName</key>
          <value>National Conference</value>
        </values>
        <values>
          <key>eventType</key>
          <value>Chargeable</value>
        </values>
        <values>
          <key>type</key>
          <value>Conference</value>
        </values>
        <values>
          <key>RN</key>
          <value>4</value>
        </values>
      </ArrayOfValues>
    </Rows>
    <RecordCount>5</RecordCount>
    <RecordStartIndex>1</RecordStartIndex>
</SerializableResultDTO>

The second call just gets a single record based on a ID number (EVT10046) getting IDnumber, event name and event type.

URL: {GET} /API/WebsiteAPI/V1/EventPublic/EVT10046
Headers:
X-apikey: <APIKEY>
X-Fields: eventID, eventName, eventType

JSON Result

{
    "Status": "OK",
    "PageIndex": 1,
    "PageSize": 100,
    "PageCount": 0,
    "Columns": [
      {
      {
        "Name": "eventID",
        "DataType": "Text",
        "MaxLength": 50
      },
        "Name": "eventName",
        "DataType": "Text",
        "MaxLength": 255
      },
      {
        "Name": "eventType",
        "DataType": "Text",
        "MaxLength": 50
      },
    ],
    "Rows": [
      {
        "eventID": "EVT10046",
        "eventName": "National Conference",
        "eventType": "Chargeable"
      }
    ],
    "RecordCount": 1,
    "RecordStartIndex": 0
}

XML Result

<?xml version="1.0" encoding="utf-16"?>
<SerializableResultDTO xmlns:xsd="http://www.w3.org/2001/XMLSchema"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
    <Status>OK</Status>
    <PageIndex>1</PageIndex>
    <PageSize>100</PageSize>
    <PageCount>0</PageCount>
    <Columns>
      <ResultDTOColumn>
        <Name>eventID</Name>
        <DataType>Text</DataType>
        <MaxLength>50</MaxLength>
      </ResultDTOColumn>
      <ResultDTOColumn>
        <Name>eventName</Name>
        <DataType>Text</DataType>
        <MaxLength>255</MaxLength>
      </ResultDTOColumn>
      <ResultDTOColumn>
        <Name>eventType</Name>
        <DataType>Text</DataType>
        <MaxLength>50</MaxLength>
      </ResultDTOColumn>
    </Columns>
    <Rows>
      <ArrayOfValues>
        <values>
          <key>eventID</key>
          <value>EVT10046</value>
        </values>
        <values>
          <key>eventName</key>
          <value>National Conference</value>
        </values>
        <values>
          <key>eventType</key>
          <value>Chargeable</value>
        </values>
      </ArrayOfValues>
    </Rows>
    <RecordCount>1</RecordCount>
    <RecordStartIndex>0</RecordStartIndex>
</SerializableResultDTO>

POST and PUT results

The following two are calls to POST (insert) and PUT (update) a new contact: these will show the possible content types.
Quick note on the difference between POST and PUT.
POST command is not idempotent, when the same POST call is made to a data store multiple times it will take effect each time it is invoked. e.g. using POST twice to insert the same user into the database will attempt to create the same user twice resulting in a duplicate entry
In contrast PUT is idempotent, making multiple duplicate PUT calls to a data store will leave the data source unchanged past the 1st recieved call.

URL: {POST/PUT} /API/WebsiteAPI/V1/ContactPublic
Headers:
X-apikey: <APIKEY>
Content:
[JSON]
{"title":"Mr",
"firstName":"John",
"keyName":"Smith",
"alumni":true,
"doNotContact":false,
"doNotEmail":false,
"doNotMail":false,
"doNotPhone":false,
"emailThirdParty":false,
"mailThirdParty":false,
"phoneThirdParty":false,
"serialNumber":"99999991",
"volunteer":false}

[XML]
<?xml version="1.0" encoding="UTF-8"?> <root>

<alumni>true</alumni>
<doNotContact>false</doNotContact>
<doNotEmail>false</doNotEmail>
<doNotMail>false</doNotMail>
<doNotPhone>false</doNotPhone>
<emailThirdParty>false</emailThirdParty>
<firstName>John</firstName>
<keyName>Smith</keyName>
<mailThirdParty>false</mailThirdParty>
<phoneThirdParty>false</phoneThirdParty>
<serialNumber>99999991</serialNumber>
<title>Mr</title>
<volunteer>false</volunteer>
</root>

JSON Result

     {
"Status": "Success",
       "Values":{
         "title": "Mr",
         "firstName": "John",
         "keyname": "Smith",
         "alumni": true,
         "doNotContact": false,
         "doNotEmail": false,
         "doNotMail": false,
         "doNotPhone": false,
         "emailThirdParty": false,
         "mailThirdParty": false,
         "phoneThirdParty": false,
         "serialNumber": "99999991",
         "volunteer": false,
         "doNotSMS": "0",
         "contactInVerification": "-1",
         "addressSerialNumber": "0",
         "addressType": "Residential"
       }
     }

Token results

URL: {POST} /API/Authentication/v1/token/
Headers:
X-apikey: <AuthAPIKEY>
x-grant_type: password
x-username: sally@crm.com
x-password: demo

JSON Result

{
  "access_token": "q8If/RVuO8ssXnWn7FQA9GNUP2zkAWuMwxgPcz8P7qo=",
  "token_type": "Bearer",
  "expires_in": 7200,
  "refresh_token": "P0X2Uxj0OVcSwmIUtjO99BoKYaifRg0AHLMmLtkQ8aY="
}

XML Result

<?xml version="1.0" encoding="utf-16"?>
<OAUTHToken
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <access_token>q8If/RVuO8ssXnWn7FQA9GNUP2zkAWuMwxgPcz8P7qo=</access_token>
  <token_type>Bearer</token_type>
  <expires_in>7200</expires_in>
  <refresh_token>P0X2Uxj0OVcSwmIUtjO99BoKYaifRg0AHLMmLtkQ8aY=</refresh_token>
</OAUTHToken>

Information object

URL: {POST} /API/WebsiteAPI/V1/ContactPublic
Headers:
X-apikey: <APIKEY>
Content:
[JSON]
{"alumni": false}
[XML]
<?xml version="1.0" encoding="UTF-8"?>
<root>
  <item>
    <element>
      <key>alumni</key>
      <value>false</value>
    </element>
  </item> </root>

JSON Result

{
  "Status": "Information",
  "Values": [
    "keyname - You must enter the keyname"
  ]
}

XML Result

<?xml version="1.0" encoding="utf-16"?>
<InfoDTO
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <Status>Information</Status>
  <Values>
    <string>keyname - You must enter the keyname.</string>
  </Values>
</InfoDTO>

Error object

URL: {GET} /API/WebsiteAPI/V1/EventPublic/
Headers:
X-apikey: <BADAPIKEY>

JSON Result

{
  "Status": "Error",
  "ErrorCode": "1001",
  "Message": "The API you attempted to connect to could not be found"
}

XML Result

<?xml version="1.0" encoding="utf-16"?>
<ErrorDTO
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <Status>Error</Status>
  <ErrorCode>1001</ErrorCode>
  <Message>The API you attempted to connect to could not be found</Message>
</ErrorDTO>