NAV Navbar
Sample Code and Data

Introduction

Updated: 2020-04-28

Welcome to the Prepaid2Cash Partner API.

The Prepaid2Cash Partner API opens the full range of P2C prepaid card services to registered Partners, including:

We offer mobile SDKs for iOS and Android, or you can integrate directly to our REST API.

Navigating this Document

Primary content and API documentation is in this middle column. The right-hand column contains corresponding API request and response samples.

Use the left-hand column navigation to quickly jump to any section of the documentation, or to search all sections.


Basics

Prepaid Debit Cards

The Prepaid2Cash API can be used to redeem both Open Loop and Closed Loop prepaid debit cards.

Open Loop prepaid debit cards are accepted at most merchants, and are typically branded with a Visa, Mastercard, American Express, or Discover logo. These cards are also known as "network-branded" cards, or generically as "prepaid" cards.

Closed Loop prepaid debit cards are only accepted by the merchant which issued the card (or an affiliate). These cards have merchant branding, do not have a network logo, and are commonly called "store cards", or "gift cards".

Environments

Prepaid2Cash operates multiple API environments. API keys are issued separately for each environment.

Production

🔵 Production Base URL

https://api.prepaid2cash.com/

The Production environment is our "live" environment. Transaction data is sent to processing networks for capture and disbursement of real funds.

Sandbox

🔵 Sandbox Base URL

https://sandbox.api.prepaid2cash.com/

The Sandbox environment is used to test integrations with Prepaid2Cash in a non-production environment. API requests to the Sandbox environment will not send transaction data to processing networks, so will never transfer funds to or from live prepaid cards.

Authentication

🔵 HTTP Authorization Header

Authorization: pk_test_ImlkX2Zvcm0iOiJkcml

Access to the Prepaid2Cash Partner API requires an API key. To request an API key, contact Prepaid2Cash support.

All requests to the Prepaid2Cash Partner API must include your API key in the Authorization HTTP header, as shown in the right-hand column.

Monetary Amounts

✅ OK ($26.00 USD)

  { "amount": 2600 }

Monetary amounts in the Prepaid2Cash API are always specified in USD, and always expressed as an integer representation of the smallest denomination of the currency, i.e. USD cents.

Timestamps

✅ OK

  { "created_at": "2019-07-04T00:00:00Z" }

Timestamps are always specified in UTC, and always represented in ISO 8601 format, except where superseding protocols specify otherwise (e.g. JSON Web Tokens).

Idempotency

The API supports idempotency for safely retrying requests without accidentally performing the same operation twice. This is useful when an API call is disrupted in transit and you do not receive a response. For example, if a request to create a transaction does not respond due to a network connection error, you can retry the request with the same idempotency key to guarantee that only one transaction is created.

🔵 HTTP Idempotency-Key Header

Idempotency-Key: b0defbc9-a53f-4534-b969-48cfcc39fd4d

To perform an idempotent request, include an Idempotency-Key header in your request.

An idempotency key is a unique value generated by the API client. We suggest using V4 UUIDs.

Errors

The API returns conventional HTTP response codes to indicate the results of each request.

Generally, response codes in the 4xx range indicate an error with submitted data, and the 5xx range indicates a server error.

When additional error information is available (e.g. most 400 and 422 responses), it will be included in the response object.


Mobile SDKs

Android Ios

Prepaid2Cash Mobile SDKs are available for iOS and Android. There are two variations of each:

We'll take a closer look at the Prepaid2Cash Mobile SDKs in the integration guides below.

To get started with Prepaid2Cash Mobile SDKs, you will need an API key.

Contact Prepaid2Cash support for downloads and access to the P2C sandbox environment. Then have a look at the SDK integration guides here on this site.


Android

Requirements

Configure Repositories

allprojects {
  repositories {
    maven {
        url 'https://developer.prepaid2cash.com/maven2'
        credentials {
            username "yourusername"
            password "yourpassword"
        }
    }
    maven { url 'https://jitpack.io' }
  }
}

In your root-level (project-level) Gradle file, add maven repositories for Prepaid2Cash and JitPack.

Customize the Card Reader View

colors.xml

<color name="p2c_color_primary">#71c05e</color>

strings.xml

<string name="p2c_card_scan_text">Hold your card here.\nIt will scan automatically.</string>

The appearance of the card reader view is configurable with custom colors, instructional text, and logo. Modifications can be made in the following files:

res/values/colors.xml

res/values/strings.xml

res/drawable-*/p2c_card_reader_logo.png

UI SDK

The Prepaid2Cash UI SDK contains the full flow required to create a transaction with a prepaid debit card. It requires just a few lines of code to integrate into your mobile app.

Add Dependencies

dependencies {
    implementation 'com.prepaid2cash:Prepaid2CashUISDK:0.6.1'
}

In your module (app-level) Gradle file, add com.prepaid2cash to your build.gradle dependencies.

Launch the Flow

Kotlin

val apiKey = "pk_test_your_api_key"
val UAT = "your_user_authorization_token"
val productionFlag = false

startActivity(Prepaid2CashActivity.getStartIntent(context, apiKey, UAT, productionFlag))

Java

String apiKey = "pk_test_your_api_key";
String UAT = "your_user_authorization_token";
boolean productionFlag = false;

startActivity(Prepaid2CashActivity.Companion.getStartIntent(this, apiKey, UAT, productionFlag));

The Prepaid2CashActivity class is the entry-point to the Prepaid2Cash SDK. It must be initialized with an API Key.

To create users and transactions, you will need to provide a User Authorization Token.

Process Transaction Results

The SDK will return a Transaction Results Token, which can be used to display results in your mobile app, and sent to your API for verification and processing.

Core SDK

The Prepaid2Cash Core SDK contains the essential methods to access the P2C API, making it easy to integrate into your app's existing flow and design.

Add Dependencies

dependencies {
    implementation 'com.prepaid2cash:Prepaid2CashSDK:0.1.4'
}

In your module (app-level) Gradle file, add com.prepaid2cash to your build.gradle dependencies.

Initialization

val yourSecretKey = "pk_test_your_api_key"
val isProduction = false

Prepaid2Cash.initApp(yourSecretKey, isProduction)

The first step is to initialize the SDK, using your API key.

To create a transaction, choose the Prepaid flow for Open Loop cards, or the GiftCard flow for Closed Loop cards.

Both flows require a User Authorization Token.

Core SDK, Open Loop Transaction

Card Data Entry via Optical Scan/OCR

Card Data Entry via Optical Scan/OCR

startActivityForResult(ScanCardActivity.getStartIntent(this), 123)
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
  super.onActivityResult(requestCode, resultCode, data)

  if (requestCode == 123) {
    if (resultCode == Activity.RESULT_OK) {
      val cardNumber = data?.getStringExtra(Prepaid2Cash.EXTRA_CARD_NUMBER)
      val cardType = data?.getStringExtra(Prepaid2Cash.EXTRA_CARD_TYPE)
      val byteArray = data?.getByteArrayExtra(Prepaid2Cash.EXTRA_BYTE_ARRAY)
    } else if (resultCode == Activity.RESULT_CANCELED) {
      Log.e(TAG, "ERROR")
    }
  }
}

The Prepaid2Cash Mobile SDKs include a card scanner to identify and read card data through your customer's phone camera.

To launch the scanner, simply start the activity ScanCardActivity, calling it with startActivityForResult.

After successfully scanning the card, you'll receive the result, overriding the onActivityResult method and reading the values from the Intent.

cardNumber and cardType will be used in the next step.

Start a Prepaid Card (Open Loop) Transaction

Start a Prepaid Card (Open Loop) Transaction

val UAT = "your_user_authorization_token"
val prepaidCard = Prepaid2CashPrepaidCard.withUAT(UAT)

prepaidCard?.processTransactionData(
  amount = amount,
  pan = cardNumber,
  expiryMonth = expiryMonth,
  expiryYear = expiryYear,
  cvv = cvv,
  description = cardType,
  prepaidCardImageByteArray = byteArray,
  isOCR = true,
  listener = object :  ResponseListener<RenderPrepaidTransaction> {
    override fun onSuccessfulResponse(response: RenderPrepaidTransaction) {
    }
    override fun onErrorResponse(error: Throwable) {
    }
  }
)

Before starting a new open loop transaction, it will be necessary to collect some transaction data from the customer:

We recommend performing basic validation on this data before proceeding.

The RenderPrepaidTransaction object will contain transaction details for the customer to confirm, e.g. payout amount and estimated delivery. The full response data is documented at Create Open Loop Transaction

Confirm Transaction

Confirm Transaction

Prepaid2CashPrepaidCard.withUAT(UAT)?.saveTransactionData(
  listener = object : ResponseListener<String>{
    override fun onSuccessfulResponse(response: String) {
    }
    override fun onErrorResponse(error: Throwable) {
    }
  }
)

The next step is to confirm and submit the transaction.

Process Transaction Results

The SDK will return a Transaction Results Token, which can be used to display results in your mobile app, and sent to your API for verification and processing.


Core SDK, Closed Loop Transaction

Fetch Merchants list

Fetch Merchants list

Prepaid2CashGiftCard.withUAT(UAT)?.getMerchants(
  listener = object : ResponseListener<List<String>> {
    override fun onSuccessfulResponse(response: List<String>) {
    }
    override fun onErrorResponse(error: Throwable) {
    }
  }
)

The first step in creating a new closed loop transaction is to fetch the list of active Gift Card merchants. The active list generally includes 80-120 merchants, and changes regularly.

Fetch ID Types list

Fetch ID Types list

Prepaid2CashGiftCard.withUAT(UAT)?.getIdTypes(
  listener = object : ResponseListener<List<String>> {
    override fun onSuccessfulResponse(response: List<String>) {
    }
    override fun onErrorResponse(error: Throwable) {
    }
  }
)

Next, we fetch the list of valid ID types. This list is static, and does not require a network request.

Start a Gift Card (Closed Loop) Transaction

Start a Gift Card (Closed Loop) Transaction

Prepaid2CashGiftCard.withUAT(UAT)?.processGiftCardInfo(
  merchantName = merchantName,
  pan = giftCardNumber,
  pin = pinNumber,
  idDescription = idType,
  idNumber = idNumber,
  listener = object : ResponseListener<RenderGiftCardTransaction> {
    override fun onSuccessfulResponse(response: RenderGiftCardTransaction) {
    }
    override fun onErrorResponse(error: Throwable) {
    }
  }
)

After fetching the Merchant and ID types lists, we can begin to collect the transaction data from the customer:

It is not necessary to submit a transaction amount for closed loop cards -- this will be retrieved from the merchant.

We recommend performing basic validation on the transaction data before proceeding.

Prepaid2Cash will attempt to find an offer to purchase the customer's closed loop card. If successful, the offer details, including payout amount and estimated delivery, should be presented to the customer to accept or decline.

The RenderGiftCardTransaction object will contain transaction details. The full response data is documented at Create Closed Loop Transaction

Accept Offer

Accept Offer

Prepaid2CashGiftCard.withUAT(UAT)?.submitGiftCard(
  listener = object : ResponseListener<String> {
    override fun onSuccessfulResponse(response: String) {
    }
    override fun onErrorResponse(error: Throwable) {
    }
  }
)

If the customer accepts the offer, call submitGiftCard to confirm.

Decline Offer

Decline Offer

Prepaid2CashGiftCard.withUAT(UAT)?.declineGiftCard(
  listener = object : ResponseListener<String> {
    override fun onSuccessfulResponse(response: String) {
    }
    override fun onErrorResponse(error: Throwable) {
    }
  }
)

If the customer declines the offer, call declineGiftCard to close the transaction.

Process Transaction Results

If the customer as accepted the offer, the SDK will return a Transaction Results Token, which can be used to display results in your mobile app, and sent to your API for verification and processing.


iOS

SDK Version 1.0.1

UI SDK

The Prepaid2Cash UI SDK contains the full flow required to create a transaction with a prepaid debit card. It requires just a few lines of code to integrate into your mobile app.

The UI flow is implemented in 5 steps:

  1. Set API Key
  2. Set User Authorization Token (UAT)
  3. Set Delegate
  4. Present Flow
  5. Process Transaction Results

Set API Key

Prepaid2Cash.config(apiKey: "pk_test_your_api_key")`

Configure in your AppDelegate, in didFinishLaunchingWithOptions.

Set User Authorization Token (UAT)

let myUAT = "my_user_authorization_token"

Prepaid2Cash.config(uat: myUAT)

The User Authorization Token should be generated and signed by your backend. More about UATs.

Set Delegate

public protocol P2CTransactionDelegate {
    func transactionFinishedSuccess(jwt: String)
    func transactionFinishedWithError(error: Error?)
    func transactionEvent(event: P2CEvent)
}
Prepaid2Cash.shared().delegate = self

Delegate needs to be set to a class that conforms to the expected protocol.

Present Flow

Prepaid2Cash.launch(navigationController: navigationController!)

The final step of the UI flow is to present the SDK view.

When the SDK flow completes, your app will return to the original presenting view.

Process Transaction Results

The SDK will return a Transaction Results Token, which can be used to display results in your mobile app, and sent to your API for verification and processing.

Core SDK

The Prepaid2Cash Core SDK contains the essential methods to access the API, making it easy to integrate into your app's existing flow and design.

Core SDK, Open Loop Transaction

For Prepaid Cards, follow these steps:

  1. Set Scanner Delegate
  2. Scan Card Using Prepaid2Cash OCR Library
  3. Collect Additional Card Data
  4. Retrieve and Select Payout Options
  5. Create Transaction
  6. Process Transaction Results

Set Scanner Delegate

Prepaid2Cash.shared().scannerDelegate = self

Scan Card Using Prepaid2Cash OCR Library

Prepaid2Cash.launchCardScannerForPrepaidTransaction(from: self)

The delegate method will receive a masked version of the card number for display in your UI.

public func scannerResult(success: Bool, maskedCardNumber: String?) 

Collect Additional Card Data

Retrieve and Select Payout Options

let cvv = "123"
let expirationDate = "05/25"
let cardAmount = 10000

Prepaid2Cash.getPayoutOptions(amount: cardAmount, cardSubtype: .open_loop, completion: { options in 
  // Handle OK
}, failure: { error in
  // Handle Error
})

The API response will include the payout amount for each payout option.

Create Transaction

Prepaid2Cash.createPrepaidCardTransaction(amount: cardAmount, cvv: cvv, expirationDate: expirationDate, payoutOption: selectedOption, completion: { transactionResult in
  // Clear Scanner after successful or failed transaction
  Prepaid2Cash.clearScanner()
  print("Transaction result: \(transactionResult.value)")
}, failure: { error in
  // Handle Error
})

Process Transaction Results

func transactionFinishedSuccess(jwt: String)

The SDK will return a Transaction Results Token, which is a signed JWT containing the details of transaction results.

This token can be decoded to display transaction results in your mobile app, and can be sent to your backend API for verification and processing.

The TRT will be returned through the delegate at the end of the transaction process.

Core SDK, Closed Loop Transaction

For Gift Cards, follow these steps:

  1. Get Active Merchant List
  2. Collect Card Data
  3. Retrieve Payout Options
  4. Create Transaction
  5. Display Offer to User
  6. Accept or Decline Offer
  7. Process Transaction Results

Get Active Merchant List

Prepaid2Cash.getMerchantList(completion: { merchantList in 
  // Handle OK
}), failure: { error in
  // Handle Error
})

Collect Card Data

Retrieve Payout Options

let merchant = merchantList.first!
let giftCardNumber = "9111111111111106"
let giftCardPin = "2749"

// Retrieve and select payout option
Prepaid2Cash.getPayoutOptions(bin: giftCardNumber, amount: nil, cardSubtype: .closed_loop, completion: { options in
  // Handle OK
}), failure: { error in
  // Handle Error
})

Create Transaction

Prepaid2Cash.createGiftCardTransaction(cardNumber: giftCardNumber, cardPin: giftCardPin, payoutOption: selectedOption, merchant: merchant, completion: { transaction in
  // Handle OK
}), failure: { error in
  // Handle Error
})

Display Offer to User

Accept or Decline Offer

// Accept Offer

Prepaid2Cash.acceptGiftCardTransaction(transaction: currentTransaction, completion: { transactionResult in
  print("Transaction result: \(transactionResult.value)")
}, failure: { error in
  // Handle Error
})
// Decline Offer

Prepaid2Cash.declineGiftCardTransaction(transaction: currentTransaction, completion: { transactionResult in
  // Handle OK
}, failure: { error in
  // Handle Error
})

Process Transaction Results

func transactionFinishedSuccess(jwt: String)

The SDK will return a Transaction Results Token, which is a signed JWT containing the details of transaction results.

This token can be decoded to display transaction results in your mobile app, and can be sent to your backend API for verification and processing.

The TRT will be returned through the delegate at the end of the transaction process.


REST API

The Prepaid2Cash Partner API is a RESTful API, which accepts and returns JSON-encoded data, and uses standard HTTP verbs, response codes, and authentication.

If you don't need the features offered by our Mobile SDKs, you can integrate directly via our REST API.


Functional Flows

New Open Loop Transaction

Open Loop ("prepaid card") transactions attempt to debit the user's prepaid card. If successful, funds are captured immediately and paid out to customer, or transferred to Partner for funding or payout, as specfied in Partner agreement.

Open Loop transactions can be created in two or three steps:

  1. Create User (Direct API-API integration only, optional)
  2. Retrieve Transaction Options by submitting initial transaction data.
  3. Create Open Loop Transaction after user has confirmed the full transaction details, including payout amount and estimated delivery.

New Closed Loop Transaction

Closed Loop ("gift card") transactions attempt to purchase the user's card for inventory sales into a secondary market. If the card is purchaseable, an offer is returned to the customer for approval. If approved, the card is purchased and proceeds are paid out to customer, or transferred to Partner for funding or payout, as specfied in Partner agreement.

Closed Loop transactions can be created in four or five steps:

  1. Create User (Direct API-API integration only, optional)
  2. Retrieve Closed Loop Merchants for the list of active gift card merchants. This list changes regularly due inventory fluctuation.
  3. Retrieve Transaction Options by submitting initial transaction data.
  4. Create Closed Loop Transaction by submitting full gift card data.
  5. Accept Closed Loop Offer after user has approved the sale amount.

API Endpoint Reference


Create User

This endpoint creates a new user record. It returns the new user ID and an SMS verification code which has been sent to the user's phone. This code will be required to submit a transaction, and we also recommend verifying this code in your interaction flow.

If identifying user details match an existing Partner user, a new record will not be created, and the existing user ID will be returned instead.

In both cases, a new SMS verification code will be sent, and included in the response.

If your application requires customers to scan a photo ID, you may send raw PDF417 data (AAMVA standard, 2005 or newer) in the id_pdf417 field. If included, only additional fields phone and email are required. Contact us to enable this feature for your partner account.

HTTP Request

POST https://api.prepaid2cash.com/v2/users

Request Body Fields

⏩ Request (PDF417 plus required and optional fields)

POST /v2/users HTTP/1.1
Host: api.prepaid2cash.com
Authorization: pk_test_ImlkX2Zvcm0iOiJkcml

{
  "id": "d54747d1-2993-4bdc-a745-321f7f33e8af",
  "phone": "+1 415-456-1000",
  "email": "norton@sf.ca.us",
  "id_pdf417": "@\n\u001e\rANSI 636004080002DL00410266ZN03070017DLDAQ123456789123\nDCSNORTON\nDDEN\nDACJOSHUA\nDDFN\nDAD\nDDGN\nDCAC\nDCBNONE\nDCDNONE\nDBD02041976\nDBB08151987\nDBA08152020\nDBC2\nDAU070 in\nDAYBRO\nDAG624 COMMERCIAL STREET\nDAISAN FRANCISCO\nDAJCA\nDAK941111010\nDCF0123456789\nDCGUSA\nDAZBRO\nDCLU  \nDCK000012345678NCSVTL01\nDDB10242014\nDDK1\nDDL1\nZNZNADUP\nZNB\nZNC0\n",
  "user_photo": ""
}
Field Type Description
id string User ID, unique in Partner domain (optional)
phone string Telephone number with country code, ITU-T E.123 format, hyphens optional
email string Email address
id_pdf417 string Raw PDF417 2D barcode data (AAMVA standard) (optional)
user_photo string Base64 encoded image data (PNG, JPG, BMP) (optional)

✅ Response (Created)

201 Created
{
  "id": "e8379ab7-fe92-4ad0-b25e-6ba4ddb0999f",
  "sms_verification_code": "12345"
}

✅ Response (OK, Found)

200 OK
{
  "id": "e8379ab7-fe92-4ad0-b25e-6ba4ddb0999f",
  "sms_verification_code": "12345"
}

Retrieve Transaction Options

This endpoint returns the options and requirements needed to create a new Open Loop transaction. The request includes preliminary transaction data. Options returned may be presented to the prepaid card customer for selection, or may be selected by Partner policy.

HTTP Request

POST https://api.prepaid2cash.com/v2/transaction_options

Request Body Fields

⏩ Request

POST /v2/transaction_options HTTP/1.1
Host: api.prepaid2cash.com
Authorization: pk_test_ImlkX2Zvcm0iOiJkcml

{
  "amount": 10000,
  "prepaid_card": {
    "card_subtype": "open_loop",
    "bin": "411111"
  },
  "user_id": "e8379ab7-fe92-4ad0-b25e-6ba4ddb0999f",
  "user_authorization": "eyJ0eXAiOiJKV1QiLCJhbGciOiJSUzI1NiIsImtpZCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCNTMjU2Ijoickp2eXZOZGFEOEk1V293bmM3ZnhMMTlQS0NFREsxWGxyMHpYQndhMGxHUSJ9.eyJpc3MiOiJwYXJ0bmVyZG9tYWluLmNvbSIsImF1ZCI6InByZXBhaWQyY2FzaC5jb20iLCJpYXQiOjE1Nzc4MzY4MDAsImV4cCI6MjU3NzgzNjgwMCwiaHR0cDovL3AyYy5pby91c2VyIjp7ImlkIjoiZDU0NzQ3ZDEtMjk5My00YmRjLWE3NDUtMzIxZjdmMzNlOGFmIiwiZmlyc3RfbmFtZSI6Ikpvc2h1YSIsImxhc3RfbmFtZSI6Ik5vcnRvbiIsImFkZHJlc3MxIjoiNjI0IENvbW1lcmNpYWwgU3QiLCJhZGRyZXNzMiI6IiIsImNpdHkiOiJTYW4gRnJhbmNpc2NvIiwic3RhdGUiOiJDQSIsImNvdW50cnkiOiJVUyIsInBvc3RhbF9jb2RlIjoiOTQxMTEtMTAxMCIsInBob25lIjoiKzEgNDE1LTQ1Ni0xMDAwIiwiZW1haWwiOiJub3J0b25Ac2YuY2EudXMiLCJkb2IiOiIxOTc2LTAyLTA0Iiwic3NuX2xhc3RfNCI6IjE3NzYiLCJpZF9mb3JtIjoiZHJpdmVyc19saWNlbnNlIiwiaWRfaXNzdWVyIjoiVVMvQ0EiLCJpZF9udW1iZXIiOiJEMTIzNDU2NzgiLCJreWNfdmVyaWZpZWQiOlsiZmlyc3RfbmFtZSJdfX0.nbywUp4vkQ8uCphzpykrDm7jLHMl8Ixt9DJPIhWQQo7RYbIV6lmr4EwOWCg_TNcjf7l4jNhd33daD1qeCSNxpKMUs45joJlVKFPnjiu32PGNzfOdRw28ElWVCXzQOTR5_OLoRvx6xX_oIOQI9FuMlHyoNq9G6hBpM1zOey74GYunO6WNJPbDjh5bUpZB9L4z_OPs3SgzKzpOuTOLS7tBiiJsgHyS8pex4-_ErTHbrXsennZ8twV5b8xHFC1GtZb3xjm4M1Zs_j1vSGagiHxqp_0VtzMCcKtN31xjS9epk0wDPi4nI_SAkGlvt1XINQ3VNKvPKiA3RlvgDenfqYbize7eZRKFf4W1afODIe1WSXtgoyLa3hbwj_Qf1cqOlnsbRNJw1WVnGX-qdnw-q5MBTZMlgbWJGz65f4YlIW47zjuOHUmDQl3Z5gKFrUGYn6L2CrImnXimIctmbG08H1vcXO75DYnqTlfWaJfRmdXuEEQ64UuRaEG4NotZf9Nq5kq95KlURC344NxraUhyb_1XWc4jDMOmsycfk1H39I77FL-NCH_iiHZ4togZvRicEkgTX7BCwwLOZIpUHaxIOo4k1WKZpTeVI4jUPTm3iX8kPPtsAAAxZF1IhSnGaPvaYjOLO3S0hAPDQfus7nA2-eH-KAQSS19OoS2nYSXyssRYTkc"
}
Field Type Description
amount integer Prepaid Card value in cents (omit for closed loop)
prepaid_card (dict)
⋯ card_subtype string Prepaid Card Subtype (open_loop or closed_loop)
⋯ bin string First 6 digits of PAN (optional)
user_id string User ID (optional)
user_authorization string Signed User Authorization Token, JWT-format (optional)

Response Fields (Success)

✅ Response (OK)

200 OK
{
  "payout_methods": [
    {
    "name": "PARTNER_ACH",
    "display_name": "Direct Deposit",
    "tag": "Most Popular",
    "description": "Direct deposit to your account.",
    "delivery_at": "2019-06-03T00:00:00Z",
    "amount": 9350,
    "fees": {
      "base": 150,
      "rate": 5000
      }
    },
    {
    "name": "PARTNER_INST",
    "display_name": "Transfer to Debit Card",
    "tag": "Fastest Delivery",
    "description": "Transfer funds to your debit card.",
    "delivery_at": "2019-06-01T00:00:00Z",
    "amount": 9100,
    "fees": {
      "base": 200,
      "rate": 7000
      }
    }
  ]
}
Field Description
payout_methods (array of dict)
⋯ name Name of payout method
⋯ display_name Name of payout method, for display
⋯ tag Auxiliary label of payout method
⋯ description Description of payout method
⋯ delivery_at Estimated delivery, ISO 8601 timestamp
⋯ amount Payout amount (integer cents, omitted for closed loop)
⋯ fees (dict)
⋯ ⋯ base Base (integer cents)
⋯ ⋯ rate Rate (integer percent * 1000)

Response Fields (Errors)

❌ Response (validation error)

400 Bad Request
{
  "errors": [
    {
      "code": "validation_error",
      "title": "Validation Error",
      "invalid_params": [
        {
          "name": "amount",
          "reason": "cannot be empty"
        }
      ]
    }
  ]
}

❌ Response (limit exceeded)

422 Unprocessable Entity
{
  "errors": [
    {
      "code": "daily_limit_exceeded",
      "title": "Limit Exceeded",
      "detail": "You have reached your daily limit of transactions."
    }
  ]
}
Field Description
errors (array of dict)
⋯ code Error code
⋯ title Title of error, for display
⋯ detail Error message, for display

Create Open Loop Transaction

This endpoint creates a new transaction.

Card data may be sent directly, or with an extra layer of encryption. Raw magnetic stripe data (MSD) can be sent in the encrypted_msd_track fields.

User data can be included in this request using the user_authorization field (see User Authorization Token). If the user has been created previously via Create User or an earlier transaction, send the user_id field instead.

HTTP Request

POST https://api.prepaid2cash.com/v2/transactions

Request Body Fields

⏩ Request (Example 1: Direct card data, with optional fields)

POST /v2/transactions HTTP/1.1
Host: api.prepaid2cash.com
Content-Type: application/json
Authorization: pk_test_ImlkX2Zvcm0iOiJkcml

{
  "amount": 10000,
  "payout_method": "PARTNER_ACH",
  "prepaid_card": {
    "card_subtype": "open_loop",
    "pan": "9123123412341234",
    "expiry_month": "01",
    "expiry_year": "2021",
    "cvv": "123",
    "pan_entry": "swipe"
  },
  "payout_card": {
    "card_subtype": "debit",
    "pan": "9123123412341234",
    "expiry_month": "01",
    "expiry_year": "2021",
    "cvv": "123",
    "pan_entry": "swipe"
  },
  "location": {
    "id": "78205-00300",
    "address": "300 Alamo Plaza, San Antonio, TX 78205, US",
    "lat": "29.42569",
    "long": "-98.48503",
  },
  "tos_accepted": true,
  "sms_verification_code": "12345",
  "user_id": "e8379ab7-fe92-4ad0-b25e-6ba4ddb0999f",
  "user_authorization": "eyJ0eXAiOiJKV1QiLCJhbGciOiJSUzI1NiIsImtpZCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCNTMjU2Ijoickp2eXZOZGFEOEk1V293bmM3ZnhMMTlQS0NFREsxWGxyMHpYQndhMGxHUSJ9.eyJpc3MiOiJwYXJ0bmVyZG9tYWluLmNvbSIsImF1ZCI6InByZXBhaWQyY2FzaC5jb20iLCJpYXQiOjE1Nzc4MzY4MDAsImV4cCI6MjU3NzgzNjgwMCwiaHR0cDovL3AyYy5pby91c2VyIjp7ImlkIjoiZDU0NzQ3ZDEtMjk5My00YmRjLWE3NDUtMzIxZjdmMzNlOGFmIiwiZmlyc3RfbmFtZSI6Ikpvc2h1YSIsImxhc3RfbmFtZSI6Ik5vcnRvbiIsImFkZHJlc3MxIjoiNjI0IENvbW1lcmNpYWwgU3QiLCJhZGRyZXNzMiI6IiIsImNpdHkiOiJTYW4gRnJhbmNpc2NvIiwic3RhdGUiOiJDQSIsImNvdW50cnkiOiJVUyIsInBvc3RhbF9jb2RlIjoiOTQxMTEtMTAxMCIsInBob25lIjoiKzEgNDE1LTQ1Ni0xMDAwIiwiZW1haWwiOiJub3J0b25Ac2YuY2EudXMiLCJkb2IiOiIxOTc2LTAyLTA0Iiwic3NuX2xhc3RfNCI6IjE3NzYiLCJpZF9mb3JtIjoiZHJpdmVyc19saWNlbnNlIiwiaWRfaXNzdWVyIjoiVVMvQ0EiLCJpZF9udW1iZXIiOiJEMTIzNDU2NzgiLCJreWNfdmVyaWZpZWQiOlsiZmlyc3RfbmFtZSJdfX0.nbywUp4vkQ8uCphzpykrDm7jLHMl8Ixt9DJPIhWQQo7RYbIV6lmr4EwOWCg_TNcjf7l4jNhd33daD1qeCSNxpKMUs45joJlVKFPnjiu32PGNzfOdRw28ElWVCXzQOTR5_OLoRvx6xX_oIOQI9FuMlHyoNq9G6hBpM1zOey74GYunO6WNJPbDjh5bUpZB9L4z_OPs3SgzKzpOuTOLS7tBiiJsgHyS8pex4-_ErTHbrXsennZ8twV5b8xHFC1GtZb3xjm4M1Zs_j1vSGagiHxqp_0VtzMCcKtN31xjS9epk0wDPi4nI_SAkGlvt1XINQ3VNKvPKiA3RlvgDenfqYbize7eZRKFf4W1afODIe1WSXtgoyLa3hbwj_Qf1cqOlnsbRNJw1WVnGX-qdnw-q5MBTZMlgbWJGz65f4YlIW47zjuOHUmDQl3Z5gKFrUGYn6L2CrImnXimIctmbG08H1vcXO75DYnqTlfWaJfRmdXuEEQ64UuRaEG4NotZf9Nq5kq95KlURC344NxraUhyb_1XWc4jDMOmsycfk1H39I77FL-NCH_iiHZ4togZvRicEkgTX7BCwwLOZIpUHaxIOo4k1WKZpTeVI4jUPTm3iX8kPPtsAAAxZF1IhSnGaPvaYjOLO3S0hAPDQfus7nA2-eH-KAQSS19OoS2nYSXyssRYTkc"
}

⏩ Request (Example 2: Encrypted card data)

POST /v2/transactions HTTP/1.1
Host: api.prepaid2cash.com
Content-Type: application/json
Authorization: pk_test_ImlkX2Zvcm0iOiJkcml

{
  "amount": 10000,
  "payout_method": "PARTNER_ACH",
  "prepaid_card": {
    "card_subtype": "open_loop",
    "encryption_alg": "3DES-112-DUKPT",
    "encryption_ksn": "FFFF9876543210E00008",
    "encrypted_card": "C25C1D1197D31CAA87285D59A5ABE72",
    "encrypted_msd_track1": "C25C1D1197D31CAA87285D59A892047426D918…",
    "encrypted_msd_track2": "C25C1D1197D31CAA87285D59A892047426D918…",
    "pan_entry": "swipe"
  },
  "payout_card": {
    "card_subtype": "debit",
    "encryption_alg": "3DES-112-DUKPT",
    "encryption_ksn": "FFFF9876543210E00008",
    "encrypted_card": "C25C1D1197D31CAA87285D59A5ABE72",
    "encrypted_msd_track1": "C25C1D1197D31CAA87285D59A892047426D918…",
    "encrypted_msd_track2": "C25C1D1197D31CAA87285D59A892047426D918…",
    "pan_entry": "swipe"
  },
  "location": {
    "id": "78205-00300",
    "address": "300 Alamo Plaza, San Antonio, TX 78205, US",
    "lat": "29.42569",
    "long": "-98.48503",
  },
  "tos_accepted": true,
  "sms_verification_code": "12345",
  "user_id": "e8379ab7-fe92-4ad0-b25e-6ba4ddb0999f",
  "user_authorization": "eyJ0eXAiOiJKV1QiLCJhbGciOiJSUzI1NiIsImtpZCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCNTMjU2Ijoickp2eXZOZGFEOEk1V293bmM3ZnhMMTlQS0NFREsxWGxyMHpYQndhMGxHUSJ9.eyJpc3MiOiJwYXJ0bmVyZG9tYWluLmNvbSIsImF1ZCI6InByZXBhaWQyY2FzaC5jb20iLCJpYXQiOjE1Nzc4MzY4MDAsImV4cCI6MjU3NzgzNjgwMCwiaHR0cDovL3AyYy5pby91c2VyIjp7ImlkIjoiZDU0NzQ3ZDEtMjk5My00YmRjLWE3NDUtMzIxZjdmMzNlOGFmIiwiZmlyc3RfbmFtZSI6Ikpvc2h1YSIsImxhc3RfbmFtZSI6Ik5vcnRvbiIsImFkZHJlc3MxIjoiNjI0IENvbW1lcmNpYWwgU3QiLCJhZGRyZXNzMiI6IiIsImNpdHkiOiJTYW4gRnJhbmNpc2NvIiwic3RhdGUiOiJDQSIsImNvdW50cnkiOiJVUyIsInBvc3RhbF9jb2RlIjoiOTQxMTEtMTAxMCIsInBob25lIjoiKzEgNDE1LTQ1Ni0xMDAwIiwiZW1haWwiOiJub3J0b25Ac2YuY2EudXMiLCJkb2IiOiIxOTc2LTAyLTA0Iiwic3NuX2xhc3RfNCI6IjE3NzYiLCJpZF9mb3JtIjoiZHJpdmVyc19saWNlbnNlIiwiaWRfaXNzdWVyIjoiVVMvQ0EiLCJpZF9udW1iZXIiOiJEMTIzNDU2NzgiLCJreWNfdmVyaWZpZWQiOlsiZmlyc3RfbmFtZSJdfX0.nbywUp4vkQ8uCphzpykrDm7jLHMl8Ixt9DJPIhWQQo7RYbIV6lmr4EwOWCg_TNcjf7l4jNhd33daD1qeCSNxpKMUs45joJlVKFPnjiu32PGNzfOdRw28ElWVCXzQOTR5_OLoRvx6xX_oIOQI9FuMlHyoNq9G6hBpM1zOey74GYunO6WNJPbDjh5bUpZB9L4z_OPs3SgzKzpOuTOLS7tBiiJsgHyS8pex4-_ErTHbrXsennZ8twV5b8xHFC1GtZb3xjm4M1Zs_j1vSGagiHxqp_0VtzMCcKtN31xjS9epk0wDPi4nI_SAkGlvt1XINQ3VNKvPKiA3RlvgDenfqYbize7eZRKFf4W1afODIe1WSXtgoyLa3hbwj_Qf1cqOlnsbRNJw1WVnGX-qdnw-q5MBTZMlgbWJGz65f4YlIW47zjuOHUmDQl3Z5gKFrUGYn6L2CrImnXimIctmbG08H1vcXO75DYnqTlfWaJfRmdXuEEQ64UuRaEG4NotZf9Nq5kq95KlURC344NxraUhyb_1XWc4jDMOmsycfk1H39I77FL-NCH_iiHZ4togZvRicEkgTX7BCwwLOZIpUHaxIOo4k1WKZpTeVI4jUPTm3iX8kPPtsAAAxZF1IhSnGaPvaYjOLO3S0hAPDQfus7nA2-eH-KAQSS19OoS2nYSXyssRYTkc"
}
Field Type Description
amount integer Prepaid Card value in cents
payout_method string Selected from transaction_options response
prepaid_card (dict)
⋯ card_subtype string Prepaid Card subtype (open_loop)
⋯ (card data) (see below)
⋯ pan_entry string PAN entry method (scan, swipe, or manual)
payout_card (dict) ⋯ (optional)
⋯ (card data) (see below)
⋯ pan_entry string PAN entry method (scan, swipe, or manual)
location (dict)
⋯ id string Identifier for physical customer location, unique within Partner domain
⋯ address string Location address
⋯ lat string Location coordinates, latitude (decimal)
⋯ long string Location coordinates, longitude (decimal)
tos_accepted boolean Indicates that user has accepted Prepaid2Cash Terms of Service
sms_verification_code string SMS verification code sent to user's mobile phone
user_id string User ID from Create User (optional, see notes above)
user_authorization string Signed User Authorization Token, JWT-format (optional, see notes above)

Card Data Fields (Example 1: Direct card data)

Field Type Description
⋯ pan string Full card PAN
⋯ expiry_month string 2-digit, zero-padded, numeric month
⋯ expiry_year string 4-digit year
⋯ cvv string Card Security Code

Card Data Fields (Example 2: Encrypted card data)

Field Type Description
⋯ encryption_alg string Encryption Algorithm (3DES-112-DUKPT or by Partner Agreement)
⋯ encryption_kid string Encryption Key ID (if applicable)
⋯ encryption_ksn string Encryption Key Serial Number (if applicable)
⋯ encrypted_msd_track1 string Magnetic Stripe Data Track 1, encrypted, raw (optional)
⋯ encrypted_msd_track2 string Magnetic Stripe Data Track 2, encrypted, raw (optional)

Response Fields

✅ Response (OK)

201 Created
{
  "transaction_results": "eyJhbGciOiJIUzI1NiIsInR5cCI6I…"
}

❌ Response (validation error)

400 Bad Request
{
  "errors": [
    {
      "code": "validation_error",
      "title": "Validation Error",
      "invalid_params": [
        {
          "name": "first_name",
          "reason": "cannot be empty"
        }
      ]
    }
  ]
}
Field Description
transaction_results Signed Transaction Results Token, JWT-format

Retrieve Closed Loop Merchants

This endpoint returns the active list of gift card merchants which are being purchased. Typical length is 80-120 merchants.

HTTP Request

GET https://api.prepaid2cash.com/v2/closed_loop_card_lines

⏩ Request

GET /v2/closed_loop_card_lines HTTP/1.1
Host: api.prepaid2cash.com
Authorization: pk_test_ImlkX2Zvcm0iOiJkcml

Response Fields

✅ Response

200 OK
[
  {
    "card_line_id": 3,
    "merchant_name": "AMC Theatres",
    "discount_rate": 0.49,
    "logo_url": "https://logo.png"
  },
  {
    "card_line_id": 4,
    "merchant_name": "American Airlines",
    "discount_rate": 0.37,
    "logo_url": "https://logo.png"
  },
  ...
]
Field Description
(array of dict)
⋯ card_line_id ID of merchant card line
⋯ merchant_name Name of Merchant, for display
⋯ discount_rate Estimated card purchase discount rate
⋯ logo_url Merchant logo URL

Create Closed Loop Transaction

This endpoint requests an offer to purchase a Closed Loop card.

It is not necessary to submit a card value -- this will be obtained from the merchant.

HTTP Request

POST https://api.prepaid2cash.com/v2/transactions

Request Body Fields

⏩ Request

POST /v2/transactions HTTP/1.1
Host: api.prepaid2cash.com
Content-Type: application/json
Authorization: pk_test_ImlkX2Zvcm0iOiJkcml

{
  "payout_method": "PARTNER_ACH",
  "prepaid_card": {
    "card_subtype": "closed_loop",
    "card_line_id": 9,
    "pan": "9111111111111102",
    "pin": "2600",
    "pan_entry": "scan"
  },
  "tos_accepted": true,
  "user_authorization": "eyJ0eXAiOiJKV1QiLCJhbGciOiJSUzI1NiIsImtpZCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCNTMjU2Ijoickp2eXZOZGFEOEk1V293bmM3ZnhMMTlQS0NFREsxWGxyMHpYQndhMGxHUSJ9.eyJpc3MiOiJwYXJ0bmVyZG9tYWluLmNvbSIsImF1ZCI6InByZXBhaWQyY2FzaC5jb20iLCJpYXQiOjE1Nzc4MzY4MDAsImV4cCI6MjU3NzgzNjgwMCwiaHR0cDovL3AyYy5pby91c2VyIjp7ImlkIjoiZDU0NzQ3ZDEtMjk5My00YmRjLWE3NDUtMzIxZjdmMzNlOGFmIiwiZmlyc3RfbmFtZSI6Ikpvc2h1YSIsImxhc3RfbmFtZSI6Ik5vcnRvbiIsImFkZHJlc3MxIjoiNjI0IENvbW1lcmNpYWwgU3QiLCJhZGRyZXNzMiI6IiIsImNpdHkiOiJTYW4gRnJhbmNpc2NvIiwic3RhdGUiOiJDQSIsImNvdW50cnkiOiJVUyIsInBvc3RhbF9jb2RlIjoiOTQxMTEtMTAxMCIsInBob25lIjoiKzEgNDE1LTQ1Ni0xMDAwIiwiZW1haWwiOiJub3J0b25Ac2YuY2EudXMiLCJkb2IiOiIxOTc2LTAyLTA0Iiwic3NuX2xhc3RfNCI6IjE3NzYiLCJpZF9mb3JtIjoiZHJpdmVyc19saWNlbnNlIiwiaWRfaXNzdWVyIjoiVVMvQ0EiLCJpZF9udW1iZXIiOiJEMTIzNDU2NzgiLCJreWNfdmVyaWZpZWQiOlsiZmlyc3RfbmFtZSJdfX0.nbywUp4vkQ8uCphzpykrDm7jLHMl8Ixt9DJPIhWQQo7RYbIV6lmr4EwOWCg_TNcjf7l4jNhd33daD1qeCSNxpKMUs45joJlVKFPnjiu32PGNzfOdRw28ElWVCXzQOTR5_OLoRvx6xX_oIOQI9FuMlHyoNq9G6hBpM1zOey74GYunO6WNJPbDjh5bUpZB9L4z_OPs3SgzKzpOuTOLS7tBiiJsgHyS8pex4-_ErTHbrXsennZ8twV5b8xHFC1GtZb3xjm4M1Zs_j1vSGagiHxqp_0VtzMCcKtN31xjS9epk0wDPi4nI_SAkGlvt1XINQ3VNKvPKiA3RlvgDenfqYbize7eZRKFf4W1afODIe1WSXtgoyLa3hbwj_Qf1cqOlnsbRNJw1WVnGX-qdnw-q5MBTZMlgbWJGz65f4YlIW47zjuOHUmDQl3Z5gKFrUGYn6L2CrImnXimIctmbG08H1vcXO75DYnqTlfWaJfRmdXuEEQ64UuRaEG4NotZf9Nq5kq95KlURC344NxraUhyb_1XWc4jDMOmsycfk1H39I77FL-NCH_iiHZ4togZvRicEkgTX7BCwwLOZIpUHaxIOo4k1WKZpTeVI4jUPTm3iX8kPPtsAAAxZF1IhSnGaPvaYjOLO3S0hAPDQfus7nA2-eH-KAQSS19OoS2nYSXyssRYTkc"
}
Field Type Description
payout_method string Selected from transaction_options response
prepaid_card (dict)
⋯ card_subtype string Prepaid Card subtype (closed_loop)
⋯ card_line_id integer (from Retrieve Closed Loop Merchants)
⋯ pan string Full card PAN
⋯ pin string Card PIN (if present)
⋯ pan_entry string PAN entry method (scan, swipe, or manual)
tos_accepted boolean Indicates that user has accepted Prepaid2Cash Terms of Service
user_authorization string Signed User Authorization Token, JWT-format

Response

✅ Response (OK)

201 Created
{
  "transaction": {
    "id": "0479d573-2c4a-4a43-a717-3d2e1cea11e4",
    "payout_amount": 9900
  }
}

❌ Response (validation error)

400 Bad Request
{
  "errors": [
    {
      "code": "validation_error",
      "title": "Validation Error",
      "invalid_params": [
        {
          "name": "first_name",
          "reason": "cannot be empty"
        }
      ]
    }
  ]
}

❌ Response (declined)

422 Unprocessable Entity
{
  "errors": [
    {
      "code": "card_not_valid_for_merchant",
      "title": "Transaction Declined",
      "detail": "Card number or PIN incorrect"
    }
  ]
}
Field Description
transaction (dict)
⋯ id Transaction ID
⋯ payout_amount Amount user will receive, if offer is accepted

Accept Closed Loop Offer

This endpoint accepts an offer to purchase a Closed Loop card.

HTTP Request

POST https://api.prepaid2cash.com/v2/transactions/{id}/accept_offer

Request Body Fields

⏩ Request

POST /v2/transactions/{id}/accept_offer HTTP/1.1
Host: api.prepaid2cash.com
Content-Type: application/json
Authorization: pk_test_ImlkX2Zvcm0iOiJkcml

(none)

Response

✅ Response (OK)

200 OK
{
  "transaction_results": "eyJhbGciOiJIUzI1NiIsInR5cCI6I…"
}
Field Description
transaction_results Signed Transaction Results Token, JWT-format

Decline Closed Loop Offer

This endpoint declines an offer to purchase a Closed Loop card.

HTTP Request

POST https://api.prepaid2cash.com/v2/transactions/{id}/decline_offer

Request Body Fields

⏩ Request

POST /v2/transactions/{id}/decline_offer HTTP/1.1
Host: api.prepaid2cash.com
Content-Type: application/json
Authorization: pk_test_ImlkX2Zvcm0iOiJkcml

(none)

Response Fields

✅ Response (OK)

204 No Content

(none)


Tokens

The Prepaid2Cash Partner API requires the exchange of trusted and verified data, such as user information and transaction results. To ensure the integrity of this data, while allowing it to be sent through potentially-insecure channels, all trusted data is cryptographically signed by the creator, and verified by the consumer.

We use the JSON Web Token (JWT) specification as a serialization format. See JWT.io and RFC7519 for details.

Tokens are cryptographically signed using the algorithm specified in Partner Agreement:

🔵 Production JWKS Endpoint

https://api.prepaid2cash.com/.well-known/jwks.json

🔵 Sandbox JWKS Endpoint

https://sandbox.api.prepaid2cash.com/.well-known/jwks.json

The Prepaid2Cash Partner API uses two kinds of signed tokens:

User Authorization Tokens

⏩ Sample UAT Request

GET /v2/sample_uat HTTP/1.1
Host: sandbox.api.prepaid2cash.com
Authorization: pk_test_ImlkX2Zvcm0iOiJkcml

UATs are generated by Partner (typically by a backend API server) and sent to Prepaid2Cash. They may be delivered directly or via Partner or third-party clients, such as mobile apps.

Prepaid2Cash will verify the signature of all UATs received.

A signed and verified UAT indicates:

UATs may include several fields of user data in the JWT Payload. Required fields will vary by Partner Agreement.

JWT Contents

🔵 User Authorization Token Sample, JSON source

{
  "typ": "JWT",
  "alg": "RS256",
  "kid": "Z6ZFSewD4FYayj8SoWRC_EArx1M",
  "x5t": "Z6ZFSewD4FYayj8SoWRC_EArx1M",
  "x5t#S256": "rJvyvNdaD8I5Wownc7fxL19PKCEDK1Xlr0zXBwa0lGQ"
}
{
  "iss": "partnerdomain.com",
  "aud": "prepaid2cash.com",
  "iat": 1577836800,
  "exp": 2577836800,
  "http://p2c.io/user": {
    "id": "d54747d1-2993-4bdc-a745-321f7f33e8af",
    "first_name": "Joshua",
    "last_name": "Norton",
    "address1": "624 Commercial St",
    "address2": "",
    "city": "San Francisco",
    "state": "CA",
    "country": "US",
    "postal_code": "94111",
    "phone": "+14154561000",
    "email": "norton@sf.ca.us",
    "dob": "1976-02-04",
    "ssn_last_4": "1776",
    "id_form": "drivers_license",
    "id_issuer": "US/CA",
    "id_number": "D12345678",
    "kyc_verified": [
      "first_name"
    ]
  }
}

🔵 User Authorization Token Sample, encoded and signed

eyJ0eXAiOiJKV1QiLCJhbGciOiJSUzI1NiIsImtpZCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCNTMjU2Ijoickp2eXZOZGFEOEk1V293bmM3ZnhMMTlQS0NFREsxWGxyMHpYQndhMGxHUSJ9.eyJpc3MiOiJwYXJ0bmVyZG9tYWluLmNvbSIsImF1ZCI6InByZXBhaWQyY2FzaC5jb20iLCJpYXQiOjE1Nzc4MzY4MDAsImV4cCI6MjU3NzgzNjgwMCwiaHR0cDovL3AyYy5pby91c2VyIjp7ImlkIjoiZDU0NzQ3ZDEtMjk5My00YmRjLWE3NDUtMzIxZjdmMzNlOGFmIiwiZmlyc3RfbmFtZSI6Ikpvc2h1YSIsImxhc3RfbmFtZSI6Ik5vcnRvbiIsImFkZHJlc3MxIjoiNjI0IENvbW1lcmNpYWwgU3QiLCJhZGRyZXNzMiI6IiIsImNpdHkiOiJTYW4gRnJhbmNpc2NvIiwic3RhdGUiOiJDQSIsImNvdW50cnkiOiJVUyIsInBvc3RhbF9jb2RlIjoiOTQxMTEtMTAxMCIsInBob25lIjoiKzEgNDE1LTQ1Ni0xMDAwIiwiZW1haWwiOiJub3J0b25Ac2YuY2EudXMiLCJkb2IiOiIxOTc2LTAyLTA0Iiwic3NuX2xhc3RfNCI6IjE3NzYiLCJpZF9mb3JtIjoiZHJpdmVyc19saWNlbnNlIiwiaWRfaXNzdWVyIjoiVVMvQ0EiLCJpZF9udW1iZXIiOiJEMTIzNDU2NzgiLCJreWNfdmVyaWZpZWQiOlsiZmlyc3RfbmFtZSJdfX0.nbywUp4vkQ8uCphzpykrDm7jLHMl8Ixt9DJPIhWQQo7RYbIV6lmr4EwOWCg_TNcjf7l4jNhd33daD1qeCSNxpKMUs45joJlVKFPnjiu32PGNzfOdRw28ElWVCXzQOTR5_OLoRvx6xX_oIOQI9FuMlHyoNq9G6hBpM1zOey74GYunO6WNJPbDjh5bUpZB9L4z_OPs3SgzKzpOuTOLS7tBiiJsgHyS8pex4-_ErTHbrXsennZ8twV5b8xHFC1GtZb3xjm4M1Zs_j1vSGagiHxqp_0VtzMCcKtN31xjS9epk0wDPi4nI_SAkGlvt1XINQ3VNKvPKiA3RlvgDenfqYbize7eZRKFf4W1afODIe1WSXtgoyLa3hbwj_Qf1cqOlnsbRNJw1WVnGX-qdnw-q5MBTZMlgbWJGz65f4YlIW47zjuOHUmDQl3Z5gKFrUGYn6L2CrImnXimIctmbG08H1vcXO75DYnqTlfWaJfRmdXuEEQ64UuRaEG4NotZf9Nq5kq95KlURC344NxraUhyb_1XWc4jDMOmsycfk1H39I77FL-NCH_iiHZ4togZvRicEkgTX7BCwwLOZIpUHaxIOo4k1WKZpTeVI4jUPTm3iX8kPPtsAAAxZF1IhSnGaPvaYjOLO3S0hAPDQfus7nA2-eH-KAQSS19OoS2nYSXyssRYTkc

Header Fields

Field Type Description
typ string Token type (always JWT)
alg string Signature algorithm (HS256, RS256)
kid string ID of signing key (when alg: RS256)
x5t string SHA1 thumbprint of signing key (when alg: RS256)
x5t#S256 string SHA256 thumbprint of signing key (when alg: RS256)

Payload Fields

Field Type Description
iss string Token issuer identifier (Partner primary domain name)
aud string Token audience identifier (always prepaid2cash.com)
iat integer Token creation time, in UNIX time format (seconds since epoch)
exp integer Token expiration, in UNIX time format (seconds since epoch)
http://p2c.io/user (dict)
⋯ id string User ID, unique in Partner domain
⋯ first_name string
⋯ last_name string
⋯ address1 string
⋯ address2 string
⋯ city string
⋯ state string State, USPS 2-letter format (CA, AL, etc)
⋯ country string Country code, ISO 3166-1 alpha-2 format (always US)
⋯ postal_code string USPS ZIP or ZIP+4 postal code
⋯ phone string Telephone number with country code, ITU-T E.123 format, hyphens optional
⋯ email string Email address
⋯ dob string Date of Birth, ISO 8601 datestamp
⋯ ssn_last_4 string Last 4 digits of SSN
⋯ id_form string Government-Issued ID form
⋯ id_issuer string Government-Issued ID issuer
⋯ id_number string Government-Issued ID number
⋯ kyc_verified array of string List of fields KYC-verified by Partner

Transaction Results Tokens

TRTs are generated by Prepaid2Cash, and sent to Partner. They may be delivered directly or via Partner or third-party clients, such as mobile apps.

A signed and verified TRT indicates:

TRTs will include several fields of transaction data in the JWT Payload.

JWT Contents

🔵 Transaction Results Token Sample, encoded and signed

eyJhbGciOiJSUzI1NiIsImtpZCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsInR5cCI6IkpXVCIsIng1dCI6Ilo2WkZTZXdENEZZYXlqOFNvV1JDX0VBcngxTSIsIng1dCNTMjU2Ijoickp2eXZOZGFEOEk1V293bmM3ZnhMMTlQS0NFREsxWGxyMHpYQndhMGxHUSJ9.eyJpc3MiOiJwcmVwYWlkMmNhc2guY29tIiwiYXVkIjoiYmF5YmFuay5leGFtcGxlLmNvbSIsImlhdCI6MTU3OTIxMzIzMSwiZXhwIjoxNTc5MjE0MTMxLCJodHRwOi8vcDJjLmlvL3RyYW5zYWN0aW9uIjp7ImlkIjoiYzk2NWRhMDkyZCIsInVzZXJfaWQiOiJkNTQ3NDdkMS0yOTkzLTRiZGMtYTc0NS0zMjFmN2YzM2U4YWYiLCJjYXJkX3N1YnR5cGUiOiJvcGVuX2xvb3AiLCJtYXNrZWRfcGFuIjoiNDExMTExKioqKioqMzM0NyIsImV4cGlyYXRpb24iOiIwNy8yMDIyIiwiYW1vdW50IjoxMDAwMCwicGF5b3V0X2Ftb3VudCI6OTQwMCwicGF5b3V0X21ldGhvZCI6IkJBWUJBTktfQUNIIiwiY2FwdHVyZWRfYXQiOiIyMDIwLTAxLTE2VDIyOjIwOjMxWiIsImRlbGl2ZXJ5X2F0IjoiMjAyMC0wMS0xN1QxNzowMDowMFoifX0.XEkoW2pG3mZnLts0Gb6sPExtMJDcsgbB8c-AV6R4YXXJfmmtscFtUW97buPH9s5r_1uFwtBfqXJAXLRYQ2-6hHiqdPXmmOQ1csaU9NXU03lTPqAaZUqVab7W7aKNqTkvSvqwRwx-KjARYXJ1pE6-P9txPz8r9dGeKmC-N4YM6yQFpe0Eo6aE_8JyU87iu1yMuu8fyYozU_GddcrRVNbB4p78hHmxMyYtMVbbJGgROAnIlAJBean8tMnt5crE_5CxxX2fkyl3HcK1oQJVewkoBQR3YmiftClftDYsvjaJTTwp7gOlDBuKcgYl7rIS7lPUQOiCqMSJSoS3MS2-bK3zX2qrwsySNv7Z4zu0jX2_05kEWyI8mm31XJfP1X55woUiegv2NcGk_MhrGle34GaHK2g0CX2_084FuhAPiROwQWEP_UVqo5qgAaTrbf2_vVkhtXgOWbg_hwLmM0Qn42NX3yrE5DSwtPSXaoXTi_-XljLDPttmI8m5En95ZaSImuEhayGpMfOj1bivuU1xDkjF801LcJguBTt_L9nxRaqPm4g0kxXtI13oGMSgUL3-9g8FFXAb_4-rfcbzJyG-TC5NnhvdD1t4_8DpiMZ-FVGoJSF2U8wLK54vIQZhYScuWguKMMxt0ZcHrAHo9RnYRYOtNPjLL3Flh2zdqNu1nnkbeGg

🔵 Transaction Results Token Sample, decoded and verified

{
  "alg": "RS256",
  "kid": "Z6ZFSewD4FYayj8SoWRC_EArx1M",
  "typ": "JWT",
  "x5t": "Z6ZFSewD4FYayj8SoWRC_EArx1M",
  "x5t#S256": "rJvyvNdaD8I5Wownc7fxL19PKCEDK1Xlr0zXBwa0lGQ"
}
{
  "iss": "prepaid2cash.com",
  "aud": "partnerdomain.com",
  "iat": 1579213231,
  "exp": 1579214131,
  "http://p2c.io/transaction": {
    "id": "c965da092d",
    "user_id": "d54747d1-2993-4bdc-a745-321f7f33e8af",
    "card_subtype": "open_loop",
    "masked_pan": "411111******3347",
    "expiration": "07/2022",
    "amount": 10000,
    "payout_amount": 9400,
    "payout_method": "PARTNER_ACH",
    "captured_at": "2020-01-16T22:20:31Z",
    "delivery_at": "2020-01-17T17:00:00Z"
  }
}

Header Fields

Field Type Description
typ string Token type (always JWT)
alg string Signature algorithm (HS256, RS256)
kid string ID of signing key (when alg: RS256)
x5t string SHA1 thumbprint of signing key (when alg: RS256)
x5t#S256 string SHA256 thumbprint of signing key (when alg: RS256)

Payload Fields

Field Type Description
iss string Token issuer identifier (always prepaid2cash.com)
aud string Token audience identifier (Partner primary domain name)
iat integer Token creation time, in UNIX time format (seconds since epoch)
http://p2c.io/transaction (dict)
⋯ id string Unique transaction ID
⋯ user_id string User ID from UAT
⋯ card_subtype string open_loop or closed_loop
⋯ masked_pan string Card PAN, masked for privacy
⋯ expiration string Card expiration
⋯ amount integer Transaction amount (cents)
⋯ payout_amount integer Payout amount (cents)
⋯ payout_method string Payout method from transaction_options
⋯ captured_at string Capture time, ISO 8601 timestamp
⋯ delivery_at string Estimated delivery time, ISO 8601 timestamp

Testing

You can use the following test data to elicit different responses from the Sandbox environment:

Open Loop Test Data

Field and value Result
amount < 2000 Error: Amount out of range
amount = 49001 Error: Card capture failed
amount = 49101 Error: User transaction limits exceeded
amount > 49800 Error: Transaction declined
amount > 50000 Error: Amount out of range

Closed Loop Test Data

Field and value Result
pin = 49001 Error: Card capture failed
pin = 49101 Error: User transaction limits exceeded
pin > 49800 Error: Transaction declined

HTTP Response Codes

The Prepaid2Cash Partner API may return the following HTTP error response codes:

Code Meaning
400 Bad Request Your request was badly-formed, or failed validation. Inspect the response object for details.
401 Unauthorized Your API key is missing or invalid.
403 Forbidden Your Partner account does not have permission to modify this resource.
404 Not Found You have requested an unknown resource.
405 Method Not Allowed You have requested a resource using an invalid method.
406 Not Acceptable You have requested an invalid resource format.
422 Unprocessable Entity Your request ran afoul of application processing. Inspect the response object for details.
429 Too Many Requests You have exceeded request rate limits.
500 Internal Server Error We encountered an error while processing your request. This is a bug!
503 Service Unavailable We are temporarily offline for maintenance. Please try again later.