Displaying Card Data

A common Fintech workflow involves retrieving card data from a third-party service—like a card issuer—and then displaying that data to the end user in a mobile or web app. Without Evervault, this workflow would require the product to handle card data in plaintext, which would drastically increase the PCI DSS scope of the environment.

Evervault offers two primitives to support this workflow while also ensuring that plaintext card data never touches your infrastructure. Adhering to this pattern significantly reduces your PCI DSS scope.

Retrieve Card Data from a Third-Party API

An isometric drawing connecting a credit card to Evervault Inputs

We’re going to use the Relay primitive to retrieve card data from a third-party service. Relay is a network proxy which can be configured to encrypt responses from a third-party API before they reach your server.

First, we’ll set up the proxy, then we’ll configure Relay to encrypt responses from the API.

Create a Relay

We’ll use PutsReq to simulate a third-party API. In practice, this could be an endpoint from a card issuer or payments processor. Selecting Create a PutsReq will provision a temporary endpoint that we can send requests to. In the Response Builder, add the following snippet and select Update.

1
const cardData = {
2
cardNumber: '4242424242424242',
3
cvv: '123',
4
expiry: '02/27',
5
};
6
7
response.headers = { 'Content-Type': 'application/json' };
8
response.body = cardData;

Now that we have our example third-party API, we can create a Relay to proxy requests to it. To create a Relay for your PutsReq endpoint, navigate to the Relays tab in the Evervault Dashboard and click Create Relay and add putsreq.com to the destination field.

A screenshot of the Evervault Dashboard showing a Relay being created with a RequestBin endpoint as its destination

Integrate the Proxy

To use Relay, first include and initialize the Evervault Node.js SDK in your application and enable Outbound Relay.

1
const Evervault = require('@evervault/sdk');
2
const evervault = new Evervault('<APP_ID>', '<API_KEY>');
3
evervault.enableOutboundRelay();

Then, using an HTTP client, we can send a request to the PutsReq endpoint. The Evervault SDK will automatically intercept the request and route it through Relay. Since Relay intercepts HTTP requests, the proxy will work regardless of whether you’re sending a request directly to a REST endpoint, or using a third-party SDK.

1
const Evervault = require('@evervault/sdk');
2
const axios = require('axios');
3
const evervault = new Evervault('<APP_ID>', '<API_KEY>');
4
5
evervault.enableOutboundRelay();
6
7
const PUTSREQ_URL = '<PUTSREQ_URL>';
8
9
(async () => {
10
const result = await axios.get(PUTSREQ_URL);
11
console.log(result.data);
12
})();

If you run the snippet, You’ll notice that the response still includes plaintext values for cardNumber, expiry and cvv. Next, we’ll configure Relay to encrypt these fields in the response.

Configure the Proxy

Relay can be configured to encrypt data in requests and responses. Since our endpoint returns plaintext card data, we’re going to configure the proxy to encrypt the response. Doing so allows us to ensure that plaintext card data never touches our server. In the Evervault dashboard, navigate to the Relay you created, and click the Add Route button to configure a new route. We want to encrypt data that is being returned from our PutsReq API, so we can enter our PutsReq path in the path field (e.g '/PYKrs01THWXghka2AYap').

Next, we can add a response action to encrypt fields we want to secure in the response body. Select Add Response Action -> Encrypt -> JSON, and enter $.cardNumber, $.expiry, and $.cvv in the fields to encrypt. JSON requests use JSONPath to match fields in the request body. Learn More

A screenshot of the Evervault Dashboard where a Relay is being configured to encrypt response fields

If you run the above code snippet again, you’ll notice that the cardNumber, expiry and cvv fields in the response have been encrypted. This architecture means you can now retrieve and encrypt card data from a third-party service without handling it in plaintext on your server — drastically reducing your PCI Compliance scope.

Next, we’ll show you how to display encrypted card data to our users in plaintext without compromising your compliance scope.

Display Card Data

An isometric drawing connecting a credit card connected to Evervault Reveal

Displaying plaintext, or unencrypted, card data to end users is a common pattern in Fintech products. Doing this without Evervault would require you to handle plaintext card data in your mobile or web app, which would significantly increase your PCI DSS scope. We can reduce this scope by using the Evervault Reveal UI Component.

Integrate Reveal

Reveal is a secure iframe, hosted by Evervault, which can be used to display plaintext card data.

Before getting started with Reveal, you’ll need an endpoint that returns the encrypted card data. This endpoint can then be proxied via Evervault’s Relay primitive to decrypt the card data in the response. The Reveal Component requires a request parameter, which receives a Request object pointing at the Relay URL, along with the Authentication headers required to access the endpoint.

For this example, lets assume this endpoint returns the card data in the following JSON:

1
{
2
"cardNumber": "ev:debug:Tk9D:number:mqrANWE1KMapRVCX:A8Gse2iNxLKy8WLnBDkqtkt+TTSaJFNyc+R/toYd2d8n:FUM5vwjOIrHI+3Jk7QnVCUpHVS0aZQC80ih/wpHtR2o=:$",
3
"cvv": "ev:debug:Tk9D:number:YG1KlyImadEc/qJz:A7odbz+iMRub/WHEO9bwLI5tig+nJ12aKCfWHtMU87rJ:/g1rhC8GeaGyWbM1Y0kaT45IKw==:$",
4
"expiry": "09/28"
5
}

Make sure that the endpoint you create sets appropriate CORS headers to ensure that it can be accessed by the Reveal iframe; otherwise, the request might fail.

To use Reveal, you first need to install the Evervault React.js SDK.

# Using npm
npm install @evervault/react
# Using yarn
yarn add @evervault/react

You can initialize the React.js SDK by passing your TeamID and App ID to the <EvervaultProvider/>. The Reveal Comnponent can be rendered by using <Reveal/> in children of the provider. After passing the constructed request as a prop, you can use the <Reveal.Text /> component to select which field from the response payload to display.

1
import { EvervaultProvider, Reveal } from '@evervault/react';
2
3
export default function App() {
4
return (
5
<EvervaultProvider teamId="<TEAM_ID>" appId="<APP_ID>">
6
<ChildComponent />
7
</EvervaultProvider>
8
);
9
}
10
11
function ChildComponent() {
12
const request = useMemo(() => {
13
return new Request("https://example-com.app-12345.relay.evervault.app/card-data", {
14
method: "GET",
15
headers: {
16
"Content-Type": "application/json"
17
}
18
})
19
}, [])
20
21
return (
22
<Request request={request}>
23
<Request.Text path="$.cardNumber" />
24
<Request.CopyButton path="$.cardNumber" text="Copy Number" />
25
</Request>
26
)
27
}

Reveal accepts some additional parameters for error handling, loading states and styling. You can find a detailed description of each parameter in the reference documentation for React SDK.

Constructing your workflow with this pattern will allow you to render card data in your mobile or web app while also minimizing your PCI Compliance scope.