Goodstack Services
Introduction
Goodstack API allows you to build good into your product by facilitating charitable donations. We achieve this through the API's listed and specified in this documentation.
For all donations using the Goodstack API, Goodstack handles the vetting and validation of the good causes and onward the disbursement of the donation.
The Goodstack API is organised around REST. All API responses, including errors, return JSON and use standard HTTP response codes. In all API requests you must specify the content-type HTTP header as application/json. All API requests must be made over HTTPS.
Authentication
Authenticate your API requests by setting your live secret API key in the Authorization header.
Two keys are provided to you, a publishable key prefixed with pk_
and a secret key prefixed with sk_
.
Publishable API keys are meant solely to identify your goodstack account. These can safely be published in places like your JavaScript code, or in an Android or iPhone app and are meant for usage with our SDKs.
Secret API keys should be kept confidential and only used in secure server environments.
Anywhere that accepts a Publishable key will also accept a Secret key, but routes marked to accept a Secret key will not accept a Publishable key.
Formats
Dates​
Dates are encoded as strings following the ISO 8601 standard 2021-08-01T12:00:00.000Z
.
Pagination​
A maximum of 100 objects can be returned per request and the limit can be specified by using the pageSize query parameter. By default, the pageSize is set to 25.
Metadata​
Some objects have a metadata parameter that you can use to store key-value data.
You can specify up to 20 keys, with key names up to 40 characters long and values up to 250 characters long.
Errors​
The API returns standard HTTP response codes to indicate success or failure of API requests. Errors may include a custom error message.
Error objects have these attributes, an id
, a message
corresponding to the HTTP response phrase of the error,
an optional invalid-params
attribute detailing issues with the request.
An example error object returned from the API:
{
error: {
code: 'bad_request',
title: 'Bad request',
message: 'Something is wrong with your request, please check any parameters and try again',
reasons: ['amount is a required field']
}
}
A summary of the HTTP status codes returned from the api is listed in the table below.
Code | Title | Description |
---|---|---|
200 | Success | Request successful |
204 | No Content | Request successful with no content returned |
400 | Bad Request | Request was invalid |
401 | Unauthorised | Authorization header is invalid |
403 | Forbidden | Insufficient permissions |
404 | Not Found | Resource does not exist |
429 | Too Many Requests | Rate Limited |
500s | Internal Server Error | Something went wrong with the Goodstack API |
Retry recommendations
Although rare, network requests may fail due to their inherent unreliability. To ensure a robust integration with Goodstack we recommend that you implement a retry mechanism, so that if a network request fails you are able to retry the request. Goodstacks API supports idempotency for safely retrying requests through usage of idempotency keys.
For some POST endpoints we allow clients to specify an idempotency key in the Idempotency-Key header. This allows the client to retry requests without accidentally performing the same operation twice.
If the same idempotency key is used for multiple requests, only the first successful request will be actioned and subsequent calls will return the result of the first request. Failed requests may be retried with the same idempotency key.
Requests with the same idempotency key and different payload will fail.
The idempotency key expires after 21 days. After this time, a new request with the same idempotency key will be treated as a new request.
Webhooks
Use webhooks to subscribe to updates on particular events that occur in Goodstack. Each time an event that you have subscribed to occurs, Goodstack submits a POST request to the designated webhook URL with information about the event.
To receive webhook notifications, use the webhook subscriptions API.
Attributes​
At the top level Webhook payloads will contain object
and data
fields. The data record contains the following fields:
Parameter | Type | Description |
---|---|---|
id | string | Id of the event |
createdAt | string | Timestamp of when the event was created |
eventType | string | The type of the event e.g. validation_request.approved |
eventData | record | Data associated with the event |
Webhook notifications​
The full list of IP addresses that webhook notifications may come from.
Production environment:
54.228.234.204
54.76.67.168
34.248.188.89
34.243.152.218
Sandbox environment:
79.125.45.124
54.76.168.240
99.81.243.145
54.220.118.167
Webhook responses​
To acknowledge receipt of a webhook notification, your endpoint must return a 2xx HTTP status code.
If the webhook is not received successfully then Goodstack will resend the webhook 4 times over the next 14 hours with increasing delays between retries.
The id of the event can be used to uniquely identify the event. We recommend making the processing of these events idempotent. While rare it is possible for multiple webhooks to be sent for the same event.
Verifying webhooks​
To verify that the webhook payload is sent from Goodstack, a header Goodstack-Signature
is included in the request, this signature is generated using HMAC with SHA-256 hashing algorithm and Hex encoded. The webhook subscription secret is used as the key.
You can compute this signature using HMAC with the SHA-256 hash function, using the webhook subscription secret as the key, and the request body as the message. You can then check that the Goodstack-Signature
value matches the computed value, verifying that the webhook was sent from Goodstack.
Validation Request events​
Event | Description |
---|---|
validation_request.approved | Validation request was approved |
validation_request.rejected | Validation request was rejected |
{
"object": "event",
"data": {
"id": "event_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"eventType": "validation_request.approved",
"createdAt": "2021-08-01T12:00:00.000Z",
"eventData": {
"id": "validationrequest_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"organisationId": "organisation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"organisationTypes": ["nonprofit", "social_impact"],
"name": "example charity",
"registryName": "National Charities Registry",
"registryId": "I34324",
"email": "example@example.com",
"addressLine1": "example street 15",
"addressLine2": null,
"city": "Katowice",
"postal": "44-100",
"state": "Silesia",
"website": "https://example.com",
"countryCode": "POL",
"createdAt": "2021-08-01T12:00:00.000Z",
"deletedAt": null,
"acceptedAt": "2021-08-01T12:00:00.000Z",
"rejectedAt": null,
"(deprecated) rejectionReason": null,
"rejectionReasonCode": null
}
}
}
Agent Verification events​
Event | Description |
---|---|
agent_verification.pending_user_verification | Agent has passed Goodstack review, and is awaiting email verification. |
agent_verification.pending_review | Agent has passed email verification and is awaiting Goodstack review. |
agent_verification.approved | Agent verification was approved. |
agent_verification.rejected | Agent verification was rejected. |
{
"object": "event",
"data": {
"id": "event_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"eventType": "agent_verification.approved",
"createdAt": "2021-08-01T12:00:00.000Z",
"eventData": {
"id": "agentverification_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"firstName": "Max",
"lastName": "Plank",
"organisationId": "organisation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"email": "example@example.com",
"language": "en-GB",
"validationRequestId": "validationRequest_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"title": "Board Member",
"(deprecated) rejectionReason": "Identity check failed",
"rejectionReasonCode": "user_failed_percent_review",
"metadata": {},
"createdAt": "2021-08-01T12:00:00.000Z",
"status": "approved"
}
}
}
Monitoring Subscription events​
Event | Description |
---|---|
monitoring_subscription.updated | Monitoring Subscription was updated. |
{
"object": "event",
"data": {
"id": "event_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"eventType": "monitoring_subscription.updated",
"createdAt": "2021-08-07T11:00:00.000Z",
"eventData": {
"id": "monitoringsubscription_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"organisationId": "organisation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"validationRequestId": null,
"status": "live",
"createdAt": "2021-08-02T11:00:00.000Z",
"results": {
"complianceStatus": "fail",
"warning": { "status": "clear" },
"sanction": { "status": "flag" },
"hateSpeech": { "status": "clear" },
"commercial": { "status": "clear" },
"adverseMedia": { "status": "clear" },
"controversial": { "status": "flag" },
"registration": { "active": "yes" }
}
}
}
}
Eligibility Subscription events​
Event | Description |
---|---|
eligibility_subscription.updated | Eligibility Subscription was updated. |
{
"object": "event",
"data": {
"id": "event_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"eventType": "eligibility_subscription.updated",
"createdAt": "2021-08-07T11:00:00.000Z",
"eventData": {
"id": "eligibilitysubscription_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"organisationId": "organisation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"validationRequestId": null,
"status": "live",
"suggestedActivitySubTags": [
{
"id": "activitysubtag_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"name": "Primary School"
"description": "Includes all primary schools",
"tag": {
"id": "activitytag_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"name": "Educational"
"description": "Includes all educational institutions"
}
}
]
"createdAt": "2021-08-02T11:00:00.000Z",
"updatedAt": "2021-08-02T11:00:00.000Z",
"results": {
"eligibilityStatus": "fail",
"confirmedActivitySubTags": [
{
"id": "activitysubtag_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"name": "Primary School"
"description": "Includes all primary schools",
"tag": {
"id": "activitytag_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"name": "Educational"
"description": "Includes all educational institutions"
}
}
],
"rejectedActivitySubTags": [
{
"id": "activitysubtag_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"name": "Primary School"
"description": "Includes all primary schools",
"tag": {
"id": "activitytag_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"name": "Educational"
"description": "Includes all educational institutions"
}
}
]
}
}
}
}
Validation Submission events​
Event | Description |
---|---|
validation_submission.created | Validation Submission was created. |
validation_submission.succeeded | Validation Submission has passed all checks |
validation_submission.failed | Validation Submission has failed |
validation_submission.updated | Validation Submission status has updated |
{
"object": "event",
"data": {
"id": "event_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"eventType": "validation_submission.created",
"createdAt": "2021-08-07T11:00:00.000Z",
"eventData": {
"id": "validationsubmission_xxxxx",
"agentVerificationId": "agentverification_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"agentVerification": {
"id": "agentverification_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"firstName": "Max",
"lastName": "Plank",
"email": "example@example.com",
"(deprecated) rejectionReason": null,
"rejectionReasonCode": null,
"status": "pending"
},
"createdAt": "2021-08-02T11:00:00.000Z",
"eligibilitySubscriptionId": "eligibilitysubscription_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"monitoringSubscriptionId": "monitoringsubscription_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"organisationId": "organisation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"partnerFields": {},
"validationSubmissionHostedConfigurationId": "hostedconfiguration_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"metadata": {},
"status": "pending",
"validationInviteId": "validationinvite_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"validationRequestId": "validationrequest_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"validationRequest": {
"id": "validationrequest_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"name": "Charity",
"acceptedAt": null,
"rejectedAt": null,
"organisationTypes": []
}
}
}
}
N.B. Please be aware that the "Validation Submission Failure" event consists of a property named failureReasons. In the given example, we have included all possible reasons that may be encountered. It is important to note that only one of these reasons will be received when encountering this event.
{
"object": "event",
"data": {
"id": "event_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"eventType": "validation_submission.failed",
"createdAt": "2021-08-07T11:00:00.000Z",
"eventData": {
"id": "validationsubmission_xxxxx",
"agentVerificationId": "agentverification_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"agentVerification": {
"id": "agentverification_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"firstName": "Max",
"lastName": "Plank",
"email": "example@example.com",
"(deprecated) rejectionReason": "Other",
"rejectionReasonCode": "other",
"status": "rejected"
},
"createdAt": "2021-08-02T11:00:00.000Z",
"eligibilitySubscriptionId": "eligibilitysubscription_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"monitoringSubscriptionId": "monitoringsubscription_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"organisationId": "organisation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"partnerFields": {},
"validationSubmissionHostedConfigurationId": "hostedconfiguration_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"metadata": {},
"status": "failed",
"validationInviteId": "validationinvite_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"validationRequestId": "validationrequest_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"validationRequest": {
"id": "validationrequest_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"name": "Charity",
"acceptedAt": "2021-08-01T12:00:00.000Z",
"rejectedAt": null,
"organisationTypes": ["nonprofit"]
},
"failureReasons": [
{
"check": "validation_request",
"reason": {
"rejectionReasonCode": "other"
}
},
{
"check": "agent_verification",
"reason": {
"rejectionReasonCode": "fake_email_used"
}
},
{
"check": "compliance",
"reason": {
"results": {
"complianceStatus": "fail",
"warning": {
"status": "flag"
},
"sanction": {
"status": "flag"
},
"hateSpeech": {
"status": "flag"
},
"commercial": {
"status": "flag"
},
"adverseMedia": {
"status": "flag"
},
"controversial": {
"status": "flag"
},
"registration": {
"active": "yes"
}
}
}
},
{
"check": "eligibility",
"reason": {
"status": "cannot_define_eligibility",
"results": {
"eligibilityStatus": null,
"confirmedActivitySubTags": null,
"rejectedActivitySubTags": null
}
}
},
{
"check": "eligibility",
"reason": {
"status": "live",
"results": {
"eligibilityStatus": "fail",
"confirmedActivitySubTags": [{
"id": "tag1",
"name": "Tag 1",
"tag": {
"id": "tag1",
"name": "Tag 1",
"description": "Tag description"
},
"description": "Sub tag description"
}],
"rejectedActivitySubTags": [{
"id": "tag1",
"name": "Tag 1",
"tag": {
"id": "tag1",
"name": "Tag 1",
"description": "Tag description"
},
"description": "Sub tag description"
}]
}
}
}
]
}
}
}
Donation events​
PII fields (firstName
, lastName
& email
) will be included or excluded depending on your partner settings.
Event | Description |
---|---|
donation.hosted.payment_received | Payment has been received from the user for a hosted donation. |
{
"object": "event",
"data": {
"id": "event_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"eventType": "donation.hosted.payment_received",
"createdAt": "2021-08-07T11:00:30.000Z",
"eventData": {
"id": "donation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"userId": "user_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"organisationId": "organisation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"type": "hosted",
"hosted": {
"sessionId": "donationsession_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
},
"amount": 10000,
"currencyCode": "AUD",
"status": "RECEIVED_PAYMENT",
"firstName": "Julia",
"lastName": "Russell",
"email": "example@example.com",
"consentedToBeContactedByOrg": "yes",
"anonymous": "no",
"giftAidId": null,
"cancelledAt": null,
"createdAt": "2021-08-07T11:00:00.000Z",
"metadata": {
"username": "jr123"
},
"subscription": {
"initial": true
}
}
}
}
Event | Description |
---|---|
donation.created | A new donation record has been successfully created in the system. |
donation.payment_successful | The payment for the donation has been successfully completed and verified. |
donation.settled | The donated funds have been received and reconciled by the partner foundation. |
donation.disbursed | The donated funds have been transferred to the intended nonprofit. |
donation.cancelled | The donation process has been terminated, either by the donor or due to system issues. |
donation.reassigned | The donation has been reassigned to another organisation, either because it has failed compliance, we are unable to pay the organisation or the organisation has been merged with another id. |
{
"object": "event",
"data": {
"id": "event_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"eventType": "donation.payment_successful",
"createdAt": "2021-08-07T11:00:30.000Z",
"eventData": {
"id": "donation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"userId": "user_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"organisationId": "organisation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"type": "direct",
"amount": { "amount": 10, "currency": "GBP" },
"status": "ACTIVE",
"firstName": "Julia",
"lastName": "Russell",
"email": "example@example.com",
"consentedToBeContacted": "yes",
"anonymous": "no",
"giftAidId": "giftaid_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"createdAt": "2021-08-07T11:00:00.000Z",
"cancelledAt": null,
"donationRequestId": "donationrequest_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"donationRequestAmount": { "amount": 10, "currency": "GBP" },
"accountId": "account_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"metadata": {"data": "hash-value"},
"settledAmount": { amount: 10, currency: "GBP" },
"settledAt": "2021-08-07T11:00:00.000Z",
"disbursedAt": "2021-08-07T11:00:00.000Z",
"reassignedAt": "2021-08-07T11:00:00.000Z",
"reassignedFromOrganisationId": "organisation_xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
}
}
}
Authentication​
- API Key: PublishableApiKey
- API Key: SecretApiKey
- API Key: DonationSessionToken
API key starting with 'pk_' that can be made public
Security Scheme Type: | apiKey |
---|---|
Header parameter name: | Authorization |
API key starting with 'sk_' that must be kept secure
Security Scheme Type: | apiKey |
---|---|
Header parameter name: | Authorization |
Donation session token
Security Scheme Type: | apiKey |
---|---|
Header parameter name: | donation-session-token |
Contact
Api support: engineering-support@goodstack.io