Skip to main content
Version: v3

Gift Packages Guide

tip

Before integrating Gift Packages into your game, please first complete setting up your game. You can obtain the Client ID and Server Secret of your game on Game Services - Configuration. This information should be kept private and shall not be shared with anyone else.

Getting Started

You can only use Gift Packages with your game if your game is capable of sending requests to our server. At this time, players cannot redeem gift codes without connecting to the internet.

You can use one of the following methods to complete the process of redeeming a gift code.

MethodAPIThe game needs to verify that the player meets the conditions of claiming the giftsThe game needs to provide an interface for delivering items to the player
Verified by game/api/v1.0/cdk/game/submit-check
Requested by game/api/v1.0/cdk/game/submit-send
Redeeming without a server/api/v1.0/cdk/game/submit-simple
Verified by game (used on a web page)/api/v1.0/cdk/page/submit-check
Requested by game (used on a web page)/api/v1.0/cdk/page/submit-send

Steps to Integrate Gift Packages Into Your Game

  • Finish setting up the game and obtain the Client ID and Server Secret.
  • Optional: Go to Operational Tools - Gifts - Server Configuration and add a server that will be used to receive notifications and deliver items to players.
  • Create gift events and export gift codes. Once you finish this step, you will be ready to test the gift codes.
  • Players redeem your gift codes and obtain the items.

Steps to Redeem a Gift Code

Below are the basic steps for redeeming a gift code:

  1. Send the gift code and configurations to the interface for redeeming gift codes.
  2. The interface will check if the information provided is valid.
  3. The system will determine if the game’s interface for secondary checks needs to be invoked. (Optional; the interface is used if it is configured)
  4. If all the checks have passed, the system will determine if the game’s interface for delivering items to players needs to be invoked. (Optional; the interface is used if it is configured)

There are three different types of interfaces for redeeming a gift code. You can integrate one of them into your game:

NameDescription
Verified by gameChoose this flow if you plan to set up a server that connects to our system, are able to have your secrets securely stored, and wish to have your app verify if a request for redeeming a code should be allowed while having an interface provided by your game invoked to deliver the items to the player.
Requested by gameChoose this flow if you plan to set up a server that connects to our system, are able to have your secrets securely stored, and wish to have an interface provided by your game invoked to deliver the items to the player.
Redeeming without a serverChoose this flow if you don’t want to build and maintain your own interface. Your game will be invoking an interface provided by our system and a value will be returned indicating whether the request succeeded.

Three Ways for Redeeming

Verified by Game

When your game sends a request to the submit-check interface for redeeming a gift code, the system will check the inventory as well as whether the submitted data is valid. If everything looks good, the system will invoke the URL you provided for secondary checks with the conditions you set up on the Developer Center as arguments. If your interface approves the request, the system will invoke your interface for delivering items to the player with the items configured on the Developer Center. Verified by game

Requested by Game

When your game sends a request to the submit-send interface for redeeming a gift code, the system will check the inventory as well as whether the submitted data is valid. If everything looks good, the system will invoke your interface for delivering items to the player with the items configured on the Developer Center. Requested by game

Redeeming Without a Server

When your game sends a request to the submit-simple interface for redeeming a gift code, the system will check the inventory as well as whether the submitted data is valid. If everything looks good, the system will return the items configured on the Developer Center to you. Redeeming without a server

Connecting to the Interface for Redeeming Items

Implementing Secondary Checks With Your Game

If you have set up custom redeeming rules on the Developer Center and you’re redeeming a gift code with an interface that has the capability of performing secondary checks, the system will send the redeeming rules to your game for a secondary check. The configuration shown in the image below will trigger the following request to the interface you set up on the Developer Center:

Custom redeeming rules

curl --location -g --request POST <URL_FOR_SECONDARY_CHECK> \
--header 'Content-Type: application/json' \
--data-raw '{"activity_id":<EVENT_ID>,"character_id":<USER_ID>,"content":"[{\"condition\":\"level\",\"operate":\"gt\",\"value\":10}]","ext":<THE_EXT_SUBMITTED>,"gift_code":<GIFT_CODE>,"nonce_str":"abcdc","server_code": <SERVER_CODE>,"sign":"42d2b58a8cd58b5e63d90524aaf63ef97e04f223","timestamp":1658825322}'

After verifying the request, please return a JSON containing the following fields to inform the system of the result.

{
"status": true,
"errorCode": "An error code string"
}

You can use snake_case as well if you wish to maintain consistency.

{
"status": true,
"error_code": "An error code string"
}

If you don’t need our system to know the specific error message, your interface can simply return a non-200 HTTP code to indicate that the redemption failed, or an HTTP code 200 to indicate that the redemption succeeded. This allows your interface to respond to requests without following the format specified above.

Request parameterTypeDescription
activity_idStringEvent ID
character_idStringUser ID
contentString (JSON)Custom redeeming rules
content.conditionStringCustom redeeming rules (condition)
content.operateStringCustom redeeming rules (comparison operator)
content.valueStringCustom redeeming rules (value)
extStringA value provided when sending a request to the interface
gift_codeStringThe gift code entered by the user
nonce_strStringA random string used for this request
server_codeStringServer code
signStringSignature
timestampNumberTimestamp with a second precision
Response parameterTypeDescription
errorCodeStringError type; same as error_code; if both of them exist, errorCode will be used
extString (optional)A string passed to the interface for delivering items to the player
statusBooleanWhether the request passed the verification; true means yes and false means no

Delivering Items to Players

After validation is completed, our system will send a request to the game’s interface for delivering items to players with the items configured on the Developer Center as well as the configured parameters. The configuration shown in the image below will trigger the following request to the interface you set up on the Developer Center: Custom redeeming rules

curl --location -g --request POST <URL_FOR_ITEM_DELIVERY> \
--header 'Content-Type: application/json' \
--data-raw '{"activity_id": <EVENT_ID>,"character_id":"uid","content":"[{\"name\": \"ITEM_NAME\", \"number\": 2}]","ext":<THE_EXT_SUBMITTED>,"gift_code":"gift_code","nonce_str":"abcdc","server_code":<SERVER_CODE>,"sign":"fbe23e6f6f5193355b29e9ea2913b1992af56346","check_ext":<RETURNED_EXT>,"timestamp":1658827492}'

After completing the request, please return a JSON containing the following fields to inform the system of the result. You can use snake_case as well if you wish to maintain consistency.

{
"status": true,
"errorCode": "An error code string"
}

If you don’t need our system to know the specific error message, your interface can simply return a non-200 HTTP code to indicate that the redemption failed, or an HTTP code 200 to indicate that the redemption succeeded. This allows your interface to respond to requests without following the format specified above.

Request parameterTypeDescription
activity_idStringEvent ID
character_idStringUser ID
server_codeStringServer code
contentString (JSON)Custom items set up on the Developer Center
content.nameStringItem name
content.numberNumberItem count
extStringA value provided when sending a request to the interface
gift_codeStringThe gift code entered by the user
nonce_strStringA random string used for this request
signStringSignature
check_extStringThe string provided by the interface that does secondary checks
timestampNumberTimestamp with a second precision
Response parameterTypeDescription
errorCodeStringError type; same as error_code; if both of them exist, errorCode will be used
statusBooleanWhether the request succeeded; true means yes and false means no

Redeeming Without a Server

With this method, you don’t have to implement an interface for validating requests and delivering items yourself.

If you choose to use this method, make sure to specify that the event doesn’t need a server when creating the event on the Developer Center.

Our system will provide the redemption result upon receiving a request. Your game will then determine whether to give the items to the player depending on the result.

Three Item Delivery Interfaces

With Secondary Check

POST /api/v1.0/cdk/game/submit-check

Header Content-Type:application/json

Request parameters:

Request parameterTypeRequiredDescription
client_idStringYesThe Client ID of the game
gift_codeStringYesThe gift code entered by the user
server_codeStringYesServer code
character_idStringYesUser ID
nonce_strStringYesA random string used for this request; it’s recommended to use 5 digits
signStringYesSignature
timestampNumberYesTimestamp with a second precision; the request will only be accepted if the timestamp is within 1 minute
extStringNoThis field will be passed to the game’s interface for secondary checks

Example request:

curl --location --request POST '<URL>/api/v1.0/cdk/game/submit-check' \
--header 'Content-Type: application/json' \
--data-raw '{"client_id":<CLIENT_ID>,"gift_code":<GIFT_CODE>,"server_code":<SERVER_CODE>,"character_id":<USER_ID>,"nonce_str":<NONCE>,"sign":<SIGNATURE>,"timestamp":<TIMESTAMP>,"ext": <THE_EXT_FIELD>}'

Without Secondary Check

POST /api/v1.0/cdk/game/submit-send

Header Content-Type:application/json

Request parameters:

Request parameterTypeRequiredDescription
client_idStringYesThe Client ID of the game
gift_codeStringYesThe gift code entered by the user
server_codeStringYesServer code
character_idStringYesUser ID
nonce_strStringYesA random string used for this request; it’s recommended to use 5 digits
signStringYesSignature
timestampNumberYesTimestamp with a second precision; the request will only be accepted if the timestamp is within 1 minute

Example request:

curl --location --request POST '<URL>/api/v1.0/cdk/game/submit-send' \
--header 'Content-Type: application/json' \
--data-raw '{"client_id":<CLIENT_ID>,"gift_code":<GIFT_CODE>,"server_code":<SERVER_CODE>,"character_id":<USER_ID>,"nonce_str":<NONCE>,"sign":<SIGNATURE>,"timestamp":<TIMESTAMP>,"ext": <THE_EXT_FIELD>}'

Redeeming Without a Server

POST /api/v1.0/cdk/game/submit-simple

Header Content-Type:application/json

Request parameters:

Request parameterTypeRequiredDescription
client_idStringYesThe Client ID of the game
gift_codeStringYesThe gift code entered by the user
character_idStringYesUser ID
nonce_strStringYesA random string used for this request; it’s recommended to use 5 digits
signStringYesSignature; use the Client ID instead of the Secret for generating this signature

Response parameters:

Response parameterTypeDescription
activity_idStringEvent ID
nonce_strStringA random string
timestampStringTimestamp
contentString (JSON)Custom items configured on the Developer Center
content.nameStringItem name
content.numberNumberItem count
signStringReturned signature used to check if the returned data has been tampered
successBooleanWhether the request succeeded

Example request:

curl --location --request POST '<URL>/api/v1.0/cdk/game/submit-simple' \
--header 'Content-Type: application/json' \
--data-raw '{"client_id":<CLIENT_ID>,"gift_code":<GIFT_CODE>,"character_id":<USER_ID>,"nonce_str":<NONCE>,"sign":<SIGNATURE>,"timestamp":<TIMESTAMP>}'
Shell Request example:
#! /usr/bin/env bash

# TapDC Background application Client ID
client_id="Replace with console's `Client ID`"
# The conversion code of this exchange can be found in the "Package Activity panel - Data - Export" and then the exported file in the format of S0LLC8ICB2MP2
gift_code="Replace with the redemption code in the package panel"

# A random character string. Five random characters are recommended
nonce_str="A2B3Z"
# Game user ID
character_id="6347de128b****3ee825e029"
# Signature, which replaces Secret with ClientId
signed=$(echo -n $(date +%s)$nonce_str$client_id |openssl sha1)
RESULT_REQUEST=`curl --location --request POST 'https://poster-api.xd.cn/api/v1.0/cdk/game/submit-simple' \
--header 'Content-Type: application/json' \
--data-raw "{\"client_id\":\"$client_id\",\"gift_code\":\"$gift_code\",\"character_id\":\"$character_id\",\"nonce_str\":\"$nonce_str\",\"sign\":\"$signed\",\"timestamp\":$(date +%s)}"`

echo $RESULT_REQUEST
Android Request example:

available for reference Android Demo Gift package system function

C# Request example:
using UnityEngine.Networking;
using System;
using System.Text;
using System.Security.Cryptography;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class APIClient : MonoBehaviour
{
private const string apiUrl = "https://poster-api.xd.cn/api/v1.0/cdk/game/submit-simple";
private const string contentType = "application/json";
private const string clientId = "hskcocvse6x1cgkklm";
private const string giftCode = "NZ4mp2cztRMXH";
private const string characterId = "879a0cdd9nnb917055ee";
private const string nonceStr = "RFG7U";

public GUISkin demoSkin;


void Start()
{

}


private void OnGUI()
{

GUI.skin = demoSkin;
float scale = 1.0f;


float btnWidth= Screen.width / 5 * 2;
float btnWidth2 = btnWidth + 80 * scale;

float btnHeight = Screen.height / 25;
float btnTop = 30 * scale;
float btnGap = 20 * scale;

GUI.skin.button.fontSize = Convert.ToInt32(13 * scale);

var style = new GUIStyle(GUI.skin.button) { fontSize = 20 };
var labelStyle = new GUIStyle(GUI.skin.label) { fontSize = 30 };

if (GUI.Button(new Rect((Screen.width - btnGap) / 2 - btnWidth, btnTop, btnWidth /2, btnHeight), "返回", style))
{
UnityEngine.SceneManagement.SceneManager.LoadSceneAsync(0);

}

btnTop += btnHeight + 20 * scale;

if (GUI.Button(new Rect((Screen.width - btnGap) / 2 - btnWidth, btnTop, btnWidth, btnHeight), "无服务器兑换", style))
{
StartCoroutine(StartRequest());
}


}

private IEnumerator StartRequest()
{
// Get the current timestamp (seconds)
int timestamp = (int)(System.DateTime.UtcNow.Subtract(new System.DateTime(1970, 1, 1))).TotalSeconds;

// Concatenate and encrypt the sign parameter
string sign = GetSign(timestamp, nonceStr, clientId);

// Build request parameter
string requestBody = "{\"client_id\":\"" + clientId + "\",\"gift_code\":\"" + giftCode + "\",\"character_id\":\"" + characterId + "\",\"nonce_str\":\"" + nonceStr + "\",\"timestamp\":" + timestamp + ",\"sign\":\"" + sign + "\"}";

// Create the UnityWebRequest object
UnityWebRequest request = new UnityWebRequest(apiUrl, "POST");

// Set request header
request.SetRequestHeader("Content-Type", contentType);

// Set request body
byte[] bodyRaw = Encoding.UTF8.GetBytes(requestBody);
request.uploadHandler = new UploadHandlerRaw(bodyRaw);
request.downloadHandler = new DownloadHandlerBuffer();

// send request
yield return request.SendWebRequest();

// Processing response
if (request.result == UnityWebRequest.Result.Success)
{
Debug.Log("API request succeeded");
Debug.Log(request.downloadHandler.text);
}
else
{

Debug.Log("API request failed: " + request.error);
Debug.Log(request.downloadHandler.text);

}
}

private string GetSign(int timestamp, string nonceStr, string clientId)
{
// Concatenate parameters and perform SHA1 encryption
string signString = timestamp.ToString() + nonceStr + clientId;
byte[] signBytes = Encoding.UTF8.GetBytes(signString);
byte[] signHash = new SHA1CryptoServiceProvider().ComputeHash(signBytes);
string sign = BitConverter.ToString(signHash).Replace("-", "").ToLowerInvariant();

return sign;
}
}


Example response:

{
"activity_id": "TDS20220928151122U9H",
"content": "[{\"name\": \"ITEM\", \"number\": 2}]",
"nonce_str": "0DD4B",
"sign": "5895f87d3dfb5e0918c1b195c015d9284609d122",
"timestamp": 1664354023,
"success": true
}

Response

If succeeded:

{
"success": true,
"messages": "Success"
}

If failed:

Response parameterDescription
errorError code
messageSummary of the error
infoError detail
info.dev_messageBrief instructions for the developer
info.hintDetailed instructions
{
"error": 100001,
"message": "Invalid input",
"info": {
"dev_message": "",
"hint": "test error"
}
}

Request Domain

TapTap editionDomain
China (taptap.cn)https://poster-api.xd.cn
Other countries and regions (taptap.io)https://poster-api.xd.com

Signature

For security reasons, when your game communicates with our system, it needs to provide a signature in each request. The “Secret” mentioned below refers to the Server Secret found on Game Services > Configuration on the Developer Center.

tip

To ensure the highest security, please never use the Server Secret of your game on the client side.

Pseudo-code for obtaining a signature:

sign == sha1(timestamp + nonce_str + secret)

You can check if the process of obtaining signatures is implemented correctly with the help of unit testing. The code below is an example in golang:

func TestMakeSign(t *testing.T) {
timestamp := 1655724586
nonceStr := "abcde"
secret := "abc"
sign := MakeSign(int64(timestamp), nonceStr, secret)
assert.Equal(t, sign, "3cb8c38833fa742e7873378faddcbe5b56088482")
//output: 3cb8c38833fa742e7873378faddcbe5b56088482
}

To ensure that the Secret never appears on the client side, when redeeming without a server, the signature can be generated with the Client ID instead of the Secret.

sign == sha1(timestamp + nonce_str + client_id)

Using the Redemption Interface on a Web Page

Our system provides a simple interface for redeeming gift codes that supports CAPTCHA. If you have a web page for your event, you can embed this interface into the web page. This interface shares the same flow as those introduced earlier. The only difference is that some of the parameters have been changed.

Getting a CAPTCHA

tip

Each CAPTCHA is only valid for 5 minutes. You can request a new CAPTCHA if one expires.

GET /api/v1.0/cdk/page/captcha-img

Header Content-Type:application/json

Response parameterDescription
imgThe CAPTCHA image in base64
keyThe key of the CAPTCHA; needs to be submitted together with the user input

Example response:

{
"img": "",
"key": "XfNTN1gQyvA2AcNOu1UN"
}

Using a Web Page With Secondary Checks

POST /api/v1.0/cdk/page/submit-check

Header Content-Type:application/json

Request parameters:

Request parameterTypeRequiredDescription
client_idStringYesThe Client ID of the game
gift_codeStringYesThe gift code entered by the user
server_codeStringYesServer code
character_idStringYesUser ID
nonce_strStringYesA random string used for this request; it’s recommended to use 5 digits
verify_captcha_keyStringYesThe key of the CAPTCHA
verify_captcha_codeStringYesThe CAPTCHA code entered by the user
extStringNoThis field will be passed to the game’s interface for secondary checks

Using a Web Page Without Secondary Checks

POST /api/v1.0/cdk/page/submit-send

Header Content-Type:application/json

Request parameters:

Request parameterTypeRequiredDescription
client_idStringYesThe Client ID of the game
gift_codeStringYesThe gift code entered by the user
server_codeStringYesServer code
character_idStringYesUser ID
nonce_strStringYesA random string used for this request; it’s recommended to use 5 digits
verify_captcha_keyStringYesThe key of the CAPTCHA
verify_captcha_codeStringYesThe CAPTCHA code entered by the user

Miscellaneous

Error Codes

Error codeDescription
100001Invalid input
100002Invalid account
100003Other error
100004Wrong CAPTCHA
100005Parameter error
100006Quota exceeded
100007Batch quota exceeded
100008The event hasn’t started yet or has already ended
100009Not open yet
100010Out of scope
100011Too many requests
100012The user redeemed the same gift too many times
100013Invalid CDKEY
100014The server is responding slower than usual
100015Failed to deliver the items
100016Invalid gift code
100017Invalid event
100018Incorrect interface used for the current event
500001Server error
500002Failed to validate with an internal interface

Comparison Operators

Comparison operator codeDescription
ltLess than
leLess than or equal to
eqEqual to
neNot equal to
geGreater than or equal to
gtGreater than

IP Addresses of the System

If your interface can only accept a limited number of IP addresses, please add the IP address listed below.

TapTap editionIP
China (taptap.cn)59.110.228.98
Other countries and regions (taptap.io)8.214.95.148