Near Field Communication (NFC) tags can be cards, phones, bracelets, or similar. They can store data and are commonly used for contactless payments, loyalty programs, festival entrance, and more.
By sending a Terminal API card acquisition request to your Adyen payment terminal, you can identify the type of NFC tag, or read and write data to it.
For more complex use cases, you can create a session to present multiple requests as a single user interaction. For example, in a single session, you could identify the NFC tag, read its balance, and write the new balance on the tag.
To understand how you can use this, here are some examples:
- Enroll shoppers in a loyalty program.
- Identify a customer when entering an area such as festival grounds or a parking garage.
- Check the balance and top up NFC tags such as loyalty and transport cards, or festival tags used to pay for drinks.
- Define multiple behaviours of the NFC tag. For example, use the tag to enter one event and reuse it to pay for drinks at a different event.
Supported products
The following NFC tags can be used on Adyen payment terminals:
- MIFARE Classic
- MIFARE DESFire
- MIFARE Ultralight, Ultralight C, and Ultralight AES
NFC tags are not supported on UX300 and UX410 payment terminals.
How it works
To use NFC tags on Adyen payment terminals, first you need to contact our Support Team to create a configuration for your use case. Usually, this includes:
- The type of NFC tag you want to use, for example MIFARE Classic.
- For NFC tags that have keys, you need to supply the keys you want to use to access data.
After the configuration is created:
- Make a card acquisition request to identify, read, or write to an NFC tag.
For more complex use cases that require multiple requests, see Create a session.
- The terminal prompts the user to present their NFC tag or NFC-enabled card.
- The user presents their NFC tag by tapping, inserting, or swiping.
- The terminal makes one or multiple read/write requests to the NFC tag.
While this is going on, the terminal shows the loading screen: One moment or Keep card inserted if the customer inserted their NFC-enabled card into the terminal.
Identify the type of NFC tag
You can use the card acquisition request to get the unique identifier and the type of the NFC tag. You can use this identifier, for example, to check if the NFC tag exists in your system or if it is still valid.
-
Create an
Operation
JSON object withOperation.Type
: NFCReadUID:Operation JSON objectExpand viewCopy link to code blockCopy code{ "Operation":[ { "Type":"NFCReadUID" } ] } -
Encode the
Operation
JSON object to Base64. You will pass the resulting string inSaleData.SaleToPOIData
.Converted to a Base64-encoded stringExpand viewCopy link to code blockCopy codeewogICAgIk9wZXJhdGlvbiI6WwogICAgICAgIHsKICAgICAgICAgICAgIlR5cGUiOiJORkNSZWFkVUlEIgogICAgICAgIH0KICAgIF0KfQ==
-
Make a card acquisition request to a Terminal API endpoint, specifying:
-
MessageHeader
: the standardSaleToPOIRequest.MessageHeader
object. Specify:Parameter Required Description ProtocolVersion
3.0 MessageClass
Service MessageCategory
CardAcquisition MessageType
Request ServiceID
Your unique ID for this request, consisting of 1-10 alphanumeric characters. Must be unique within the last 48 hours for the terminal ( POIID
) being used.SaleID
Your unique ID for the POS system component to send this request from. POIID
The unique ID of the terminal to send this request to. Format: [device model]-[serial number].
-
CardAcquisitionRequest.SaleData
:Parameter Required Description SaleTransactionID
An object with: TransactionID
: your reference to identify the transaction. We recommend using a unique value.TimeStamp
: date and time of the request in UTC format.
SaleToPOIData
The Base64-encoded Operation
JSON object. -
CardAcquisitionRequest.CardAcquisitionTransaction
:Parameter Required Description TotalAmount
The transaction amount. You can omit the TotalAmount
field when getting theUID
of the NFC tag.If you omitTotalAmount
, you still have to include an emptyCardAcquisitionTransaction
field in the request.
The example below shows a card acquisition request to identify the NFC tag:
Identify NFC tag requestExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"CardAcquisition", "MessageType":"Request", "ServiceID":"1020711110", "SaleID":"POSSystemID12345", "POIID":"V400m-346403161" }, "CardAcquisitionRequest":{ "SaleData":{ "SaleTransactionID":{ "TransactionID":"02072", "TimeStamp":"2020-01-07T14:14:04+00:00" }, "SaleToPOIData":"ewogICAgIk9wZXJhdGlvbiI6WwogICAgICAgIHsKICAgICAgICAgICAgIlR5cGUiOiJORkNSZWFkVUlEIgogICAgICAgIH0KICAgIF0KfQ==" }, "CardAcquisitionTransaction":{ } } } } -
-
In the
AdditionalResponse
of theCardAcquisitionResponse
note:NFC.uid
: the unique identifier of the NFC tag.NFC.variant
: the type of the NFC tag, for example mf_classic for MIFARE Classic.
Identify NFC tag responseExpand viewCopy link to code blockCopy code{ "SaleToPOIResponse":{ "CardAcquisitionResponse":{ "POIData":{ "POIReconciliationID": "1000", "POITransactionID":{ "TimeStamp": "2022-10-10T12:50:31.000Z", "TransactionID": "CglQ001665406231000" } }, "PaymentInstrumentData":{ "CardData": {}, "PaymentInstrumentType": "Card" }, "Response":{ "AdditionalResponse": "...NFC.uid=96924E37...NFC.variant=mf_classic...", "Result": "Success" }, "SaleData":{ "SaleTransactionID":{ "TimeStamp": "2022-10-03T10:56:42.198Z", "TransactionID": "729" } } }, "MessageHeader": {...} } } After you have identified the NFC tag, the loading screen continues to show until you send an enable service request` (see the next step).
-
To stop the loading screen, make a POST request to a Terminal API endpoint, specifying:
-
MessageHeader
: the standardSaleToPOIRequest.MessageHeader
object. Specify:Parameter Required Description ProtocolVersion
3.0 MessageClass
Service MessageCategory
EnableService MessageType
Request ServiceID
Your unique ID for this request, consisting of 1-10 alphanumeric characters. Must be unique within the last 48 hours for the terminal ( POIID
) being used.SaleID
Your unique ID for the POS system component to send this request from. POIID
The unique ID of the terminal to send this request to. Format: [device model]-[serial number].
-
EnableServiceRequest with:
Parameter Required Description TransactionAction
AbortTransaction. DisplayOutput
Optional object to show your own message and an 'Approved' icon (green check mark) or a 'Declined' icon (red cross). If you omitDisplayOutput
, the terminal shows Canceled, a red cross , and Transaction canceled.
The following example is the basic request, without the
DisplayOutput
object, to stop the card acquisition flow.Basic request to cancel the card acquisition dataExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"EnableService", "MessageType":"Request", "ServiceID":"3020711110", "SaleID":"POSSystemID12345", "POIID":"V400m-346403161" }, "EnableServiceRequest":{ "TransactionAction":"AbortTransaction" } } } -
Read data
NFC tags have multiple parts that can store data. Because the data structure is different for every NFC tag type, the request needs to specify where to read the data from. For some tags, you need to include the key to read data from a sector
, aid
, orpage
.
Select a tab to see the parameters that you need to specify to read data from MIFARE Classic, MIFARE DESFire, or MIFARE Ultralight.
-
Create an
Operation
JSON object and specify:Parameter Required Description Type
NFCRead: reads data from the specified sector. Variant
MifareClassic: the type of NFC tag. NFCData
An array with:If you leave theNFCData
array empty, the terminal will try to read all sectors.sector
: specifies what sector on MifareClassic to read from.keyType
: specifies what key is used to access a sector, a (default) or b.You cannot usekeyType
: b on Android terminals due to a limitation from Castles.keyId
: specifies what unique key for the specifiedkeyType
to use to access a sector. Can be any positive numerical value.
Structure of Operation object for read data request - MIFARE ClassicExpand viewCopy link to code blockCopy code{ "Operation":[ { "Type":"NFCRead", "Variant":"MifareClassic", "NFCData":[ { "sector":1, "keyType":"a", "keyId":1 }, { "sector":2, "keyType":"b", "keyId": 2 } ] } ] } -
Encode the
Operation
JSON object to Base64. You will pass the resulting string inSaleData.SaleToPOIData
.Converted to a Base64-encoded stringExpand viewCopy link to code blockCopy codeewogICAiT3BlcmF0aW9uIjpbCiAgICAgIHsKICAgICAgICAgIlR5cGUiOiJORkNSZWFkIiwKICAgICAgICAgIlZhcmlhbnQiOiJNaWZhcmVDbGFzc2ljIiwKICAgICAgICAgIk5GQ0RhdGEiOlsKICAgICAgICAgICAgewogICAgICAgICAgICAgICAic2VjdG9yIjoxLAogICAgICAgICAgICAgICAia2V5VHlwZSI6ImEiLAogICAgICAgICAgICAgICAia2V5SWQiOjEKICAgICAgICAgICAgfSwKICAgICAgICAgICAgewogICAgICAgICAgICAgICAic2VjdG9yIjoyLAogICAgICAgICAgICAgICAia2V5VHlwZSI6ImIiLAogICAgICAgICAgICAgICAia2V5SWQiOjIKICAgICAgICAgICAgfQogICAgICAgICBdCiAgICAgIH0KICAgXQp9
-
Make a card acquisition request to a Terminal API endpoint, specifying:
-
MessageHeader
: the standardSaleToPOIRequest.MessageHeader
object. This includes:Parameter Required Description MessageClass
Service MessageCategory
CardAcquisition
-
CardAcquisitionRequest.SaleData
:Parameter Required Description SaleTransactionID
An object with: TransactionID
: your reference to identify the transaction. We recommend using a unique value.TimeStamp
: date and time of the request in UTC format.
SaleToPOIData
The Base64-encoded Operation
JSON object. -
CardAcquisitionRequest.CardAcquisitionTransaction
:Parameter Required Description TotalAmount
The transaction amount. You can omit the TotalAmount
field when reading the NFC tag.If you omitTotalAmount
, you still have to include an emptyCardAcquisitionTransaction
field in the request.
The example below shows a card acquisition request to read data from sectors 1 and 2 on a MifareClassic NFC tag:
Read data request - MIFARE ClassicExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"CardAcquisition", "MessageType":"Request", "ServiceID":"282", "SaleID":"POSSystemID12345", "POIID":"V400m-346403161" }, "CardAcquisitionRequest":{ "SaleData":{ "SaleTransactionID":{ "TransactionID":"369", "TimeStamp":"2021-03-05T12:22:57.449Z" }, "SaleToPOIData":"ewogICAiT3BlcmF0aW9uIjpbCiAgICAgIHsKICAgICAgICAgIlR5cGUiOiJORkNSZWFkIiwKICAgICAgICAgIlZhcmlhbnQiOiJNaWZhcmVDbGFzc2ljIiwKICAgICAgICAgIk5GQ0RhdGEiOlsKICAgICAgICAgICAgewogICAgICAgICAgICAgICAic2VjdG9yIjoxLAogICAgICAgICAgICAgICAia2V5VHlwZSI6ImEiLAogICAgICAgICAgICAgICAia2V5SWQiOjEKICAgICAgICAgICAgfSwKICAgICAgICAgICAgewogICAgICAgICAgICAgICAic2VjdG9yIjoyLAogICAgICAgICAgICAgICAia2V5VHlwZSI6ImIiLAogICAgICAgICAgICAgICAia2V5SWQiOjIKICAgICAgICAgICAgfQogICAgICAgICBdCiAgICAgIH0KICAgXQp9" }, "CardAcquisitionTransaction":{ } } } } -
-
When the card acquisition succeeds, in the
AdditionalResponse
of theCardAcquisitionResponse
note:NFC.data
: the data from the specified sectors.- For a single sector, the data shows as NFC.data=afafafafafafafafafa....
- For multiple sectors, the data from each sector shows as value of the
NFC.data
for that sector, for example: NFC.data.S01=afafafafafafafafafa, NFC.data.S02=2222222222222222.
NFC.status
: the status of the read action for everysector
, for example NFC.status=s01.OKs02.OKS03.AUTH_ERROR.NFC.uid
: the unique identifier of the NFC tag.NFC.variant
: the type of the NFC tag, in this case MIFARE Classic.
Read data response - MIFARE ClassicExpand viewCopy link to code blockCopy code{ "SaleToPOIResponse":{ "CardAcquisitionResponse":{ "POIData":{ "POIReconciliationID":"1000", "POITransactionID":{ "TimeStamp":"2022-10-17T11:35:04.000Z", "TransactionID":"CglQ001666006504012" } }, "PaymentInstrumentData":{ "CardData":{ }, "PaymentInstrumentType":"Card" }, "Response":{ "AdditionalResponse":"tid=47353332&transactionType=GOODS_SERVICES&giftcardIndicator=false&posAmountGratuityValue=0&store=YourStore&iso8601TxDate=2022-10-17T11%3a35%3a04.733Z&posOriginalAmountValue=0&txtime=13%3a35%3a04&NFC.data.S01=AFAFAFAFAFAFAFAF&NFC.uid=96924E37&txdate=17-10-2022&NFC.data.S02=2222222222222222&merchantReference=868&posadditionalamounts.originalAmountCurrency=EUR&NFC.ref=mifareClassicCard&posAuthAmountCurrency=EUR&message=CARD_ACQ_COMPLETED&posAmountCashbackValue=0&NFC.variant=mf_classic&posEntryMode=CLESS_SWIPE&NFC.status=S01.OKS02.OK&posAuthAmountValue=0", "Result":"Success" }, "SaleData":{ "SaleTransactionID":{ "TimeStamp":"2022-10-03T10:56:42.966Z", "TransactionID":"868" } } }, "MessageHeader":{ "MessageCategory":"CardAcquisition", "MessageClass":"Service", "MessageType":"Response", "POIID":"V400m-346403161", "ProtocolVersion":"3.0", "SaleID":"POSSystemID12345", "ServiceID":"40" } } } After you have read the specified sectors on the NFC tag, the loading screen continues to show until you send an
enable service request` (see the next step). -
To stop the loading screen, make an enable service request to a Terminal API endpoint, specifying:
-
MessageHeader
: the standardSaleToPOIRequest.MessageHeader
object. This includes:Parameter Required Description MessageClass
Service MessageCategory
EnableService -
EnableServiceRequest with:
Parameter Required Description TransactionAction
AbortTransaction. DisplayOutput
Optional object to show your own message and an 'Approved' icon (green check mark) or a 'Declined' icon (red cross). If you omitDisplayOutput
, the terminal shows Canceled, a red cross , and Transaction canceled.
The following example is the basic request, without the
DisplayOutput
object, to stop the card acquisition flow.Basic request to cancel the card acquisition dataExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"EnableService", "MessageType":"Request", "ServiceID":"3020711110", "SaleID":"POSSystemID12345", "POIID":"V400m-346403161" }, "EnableServiceRequest":{ "TransactionAction":"AbortTransaction" } } } -
Write data
NFC tags have multiple parts that can store data. Because the data structure is different for every NFC tag type, the request needs to specify the sector
, aid
, or page
where to write the data to.
Select a tab to see the parameters that you need to specify to write data to MIFARE Classic, MIFARE DESFire, or MIFARE Ultralight.
-
Create an
Operation
JSON object and specify:Parameter Required Description Type
NFCWrite: writes data to the specified sector. Variant
MifareClassic: specifies the type of NFC tag used. NFCData
An array specifying:If you leave theNFCData
array empty when trying to write, the response will return theUID
and thevariant
but will not write to the card.sector
: specifies the sector to write to.keyType
: specifies whatkey
is used to write to a sector, a (default) or b.You cannot usekeyType
: b on Android terminals due to a limitation from Castles.keyId
: specifies what unique key for the specifiedkeyType
to use to access a sector. Can be any positive numerical value.data
: data to write in the specified sector.
ExpectedUid
Specifies the UID
of the NFC tag to which you want to write the data. If theUID
of the presented NFC tag is different from theExpectedUid
, the request fails andWRONG_CARD
is returned in theAdditionalResponse
. If passed without a value, the parameter is ignored.Structure of Operation object for write data request - Mifare ClassicExpand viewCopy link to code blockCopy code{ "Operation":[ { "Type":"NFCWrite", "Variant":"MifareClassic", "NFCData":[ { "sector":1, "data":"afafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafaf" }, { "sector":2, "data":"afafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafafaf", "keyType":"b", "keyId":1 } ] } ] } -
Encode the
Operation
JSON object to Base64. You will pass the resulting string inSaleData.SaleToPOIData
.Converted to a Base64-encoded stringExpand viewCopy link to code blockCopy codeewogICAiT3BlcmF0aW9uIjpbCiAgICAgIHsKICAgICAgICAgIlR5cGUiOiJORkNXcml0ZSIsCiAgICAgICAgICJWYXJpYW50IjoiTWlmYXJlQ2xhc3NpYyIsCiAgICAgICAgICJORkNEYXRhIjpbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgInNlY3RvciI6MSwKICAgICAgICAgICAgICAgImRhdGEiOiJhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWYiCiAgICAgICAgICAgIH0sCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgInNlY3RvciI6MiwKICAgICAgICAgICAgICAgImRhdGEiOiJhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWYiLAogICAgICAgICAgICAgICAia2V5VHlwZSI6ImIiLAogICAgICAgICAgICAgICAia2V5SWQiOjEKICAgICAgICAgICAgfQogICAgICAgICBdCiAgICAgIH0KICAgXQp9
-
Make a card acquisition request to a Terminal API endpoint, specifying:
-
MessageHeader
: the standardSaleToPOIRequest.MessageHeader
object. This includes:Parameter Required Description MessageClass
Service MessageCategory
CardAcquisition
-
CardAcquisitionRequest.SaleData
:Parameter Required Description SaleTransactionID
An object with: TransactionID
: your reference to identify the transaction. We recommend using a unique value.TimeStamp
: date and time of the request in UTC format.
SaleToPOIData
The Base64-encoded Operation
JSON object. -
CardAcquisitionRequest.CardAcquisitionTransaction
:Parameter Required Description TotalAmount
The transaction amount. You can omit the TotalAmount
field when writing to the NFC tag.If you omitTotalAmount
, you still have to include an emptyCardAcquisitionTransaction
field in the request.
The example below shows a card acquisition request to write data to sectors 1 and 2 on a MifareClassic NFC tag:
Write data request - MIFARE ClassicExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"CardAcquisition", "MessageType":"Request", "ServiceID":"283", "SaleID":"POSSystemID12345", "POIID":"V400m-346403161" }, "CardAcquisitionRequest":{ "SaleData":{ "SaleTransactionID":{ "TransactionID":"369", "TimeStamp":"2021-03-05T12:22:57.449Z" }, "TokenRequestedType":"Customer", "SaleToPOIData":"ewogICAiT3BlcmF0aW9uIjpbCiAgICAgIHsKICAgICAgICAgIlR5cGUiOiJORkNXcml0ZSIsCiAgICAgICAgICJWYXJpYW50IjoiTWlmYXJlQ2xhc3NpYyIsCiAgICAgICAgICJORkNEYXRhIjpbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgInNlY3RvciI6MSwKICAgICAgICAgICAgICAgImRhdGEiOiJhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWYiCiAgICAgICAgICAgIH0sCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgInNlY3RvciI6MiwKICAgICAgICAgICAgICAgImRhdGEiOiJhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWZhZmFmYWYiLAogICAgICAgICAgICAgICAia2V5VHlwZSI6ImIiLAogICAgICAgICAgICAgICAia2V5SWQiOjEKICAgICAgICAgICAgfQogICAgICAgICBdCiAgICAgIH0KICAgXQp9" }, "CardAcquisitionTransaction":{ } } } } -
-
When the card acquisition succeeds, in the
AdditionalResponse
of theCardAcquisitionResponse
note:NFC.data
: the data written to the specified sectors.- For a single sector, the data shows as NFC.data=afafafafafafafafafa....
- For multiple sectors, the data from each sector shows as value of the
NFC.data
for that sector, for example: NFC.data.S01=afafafafafafafafafa, NFC.data.S02=2222222222222222.
NFC.status
: the status of the write action for everysector
, for example NFC.status=s01.OKs02.OKS03.AUTH_ERROR.NFC.uid
: the unique identifier of the NFC tag.NFC.variant
: the type of the NFC tag, in this case MIFARE Classic.
Write data response - MIFARE ClassicExpand viewCopy link to code blockCopy code{ "SaleToPOIResponse":{ "CardAcquisitionResponse":{ "POIData":{ "POIReconciliationID":"1000", "POITransactionID":{ "TimeStamp":"2022-10-18T08:17:51.000Z", "TransactionID":"CglQ001666081071002" } }, "PaymentInstrumentData":{ "CardData":{ }, "PaymentInstrumentType":"Card" }, "Response":{ "AdditionalResponse":"tid=47353332&transactionType=GOODS_SERVICES&giftcardIndicator=false&posAmountGratuityValue=0&store=YourStore&iso8601TxDate=2022-10-18T08%3a17%3a51.591Z&posOriginalAmountValue=0&txtime=10%3a17%3a51&NFC.uid=96924E37&txdate=18-10-2022&merchantReference=889&posadditionalamounts.originalAmountCurrency=EUR&NFC.ref=mifareClassicCard&posAuthAmountCurrency=EUR&message=CARD_ACQ_COMPLETED&NFC.data.1=AFAFAFAFAFAFAFAF&posAmountCashbackValue=0&NFC.data.2=AFAFAFAFAFAFAFAF&NFC.variant=mf_classic&posEntryMode=CLESS_SWIPE&NFC.status=S01.OKS02.OK&posAuthAmountValue=0", "Result":"Success" }, "SaleData":{ "SaleTransactionID":{ "TimeStamp":"2022-10-03T10:56:42.966Z", "TransactionID":"889" } } }, "MessageHeader":{ "MessageCategory":"CardAcquisition", "MessageClass":"Service", "MessageType":"Response", "POIID":"V400m-346403161", "ProtocolVersion":"3.0", "SaleID":"POSSystemID12345", "ServiceID":"833" } } } After you write to the specified sectors on the NFC tag, the loading screen continues to show until you send an
enable service request (see the next step). -
To stop the loading screen, make an enable service request to a Terminal API endpoint, specifying:
-
MessageHeader
: the standardSaleToPOIRequest.MessageHeader
object. This includes:Parameter Required Description MessageClass
Service MessageCategory
EnableService -
EnableServiceRequest with:
Parameter Required Description TransactionAction
AbortTransaction. DisplayOutput
Optional object to show your own message and an 'Approved' icon (green check mark) or a 'Declined' icon (red cross). If you omitDisplayOutput
, the terminal shows Canceled, a red cross , and Transaction canceled.
The following example is the basic request, without the
DisplayOutput
object, to stop the card acquisition flow.Basic request to cancel the card acquisition dataExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"EnableService", "MessageType":"Request", "ServiceID":"3020711110", "SaleID":"POSSystemID12345", "POIID":"V400m-346403161" }, "EnableServiceRequest":{ "TransactionAction":"AbortTransaction" } } } -
Create a session
If your use case requires multiple read/write actions on an NFC tag, you can create a session to make multiple requests part of the same user interaction. While the session is in progress, the terminal shows the One moment screen.
To start the session you need to add Session.Type
: Begin in the SaleToPOIData
of your first identify, read, or write request. You then send multiple requests and, to finish the session, you add Session.Type
: End in the SaleToPOIData
of the last request. All requests that belong to the session must contain the same Session.Id
in the SaleToPOIData
.
When the session ends, the terminal continues to show the loading screen until you send an enable service request.
Here's an example of a session that consists of three card acquisition requests:
Step 1: Start a session and identify the NFC tag type
When starting the session, you need specify the Id
to connect the requests into a single session. Here's example of how identify the tag and start the session:
-
Create a JSON object with:
Parameter Required Description Session.Type
Begin: starts the session. Session.Id
Your unique reference of the session. Session.Timeout
- How long the One moment screen is shown on the terminal display, in milliseconds.
- If there is no end session request, how long before the session data is deleted and new requests can be sent.
Operation.Type
NFCReadUID: returns the NFC tag identifier and type. JSON objectExpand viewCopy link to code blockCopy code{ "Session": { "Type": "Begin", "Id": "50", "TimeoutMs": "60000" }, "Operation":[ { "Type":"NFCReadUID" } ] } -
Encode the JSON object to Base64. You will pass the resulting string in
SaleData.SaleToPOIData
.Converted to a Base64-encoded stringExpand viewCopy link to code blockCopy codeewogICAgIlNlc3Npb24iOnsKICAgICAgICAiVHlwZSI6IkJlZ2luIiwKICAgICAgICAiSWQiOiI1MCIsCiAgICAgICAgIlRpbWVvdXRNcyI6IjYwMDAwIgogICAgfSwKICAgICJPcGVyYXRpb24iOlsKICAgICAgICB7CiAgICAgICAgICAgICJUeXBlIjoiTkZDUmVhZFVJRCIKICAgICAgICB9CiAgICBdCn0=
-
Make a card acquisition request with the Base64-encoded string in the
SaleToPOIData
.Start a session and identify the NFC tag typeExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"CardAcquisition", "MessageType":"Request", "ServiceID":"1020711110", "SaleID":"POSSystemID12345", "POIID":"V400m-346403161" }, "CardAcquisitionRequest":{ "SaleData":{ "SaleTransactionID":{ "TransactionID":"02072", "TimeStamp":"2020-01-07T14:14:04+00:00" }, "SaleToPOIData":"ewogICAgIlNlc3Npb24iOnsKICAgICAgICAiVHlwZSI6IkJlZ2luIiwKICAgICAgICAiSWQiOiI1MCIsCiAgICAgICAgIlRpbWVvdXRNcyI6IjYwMDAwIgogICAgfSwKICAgICJPcGVyYXRpb24iOlsKICAgICAgICB7CiAgICAgICAgICAgICJUeXBlIjoiTkZDUmVhZFVJRCIKICAgICAgICB9CiAgICBdCn0=" }, "CardAcquisitionTransaction":{ } } } }
Step 2: Send read data request with Session.Id
Make sure to include the Session.Id
specified in the SaleToPOIData
of the first request. Here's example of how to read data from MIFARE Classic as part of the session:
-
Create a JSON object with:
Parameter Required Description Session.Id
Your unique reference of the session, specified in the first request. Operation.Type
NFCRead: reads data from the specified sector. Operation.Variant
MifareClassic: the type of NFC tag. Operation.NFCData
An array with:If you leave theNFCData
array empty, the terminal will try to read all sectors.sector
: specifies what sector on MifareClassic to read from.keyType
: specifies what key is used to access a sector, a (default) or b.You cannot usekeyType
: b on Android terminals due to a limitation from Castles.
JSON objectExpand viewCopy link to code blockCopy code{ "Session":{ "Id":"50" }, "Operation":[ { "Type":"NFCRead", "Variant":"MifareClassic", "NFCData":[ { "sector":1, "keyType":"a" } ] } ] } -
Encode the JSON object to Base64. You will pass the resulting string in
SaleData.SaleToPOIData
.Converted to a Base64-encoded stringExpand viewCopy link to code blockCopy codeewogICAgIlNlc3Npb24iOnsKICAgICAgICAiSWQiOiI1MCIKICAgIH0sCiAgICAiT3BlcmF0aW9uIjpbCiAgICAgICAgewogICAgICAgICAgICAiVHlwZSI6Ik5GQ1JlYWQiLAogICAgICAgICAgICAiVmFyaWFudCI6Ik1pZmFyZUNsYXNzaWMiLAogICAgICAgICAgICAiTkZDRGF0YSI6WwogICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgICJzZWN0b3IiOjEsCiAgICAgICAgICAgICAgICAgICAgImtleVR5cGUiOiJhIgogICAgICAgICAgICAgICAgfQogICAgICAgICAgICBdCiAgICAgICAgfQogICAgXQp9
-
Make a card acquisition request with the Base64-encoded string in the
SaleToPOIData
.
The example below shows how to read data from MifareClassic NFC tag as part of the session:Read data request with Sessions.IdExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"CardAcquisition", "MessageType":"Request", "ServiceID":"282", "SaleID":"POSSystemID12345", "POIID":"V400m-346403161" }, "CardAcquisitionRequest":{ "SaleData":{ "SaleTransactionID":{ "TransactionID":"369", "TimeStamp":"2021-03-05T12:22:57.449Z" }, "TokenRequestedType":"Customer" }, "SaleToPOIData":"ewogICAgIlNlc3Npb24iOnsKICAgICAgICAiSWQiOiI1MCIKICAgIH0sCiAgICAiT3BlcmF0aW9uIjpbCiAgICAgICAgewogICAgICAgICAgICAiVHlwZSI6Ik5GQ1JlYWQiLAogICAgICAgICAgICAiVmFyaWFudCI6Ik1pZmFyZUNsYXNzaWMiLAogICAgICAgICAgICAiTkZDRGF0YSI6WwogICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgICJzZWN0b3IiOjEsCiAgICAgICAgICAgICAgICAgICAgImtleVR5cGUiOiJhIgogICAgICAgICAgICAgICAgfQogICAgICAgICAgICBdCiAgICAgICAgfQogICAgXQp9", "CardAcquisitionTransaction":{ } } } }
Step 3: Write data to the NFC tag and end the session
In the final request of the interaction, make sure to include the same Session.Id
and Session.Type
: End. Here's example of how to write data to MIFARE Classic and end the session:
-
Create a JSON object with:
Parameter Required Description Session.Type
End: ends the session. Session.Id
Your unique reference of the session. Operation.Type
NFCWrite: reads data from the specified sector. Operation.Variant
MifareClassic: the type of NFC tag. Operation.NFCData
An array with:If you leave theNFCData
array empty when trying to write, the response will return theUID
and thevariant
but will not write to the card.sector
: specifies what sector on MifareClassic to read from.keyType
: specifies what key is used to access a sector, a (default) or b.You cannot usekeyType
: b on Android terminals due to a limitation from Castles.
JSON objectExpand viewCopy link to code blockCopy code{ "Session":{ "Type": "End", "Id":"50" }, "Operation":[ { "Type": "NFCWrite", "Variant": "MifareClassic", "NFCData":[ { "sector": 1, "data":"..." } ] } ] } -
Encode the JSON object to Base64. You will pass the resulting string in
SaleData.SaleToPOIData
.Converted to a Base64-encoded stringExpand viewCopy link to code blockCopy codeewogICAgIlNlc3Npb24iOnsKICAgICAgICAiVHlwZSI6IkVuZCIsCiAgICAgICAgIklkIjoiNTAiCiAgICB9LAogICAgIk9wZXJhdGlvbiI6WwogICAgICAgIHsKICAgICAgICAgICAgIlR5cGUiOiJORkNXcml0ZSIsCiAgICAgICAgICAgICJWYXJpYW50IjoiTWlmYXJlQ2xhc3NpYyIsCiAgICAgICAgICAgICJORkNEYXRhIjpbCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgInNlY3RvciI6MSwKICAgICAgICAgICAgICAgICAgICAiZGF0YSI6Ii4uLiIKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgXQogICAgICAgIH0KICAgIF0KfQ==
-
Make a card acquisition request with the Base64-encoded string in the
SaleToPOIData
.
The example below shows how to write data MifareClassic NFC tag and end the session:Write data to the NFC tag and end the sessionExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"CardAcquisition", "MessageType":"Request", "ServiceID":"283", "SaleID":"POSSystemID12345", "POIID":"V400m-346403161" }, "CardAcquisitionRequest":{ "SaleData":{ "SaleTransactionID":{ "TransactionID":"369", "TimeStamp":"2021-03-05T12:22:57.449Z" }, "TokenRequestedType":"Customer" }, "SaleToPOIData":"ewogICAgIlNlc3Npb24iOnsKICAgICAgICAiVHlwZSI6IkVuZCIsCiAgICAgICAgIklkIjoiNTAiCiAgICB9LAogICAgIk9wZXJhdGlvbiI6WwogICAgICAgIHsKICAgICAgICAgICAgIlR5cGUiOiJORkNXcml0ZSIsCiAgICAgICAgICAgICJWYXJpYW50IjoiTWlmYXJlQ2xhc3NpYyIsCiAgICAgICAgICAgICJORkNEYXRhIjpbCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgInNlY3RvciI6MSwKICAgICAgICAgICAgICAgICAgICAiZGF0YSI6Ii4uLiIKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgXQogICAgICAgIH0KICAgIF0KfQ==", "CardAcquisitionTransaction":{ } } } } After you end the session, the loading screen continues to show until you send an enable service request (see the
next step). -
To stop the loading screen, make an enable service request to a Terminal API endpoint, specifying:
-
MessageHeader
: the standardSaleToPOIRequest.MessageHeader
object. This includes:Parameter Required Description MessageClass
Service MessageCategory
EnableService -
EnableServiceRequest with:
Parameter Required Description TransactionAction
AbortTransaction. DisplayOutput
Optional object to show your own message and an 'Approved' icon (green check mark) or a 'Declined' icon (red cross). If you omitDisplayOutput
, the terminal shows Canceled, a red cross , and Transaction canceled.
The following example is the basic request, without the
DisplayOutput
object, to stop the card acquisition flow.Basic request to cancel the card acquisition dataExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"EnableService", "MessageType":"Request", "ServiceID":"3020711110", "SaleID":"POSSystemID12345", "POIID":"V400m-346403161" }, "EnableServiceRequest":{ "TransactionAction":"AbortTransaction" } } } -