Skip to main content

Wallet

Overview

This service facilitates the integration of your wallet with our betting system, and the accompanying documents provide specifications for authentication, betting & wallet transaction handling, and communication with our API. The diagram below outlines the general flow of the implementation.

Table of Content

  • Authentication (Frontend - BetMakers)
  • Wallet Integration (Betmakers - Wallet)
  • On demand services (Wallet - BetMakers)

Authentication (Frontend - BetMakers)

We support two types of authentication methods. We recommend using JWT authentication; however, you are free to choose either one of them.

  • JWT Authentication.
  • Custom Authentication.

JWT Authentication.

JWT is employed for authentication between systems, and JWKS is utilised to validate the tokens in our integration processes. The diagram below outlines the fundamental flow for implementing authentication in the integration process.

JWT Specification

Following claims needs to be present on JWT body.

"client_id": "provided-client-id",
"aud": "user-uuid",
"token_use": "access",
"username": "user-uuid",

JWKS Specification.

The JWKS key sets must be configured using the RS256 standard for our integration.

Key sets should be made available through a publicly accessible endpoint. An example response body is provided below.

{
"keys": [
{
"kty": "RSA",
"use": "sig",
"alg": "RS256",
"kid": "64527lkxjcvnkawk11213a",
"n": "0oDGt_owqzVNAW4D0-LnJ7LwLK",
"e": "AQAB"
}
]
}

Custom Authentication.

With the Custom Authentication method, clients are permitted to generate any kind of encrypted token. Token verification is expected to be performed at the reserveFunds call (refer to the wallet integration section). User claims should be passed via the user metadata object (Refer host.state.user event documentation).

Wallet Integration (Betmakers - Wallet)

Implementing Auth.

We utilise REST endpoints for posting data to external client wallets. Betmakers provide two authentication types for machine-to-machine communication. You must implement one of these authentication methods according to the given specification.

  • Client Credentials Auth.
  • API Key Auth.

Implementing Client Credentials Authentication.

In order to implement Client Credentials based auth you need to implement authentication endpoint (/auth/token). The token endpoint will be called with provided client_id and client_secret to obtain the token, which will then serve as a bearer token for authorising calls to the external wallet, as illustrated in the accompanying diagram depicting the basic flow.

POST {ExternalWalletEndpoint}/auth/token

Request body:
{
"grant_type": "client_credentials",
"client_id": "TestClient",
"client_secret": "TestSecret"
}

Response:
{
"access_token": "03807cb390319329bdf6c777d4dfae9c0d3b3c35",
"expires_in": 3600,
"token_type": "bearer",
"scope": null
}

Implementing API Key Authentication.

Two parties will utilize a shared API Key for authentication purposes between betmakers and wallets. This method necessitates the implementation of an authentication endpoint.

Wallet Transactions.

Our system supports four transactions, and clients are required to implement them based on the provided specifications. The minimum payloads to be sent to the wallet are outlined below, with the possibility of additional data being included.

info

All HTTP requests are configured with a 10-second timeout.

  • Reserve Transaction.
  • Confirm Transaction.
  • Refund Bet Transaction.
  • Settle Bet Transaction.
  • Debit Transaction.
  • Credit Transaction.

Reserve Transaction.

This transaction will be used to reserve funds from the user's account and will be sent at the time of bet placement. We forward the token and metadata from the bet placement. Please utilize these details to conduct server-side validation.

Example Request:
POST {ExternalWalletEndpoint}/reserve

Request body:
{
"transactionId": "transaction-uuid",
"betId": "bet-uuid",
"userId": "user-uuid",
"bonusFlag": false, // True for bonus bets.
"amount": 1000,
"token": "token", // Token used on placement.
"metadata": {
"userId": "user-id",
"brandName": "betmakers",
"brandUserId": "brand-user-id",
"data": ""
}
}

Success Response:
{
"success": true
}

Error Response:
{
"message": "Readable message.",
"code": "ErrCode",
"statusCode": 401
}

Confirm Transaction.

This request serves to confirm the reserved funds are accepted along with the bet. Accepted amount will be reflected on the amount field which needs to be debited from either the bonus account or cash account based on the bonus flag.

Example Request:
POST {ExternalWalletEndpoint}/confirm

Request body:
{
"transactionId": "transaction-uuid",
"userId": "user-uuid",
"betId": "bet-uuid",
"bonusFlag": false, // True for bonus bets.
"amount": 2200,
"betMetadata": {
"id": "bet-uuid",
"ticketId": "ticket-uuid",
"tenantId": "tenant-uuid",
"userId": "user-uuid",
"betTypeId": 2,
"statusId": 7,
"createdAt": "2024-10-31T08:36:09Z",
"amount": 2200,
"fixedOdds": 1.38,
"percentage": 2200,
"legs": [
{
"betTypeId": 2,
"eventId": "b7137ff3-92a8-58e3-8d0c-7006feb154ab",
"productKeyword": "TBF",
"processedDividend": 0,
"legType": "Racing",
"marketId": null,
"selection": [
{
"statusId": 1,
"fixedOdds": 1.38,
"margin": 0,
"position": 0,
"selectionId": [
"257ec2f2-b7be-5058-9b8d-94ca94c3744c"
],
"lineTypeId": 0,
"line": 0
}
]
}
]
}
}

Success Response:
{
"success": true
}

Error Response:
{
"message": "Readable message.",
"code": "ErrCode",
"statusCode": 401
}

RefundBet Transactions.

info

The refund transaction call will only be triggered upon the successful reservation of funds.

Example Request:
POST {ExternalWalletEndpoint}/refundBet

Request body:
{
"transactionId": "transaction-uuid",
"betId": "bet-uuid",
"userId": "user-uuid",
"bonusFlag": false, // True for bonus bets.
"amount": 200
}

Success Response:
{
"success": true
}

Error Response:
{
"message": "Readable message.",
"code": "ErrCode",
"statusCode": 401
}

Settle Transactions.

The settle request is employed to convey that the bet has been settled. In the case of a winning bet, the payout will be passed through the amount field, and it is necessary to credit that amount to the client's cash balance.

Retry Mechanism on settleBet

In the event of a failure while handling the settleBet request, the system will automatically retry up to five times with the same details. The retry intervals will be as follows: after 5 minutes, 10 minutes, 20 minutes, and then hourly. The transactionId can be used to check if the transaction has already been processed on your end.

ExampleBet Request:
POST {ExternalWalletEndpoint}/settleBet

Request body:
{
"transactionId": "transaction-uuid",
"betId": "bet-uuid",
"userId": "user-uuid",
"bonusFlag": false, // True for bonus bets.
"amount": 2590 // Winnings if available.
}

Success Response:
{
"success": true
}

Error Response:
{
"message": "Readable message.",
"code": "ErrCode",
"statusCode": 401
}

Debit Transaction.

Additional transactions, such as reverted winnings, will be posted to this endpoint. It is essential to debit the specified amount from the client's cash or bonus balance based on the bonus flag.

Expected Transaction Types:
CodeDescription
deductbetwinReversing winnings of the bet.
betwincancelledReversing winnings of the bet.
Example Request:
POST {ExternalWalletEndpoint}/debit

Request body:
{
"transactionId": "transaction-uuid",
"betId": "bet-uuid",
"userId": "user-uuid",
"bonusFlag": false, // True for bonus bets.
"transactionType": "betwincancelled",
"amount": 100,
}

Success Response:
{
"success": true
}

Error Response:
{
"message": "Readable message.",
"code": "ErrCode",
"statusCode": 401
}

Credit Transactions.

Additional transactions, such as manual adjustments, will be posted to this endpoint. It is essential to credit the specified amount to the client's cash or bonus balance based on the bonus flag.

Expected Transaction Types:
CodeDescription
adjustreversetransactionManual adjusted transactions.
cashoutCash out transactions.
betpartialrefundPartial refunds for the bet.
Example Request:
POST {ExternalWalletEndpoint}/credit

Request body:
{
"transactionId": "transaction-uuid",
"betId": "bet-uuid",
"userId": "user-uuid",
"bonusFlag": false, // True for bonus bets.
"transactionType": "betwincancelled",
"amount": 100, // Refunded amount.
}

Success Response:
{
"success": true
}

Error Response:
{
"message": "Readable message.",
"code": "ErrCode",
"statusCode": 401
}

On-Demand Services (BackOffice APIs)

On-demand query services operate as an open endpoint, requiring only the inclusion of the x-api-key in the request headers. For user-related queries or subscriptions, the access token needs to be passed in the header. Components designed for the tenant frontend can access both bet subscriptions and query data.

Getting Access Token

Access token can be aquired by calling provide oAuth endpoint

Example Request:
POST {BackOffice_TokenEndpoint}

Request body:
{
"grant_type": "client_credentials",
"client_id": "{BackOffice_ClientID}",
"client_secret": "{BackOffice_ClientSecret}"
}

Response:
{
"access_token": "03807cb390319329bdf6c777d4dfae9c0d3b3c35",
"expires_in": 3600,
"token_type": "bearer",
"scope": null
}
Example Curl:
curl --location '<BackOffice_TokenEndpoint>' \
--header 'Content-Type: application/x-www-form-urlencoded' \
--header 'Cookie: XSRF-TOKEN=efa6365e-c468-4fc3-aba0-0b364be0c8b9' \
--data-urlencode 'grant_type=client_credentials' \
--data-urlencode 'client_id=<BackOffice_ClientID>' \
--data-urlencode 'client_secret=<BackOffice_ClientSecret>'

Available APIs

Query: GetBets

This query retrieves all bets for a for given ids for tenant(s).

Example Schema:
Headers:
{
"X-API-Key": "<BackOffice_ClientID>",
"Authorization": "Bearer <ACCESS_TOKEN>"
}

Query:
query GetBets {
getBets(
input: {
tenant: {
ids: ["<tenant-uuid>"]
bet_ids: ["<bet-uuid>"]
}
}
) {
bets {
id
created_at
legs {
bet_type
dividend
event_id
...
}
metadata {
created_at
tenant_id
user_id
...
}
}
pagination {
last_id
count
}
}
}
Example Curl:
curl '<BackOffice_Endpoint>/query' \
-H 'accept: */*' \
-H 'content-type: application/json' \
-H 'authorization: bearer <ACCESS_TOKEN>' \
-H 'x-api-key: <BackOffice_ClientID>' \
--data-raw '{"query":"query GetBets {\n getBets(\n input: {\n tenant: {\n ids: [\"<tenant-uuid>\"]\n bet_ids: [\"1f2eeaff-0f79-40fa-92d6-b1ffbd9ce8ba\"]\n }\n }\n ) {\n bets {\n id\n created_at\n legs {\n bet_type\n dividend\n event_id\n event_name\n event_time\n leg_type\n meeting_id\n meeting_name\n postions_paid\n product_keyword\n sport_id\n sport_name\n race_number\n }\n metadata {\n created_at\n tenant_id\n user_id\n account_id\n account_type\n ticket_id\n fixed_odds\n original_odds\n amount\n original_amount\n percentage\n original_percentage\n is_partial\n status\n source\n medium\n bet_limit_id\n reason\n reason_id\n message\n action_id\n liability_percentage\n liability\n is_fixed\n is_flexi\n is_boxed\n bet_type\n teller_id\n is_cashout\n is_livebet\n ip_address\n }\n }\n pagination {\n last_id\n count\n }\n }\n}\n","operationName":"GetBets"}'

Mutation: cancelBets

The following mutation lets the client cancel the bets.

Example Schema:
Headers:
{
"X-API-Key": "<BackOffice_ClientID>",
"Authorization": "<ACCESS_TOKEN>"
}

Query:
mutation CancelBet {
cancelBet(
bet: {
id: "<bet-uuid>"
tenant_id: "<tenant-uuid>"
}
)
}
Example Curl:
curl '<BackOffice_Endpoint>/query' \
-H 'accept: */*' \
-H 'content-type: application/json' \
-H 'authorization: bearer <ACCESS_TOKEN>' \
-H 'x-api-key: <BackOffice_ClientID>' \
--data-raw '{"query":"mutation CancelBet {\n cancelBet(\n bet: {\n id: \"1f2eeaff-0f79-40fa-92d6-b1ffbd9ce8ba\"\n tenant_id: \"<tenant-uuid>\"\n }\n )\n}\n","operationName":"CancelBet"}'