Skyflow’s android SDK can be used to securely collect, tokenize, and display sensitive data in the mobile without exposing your front-end infrastructure to sensitive data.
- Installation
- Initializing Skyflow-android
- Securely collecting data client-side
- Securely revealing data client-side
- Android Gradle plugin 4.2.0 and above
- Android 5.0 (API level 21) and above
- Inside you GitHub account:
- Settings -> Developer Settings -> Personal Access Tokens -> Generate new token
- Make sure you select the following scopes (“read:packages”) and Generate a token
- After Generating make sure to copy your new personal access token. You cannot see it again! The only option is to generate a new key.
- Create a github.properties file within your root Android project
- In case of a public repository make sure you add this file to .gitignore for keep the token private
- Add properties gpr.usr=GITHUB_USER_NAME and gpr.key=PERSONAL_ACCESS_TOKEN
- Replace GITHUB_USER_NAME with personal / organisation Github user NAME and PERSONAL_ACCESS_TOKEN with the token generated in Step 1
Alternatively you can also add the GPR_USER_NAME and GPR_PAT values to your environment variables on you local machine or build server to avoid creating a github properties file
-
Add the Github package registry to your root project build.gradle file
def githubProperties = new Properties() githubProperties.load(new FileInputStream(file(“github.properties”))) allprojects { repositories { ... maven { url "https://maven.pkg.github.com/skyflowapi/skyflow-android-sdk" credentials { username = githubProperties['gpr.usr'] ?: System.getenv("GPR_USER_NAME") password = githubProperties['gpr.key'] ?: System.getenv("GPR_PAT") } } }
-
Add the dependency to your application's build.gradle file
implementation 'com.skyflowapi.android:skyflow-android-sdk:1.18.0'
- Add the Github package registry in the repositories tag and the GITHUB_USER_NAME, PERSONAL_ACCESS_TOKEN collected from Step1 in the server tag to your project's settings.xml file. Make sure that the id's for both these tags are the same.
<repositories>
<repository>
<id>github</id>
<url>https://maven.pkg.github.com/skyflowapi/skyflow-android-sdk</url>
</repository>
</repositories>
<servers>
<server>
<id>github</id>
<username>GITHUB_USER_NAME</username>
<password>PERSONAL_ACCESS_TOKEN</password>
</server>
</servers>
- Add the package dependencies to the dependencies element of your project pom.xml file
<dependency>
<groupId>com.skyflowapi.android</groupId>
<artifactId>skyflow-android-sdk</artifactId>
<version>1.18.0</version>
</dependency>
Use the init()
method to initialize a Skyflow client as shown below.
val demoTokenProvider = DemoTokenProvider() /*DemoTokenProvider is an implementation of
the Skyflow.TokenProvider interface*/
val config = Skyflow.Configuration(
vaultID = <VAULT_ID>,
vaultURL = <VAULT_URL>,
tokenProvider = demoTokenProvider,
options: Skyflow.Options(
logLevel : Skyflow.LogLevel, // optional, if not specified loglevel is ERROR.
env: SKyflow.Env //optiuona, if not specified env is PROD.
)
)
val skyflowClient = Skyflow.init(config)
For the tokenProvider parameter, pass in an implementation of the Skyflow.TokenProvider interface that declares a getAccessToken method which retrieves a Skyflow bearer token from your backend. This function will be invoked when the SDK needs to insert or retrieve data from the vault.
For example, if the response of the consumer tokenAPI is in the below format
{
"accessToken": string,
"tokenType": string
}
then, your Skyflow.TokenProvider Implementation should be as below
class DemoTokenProvider: Skyflow.TokenProvider {
override fun getBearerToken(callback: Callback) {
val url = "http://10.0.2.2:8000/js/analystToken"
val request = okhttp3.Request.Builder().url(url).build()
val okHttpClient = OkHttpClient()
try {
val thread = Thread {
run {
okHttpClient.newCall(request).execute().use { response ->
if (!response.isSuccessful)
throw IOException("Unexpected code $response")
val accessTokenObject = JSONObject(
response.body()!!.string().toString()
)
val accessToken = accessTokenObject["accessToken"]
callback.onSuccess("$accessToken")
}
}
}
thread.start()
}catch (exception:Exception){
callback.onFailure(exception)
}
}
}
NOTE: You should pass access token as String
value in the success callback of getBearerToken.
For logLevel
parameter, there are 4 accepted values in Skyflow.LogLevel
-
DEBUG
When
Skyflow.LogLevel.DEBUG
is passed, all level of logs will be printed(DEBUG, INFO, WARN, ERROR). -
INFO
When
Skyflow.LogLevel.INFO
is passed, INFO logs for every event that has occurred during the SDK flow execution will be printed along with WARN and ERROR logs. -
WARN
When
Skyflow.LogLevel.WARN
is passed, WARN and ERROR logs will be printed. -
ERROR
When
Skyflow.LogLevel.ERROR
is passed, only ERROR logs will be printed.
Note
:
- The ranking of logging levels is as follows : DEBUG < INFO < WARN < ERROR
- since
logLevel
is optional, by default the logLevel will beERROR
.
For env
parameter, there are 2 accepted values in Skyflow.Env
-
PROD
-
DEV
In Event Listeners, actual value of element can only be accessed inside the handler when the
env
is set toDEV
.
Note
:
- since
env
is optional, by default the env will bePROD
. - Use
env
option with caution, make sure the env is set toPROD
when usingskyflow-android
in production.
- Inserting data into the vault
- Using Skyflow Elements to collect data
- Event Listener on Collect Elements
- UI Error for Collect Eements
- Set and Clear value for Collect Elements (DEV ENV ONLY)
To insert data into the vault from the integrated application, use the insert(records: JSONObject, options: InsertOptions?= InsertOptions() , callback: Skyflow.Callback)
method of the Skyflow client. The records parameter takes a JSON object of the records to be inserted in the below format. The options parameter takes a object of optional parameters for the insertion. insert
method also support upsert operations. See below:
{
"records": [
{
table: "string", //table into which record should be inserted
fields: {
column1: "value", //column names should match vault column names
///... additional fields
}
},
///...additional records
]
}
An example of an insert call is given below:
//Upsert options
val upsertArray = JSONArray()
val upsertColumn = JSONObject()
upsertColumn.put("table", "cards")
upsertColumn.put("column", "card_number")
upsertArray.put(upsertColumn)
val insertOptions = Skyflow.InsertOptions(tokens= false,upsert= upsertArray) /*indicates whether or not tokens should be returned for the inserted data. Defaults to 'true'*/
val insertCallback = InsertCallback() //Custom callback - implementation of Skyflow.Callback
val records = JSONObject()
val recordsArray = JSONArray()
val record = JSONObject()
record.put("table", "cards")
val fields = JSONObject()
fields.put("expiry_date", "12/2028")
fields.put("cardNumber", "41111111111")
record.put("fields", fields)
recordsArray.put(record)
records.put("records", recordsArray)
skyflowClient.insert(records = records, options = insertOptions, callback = insertCallback);
Response :
{
"records": [
{
"table": "cards",
"fields":{
"cardNumber": "f3907186-e7e2-466f-91e5-48e12c2bcbc1",
"expiry_date": "1989cb56-63da-4482-a2df-1f74cd0dd1a5"
}
}
]
}
Skyflow Elements provide developers with pre-built form elements to securely collect sensitive data client-side. This reduces your PCI compliance scope by not exposing your front-end application to sensitive data. Follow the steps below to securely collect data with Skyflow Elements in your application.
First create a container for the form elements using the skyflowClient.container(type: Skyflow.ContainerType)
method as show below
val container = skyflowClient.container(Skyflow.ContainerType.COLLECT)
To create a collect element, we must first construct Skyflow.CollectElementInput
object defined as shown below:
Skyflow.CollectElementInput(
table : String, //the table this data belongs to
column : String, //the column into which this data should be inserted
type: Skyflow.ElementType //Skyflow.ElementType enum
inputStyles: Skyflow.Styles, //optional styles that should be applied to the form element
labelStyles: Skyflow.Styles, //optional styles that will be applied to the label of the collect element
errorTextStyles: Skyflow.Styles, //optional styles that will be applied to the errorText of the collect element
label: String, //optional label for the form element
placeholder: String, //optional placeholder for the form element
altText: String, //(DEPRECATED) optional string that acts as an initial value for the collect element
validations: ValidationSet // optional set of validations for collect element
)
The table
and column
parameters indicate which table and column in the vault the Element corresponds to.
Note: Use dot delimited strings to specify columns nested inside JSON fields (e.g. address.street.line1).
The inputStyles
field accepts a Skyflow.Styles object which consists of multiple Skyflow.Style
objects which should be applied to the form element in the following states:
base
: all other variants inherit from these stylescomplete
: applied when the Element has valid inputempty
: applied when the Element has no inputfocus
: applied when the Element has focusinvalid
: applied when the Element has invalid input
Each Style object accepts the following properties, please note that each property is optional:
Skyflow.Style(
borderColor: Int //optional
cornerRadius: Float //optional
padding: Skyflow.Padding //optional
borderWidth: Int //optional
font: Int //optional
textAlignment: Int //optional
textColor: Int //optional
)
Here Skyflow.Padding is a class which can be used to set the padding for the collect element which takes all the left, top, right, bottom padding values.
Skyflow.Padding(left: Int, top: Int, right: Int, bottom: Int)
An example Skyflow.Styles object
val inputStyles = Skyflow.Styles(
base = Skyflow.Style(), //optional
complete = Skyflow.Style(), //optional
empty = Skyflow.Style(), //optional
focus = Skyflow.Style(), //optional
invalid = Skyflow.Style() //optional
)
The labelStyles
and errorTextStyles
fields accept the above mentioned Skyflow.Styles
object which are applied to the label
and errorText
text views respectively.
The states that are available for labelStyles
are base
and focus
.
The state that is available for errorTextStyles
is only the base
state, it shows up when there is some error in the collect element.
The parameters in Skyflow.Style
object that are respected for label
and errorText
text views are
- padding
- font
- textColor
- textAlignment
Other parameters in the Skyflow.Style
object are ignored for label
and errorText
text views.
Finally, the type
field takes a Skyflow ElementType. Each type applies the appropriate regex and validations to the form element. There are currently 5 types:
INPUT_FIELD
CARDHOLDER_NAME
CARD_NUMBER
EXPIRATION_DATE
EXPIRATION_MONTH
EXPIRATION_YEAR
CVV
PIN
The INPUT_FIELD
type is a custom UI element without any built-in validations. See the section on validations
for more information on validations.
Along with CollectElementInput
we can define other options which are optional inside the CollectElementOptions
object which is described below.
Skyflow.CollectElementOptions(
required: Boolean, //indicates whether the field is marked as required. Defaults to 'false'
enableCardIcon: Boolean, //indicates whether card icon should be enabled (only for CARD_NUMBER inputs)
format: String //Format for the element (only applicable currently for "EXPIRATION_DATE")
)
required
parameter indicates whether the field is marked as required or not, if not provided, it defaults to false
enableCardIcon
paramenter indicates whether the icon is visible for the CARD_NUMBER
element, defaults to true
format
parameter takes string value and indicates the format pattern applicable to the element type, It's currently only applicable to EXPIRATION_DATE
and EXPIRATION_YEAR
element types.
The values that are accepted for EXPIRATION_DATE
are
- mm/yy (default)
- mm/yyyy
- yy/mm
- yyyy/mm
The values that are accepted for EXPIRATION_YEAR
are
- yy (default)
- yyyy
NOTE
: If not specified or invalid value is passed to the format
then it takes default value.
Once the Skyflow.CollectElementInput
and Skyflow.CollectElementOptions
objects are defined, add to the container using the create(context:Context,input: CollectElementInput, options: CollectElementOptions)
method as shown below. The input
param takes a Skyflow.CollectElementInput
object as defined above and the options
parameter takes a Skyflow.CollectElementOptions
,
the context
param takes android Context
object as described below:
val collectElementInput = Skyflow.CollectElementInput(
table = "string", //the table this data belongs to
column = "string", //the column into which this data should be inserted
type = Skyflow.ElementType.CARD_NUMBER, //Skyflow.ElementType enum
inputStyles = Skyflow.Styles(), /*optional styles that should be applied to the form element*/
labelStyles = Skyflow.Styles(), //optional styles that will be applied to the label of the collect element
errorTextStyles = Skyflow.Styles(), //optional styles that will be applied to the errorText of the collect element
label = "string", //optional label for the form element
placeholder = "string", //optional placeholder for the form element
altText: String, //(DEPRECATED) optional string that acts as an initial value for the collect element
validations = ValidationSet() // optional set of validations for the input element
)
val collectElementOptions = Skyflow.CollectElementOptions(
required = false, //indicates whether the field is marked as required. Defaults to 'false'
enableCardIcon = true //indicates whether card icon should be enabled (only for CARD_NUMBER inputs)
format = "mm/yy" //Format for the element (only applies currently for EXPIRATION_DATE element type)
)
const element = container.create(context = Context, collectElementInput, collectElementOptions)
To specify where the Elements will be rendered on the screen, set layout params to the view and add it to a layout in your app programmatically.
val layoutParams = LinearLayout.LayoutParams(
LinearLayout.LayoutParams.MATCH_PARENT,
LinearLayout.LayoutParams.WRAP_CONTENT
)
element.layoutParams = layoutParams
existingLayout.addView(element)
The Skyflow Element is an implementation of native android View so it can be used/mounted similarly.Alternatively, you can use the unmount
method to reset any element to it's initial state.
fun clearFields(elements: List<TextField>) {
//resets all elements to initial value
for element in elements {
element.unmount()
}
}
When the form is ready to be submitted, call the collect(options: Skyflow.CollectOptions? = nil, callback: Skyflow.Callback) method on the container object. The options parameter takes Skyflow.CollectOptions
object.
Skyflow.CollectOptions
takes two optional fields
tokens
: indicates whether tokens for the collected data should be returned or not. Defaults to 'true'additionalFields
: Non-PCI elements data to be inserted into the vault which should be in therecords
object format as described in the above Inserting data into vault section.
// NON-PCI fields object creation
val nonPCIRecords = JSONObject()
val recordsArray = JSONArray()
val record = JSONObject()
record.put("table", "persons")
val fields = JSONObject()
fields.put("gender", "MALE")
record.put("fields", fields)
recordsArray.put(record)
nonPCIRecords.put("records", recordsArray)
val options = Skyflow.CollectOptions(tokens = true, additonalFields = nonPCIRecords)
val insertCallback = InsertCallback() //Custom callback - implementation of Skyflow.callback
container.collect(options, insertCallback)
//Initialize skyflow configuration
val config = Skyflow.Configuration(vaultId = VAULT_ID, vaultURL = VAULT_URL, tokenProvider = demoTokenProvider)
//Initialize skyflow client
val skyflowClient = Skyflow.initialize(config)
//Create a CollectContainer
val container = skyflowClient.container(type = Skyflow.ContainerType.COLLECT)
//Initialize and set required options
val options = Skyflow.CollectElementOptions(required = true)
//Create Skyflow.Styles with individual Skyflow.Style variants
val baseStyle = Skyflow.Style(borderColor = Color.BLUE)
val baseTextStyle = Skyflow.Style(textColor = Color.BLACK)
val completedStyle = Skyflow.Style(textColor = Color.GREEN)
val focusTextStyle = Skyflow.Style(textColor = Color.RED)
val inputStyles = Skyflow.Styles(base = baseStyle, complete = completedStyle)
val labelStyles = Skyflow.Styles(base = baseTextStyle, focus = focusTextStyle)
val errorTextStyles = Skyflow.Styles(base = baseTextStyle)
//Create a CollectElementInput
val input = Skyflow.CollectElementInput(
table = "cards",
column = "cardNumber",
type = Skyflow.ElementType.CARD_NUMBER
inputStyles = inputStyles,
labelStyles = labelStyles,
errorTextStyles = errorTextStyles,
label = "card number",
placeholder = "card number",
)
//Create a CollectElementOptions instance
val options = Skyflow.CollectElementOptions(required = true)
//Create a Collect Element from the Collect Container
val skyflowElement = container.create(context = Context,input, options)
//Can interact with this object as a normal UIView Object and add to View
// Non-PCI fields data
val nonPCIRecords = JSONObject()
val recordsArray = JSONArray()
val record = JSONObject()
record.put("table", "persons")
val fields = JSONObject()
fields.put("gender", "MALE")
record.put("fields", fields)
recordsArray.put(record)
nonPCIRecords.put("records", recordsArray)
//Initialize and set required options for insertion
val collectOptions = Skyflow.CollectOptions(tokens = true, additionalFields = nonPCIRecords)
//Implement a custom Skyflow.Callback to be called on Insertion success/failure
public class InsertCallback: Skyflow.Callback {
override fun onSuccess(responseBody: Any) {
print(responseBody)
}
override fun onFailure(_ error: Error) {
print(error)
}
}
//Initialize InsertCallback which is an implementation of Skyflow.Callback interface
val insertCallback = InsertCallback()
//Call collect method on CollectContainer
container.collect(options = collectOptions, callback = insertCallback)
{
"records": [
{
"table": "cards",
"fields": {
"cardNumber": "f3907186-e7e2-466f-91e5-48e12c2bcbc1"
}
},
{
"table": "persons",
"fields": {
"gender": "12f670af-6c7d-4837-83fb-30365fbc0b1e",
}
}
]
}
val config = Skyflow.Configuration(vaultId = VAULT_ID, vaultURL = VAULT_URL, tokenProvider = demoTokenProvider)
val skyflowClient = Skyflow.initialize(config)
val container = skyflowClient.container(type = Skyflow.ContainerType.COLLECT)
val options = Skyflow.CollectElementOptions(required = true)
val baseStyle = Skyflow.Style(borderColor = Color.BLUE)
val baseTextStyle = Skyflow.Style(textColor = Color.BLACK)
val completedStyle = Skyflow.Style(textColor = Color.GREEN)
val focusTextStyle = Skyflow.Style(textColor = Color.RED)
val inputStyles = Skyflow.Styles(base = baseStyle, complete = completedStyle)
val labelStyles = Skyflow.Styles(base = baseTextStyle, focus = focusTextStyle)
val errorTextStyles = Skyflow.Styles(base = baseTextStyle)
val cardNumberInput = Skyflow.CollectElementInput(
table = "cards",
column = "card_number",
type = Skyflow.ElementType.CARD_NUMBER
inputStyles = inputStyles,
labelStyles = labelStyles,
errorTextStyles = errorTextStyles,
label = "Card number",
placeholder = "enter your card number",
)
val nameInput = Skyflow.CollectElementInput(
table = "cards",
column = "full_name",
type = Skyflow.ElementType.CARD_NUMBER
inputStyles = inputStyles,
labelStyles = labelStyles,
errorTextStyles = errorTextStyles,
label = "Full name",
placeholder = "enter your name",
)
val cardNumberElement = container.create(context = Context,cardNumberInput, options)
val nameElement = container.create(context = Context,namerInput, options)
//Upsert options
val upsertArray = JSONArray()
val upsertColumn = JSONObject()
upsertColumn.put("table", "cards")
upsertColumn.put("column", "card_number")
upsertArray.put(upsertColumn)
val collectOptions = Skyflow.CollectOptions(tokens = true,upsert = upsertArray)
public class InsertCallback: Skyflow.Callback {
override fun onSuccess(responseBody: Any) {
print(responseBody)
}
override fun onFailure(_ error: Error) {
print(error)
}
}
val insertCallback = InsertCallback()
container.collect(options = collectOptions, callback = insertCallback)
{
"records": [
{
"table": "cards",
"fields": {
"card_number": "f3907186-e7e2-466f-91e5-48e12c2bcbc1",
"name": "f3907186-e7e2-464f-91e5-48e12c2bfsi9"
}
}
]
}
skyflow-android provides two types of validations on Collect Elements
Every Collect Element except of type INPUT_FIELD
has a set of default validations listed below:
CARD_NUMBER
: Card number validation with checkSum algorithm(Luhn algorithm), available card lengths for defined card typesCARD_HOLDER_NAME
: Name, should be 2 or more symbols, valid characters shold match pattern^([a-zA-Z\\ \\,\\.\\-\\']{2,})$
CVV
: Card CVV can have 3-4 digitsEXPIRATION_DATE
: Any date starting from current month. By default valid expiration date should be in short year format -MM/YY
PIN
: Can have 4-12 digits
Custom validations can be added to any element which will be checked after the default validations have passed. The following Custom validation rules are currently supported:
RegexMatchRule
: You can use this rule to specify any Regular Expression to be matched with the text field valueLengthMatchRule
: You can use this rule to set the minimum and maximum permissible length of the textfield valueElementValueMatchRule
: You can use this rule to match the value of one element with another
The Sample code below illustrates the usage of custom validations:
/*
Reset Password - A simple example that illustrates custom validations. The below code shows two input fields with custom validations, one to enter a Password and the second to confirm the same Password.
*/
var myRuleset = ValidationSet()
val strongPasswordRule = RegexMatchRule(regex= "^^(?=.*[A-Za-z])(?=.*\\d)[A-Za-z\\d]*$", error = "At least one letter and one number") // This rule enforces a strong password
val lengthRule = LengthMatchRule(minLength = 8, maxLength = 16, error = "Must be between 8 and 16 digits") // this rule allows input length between 8 and 16 characters
// for the Password element
myRuleset.add(rule = strongPasswordRule)
myRuleset.add(rule = lengthRule)
val passwordInput = CollectElementInput(inputStyles = styles, label = "Password", placeholder = "****", type = ElementType.INPUT_FIELD, validations = myRuleset)
val Password = container.create(passwordInput)
// For confirm Password element - shows error when the PINs don't match
val elementMatchRule = ElementMatchRule(element = Password, error = "PINs don't match")
val confirmPasswordinput = CollectElementInput(inputStyles = styles, label = "Confirm Password", placeholder = "****", type = ElementType.INPUT_FIELD, validations = ValidationSet(rules = mutableListOf(strongPasswordRule, lengthRule, elementMatchRule)))
val confirmPassword = container.create(input = confirmPasswordinput)
//mount elements to the screen
addView(Password)
addView(confirmPassword)
Helps to communicate with skyflow elements by listening to an event
element.on(eventName: Skyflow.EventName) { state ->
//handle function
}
There are 4 events in Skyflow.EventName
CHANGE
Change event is triggered when the Element's value changes.READY
Ready event is triggered when the Element is fully renderedFOCUS
Focus event is triggered when the Element gains focusBLUR
Blur event is triggered when the Element loses focus. The handler(state: JSONObject) -> Unit
is a callback function you provide, that will be called when the event is fired with the state object as shown below.
val state = {
"elementType": Skyflow.ElementType,
"isEmpty": Bool,
"isFocused": Bool,
"isValid": Bool,
"value": String
}
Note:
values of SkyflowElements will be returned in element state object only when env
is DEV
, else it is empty string i.e, '', but in case of CARD_NUMBER type element when the env
is PROD
for all the card types except AMEX, it will return first eight digits, for AMEX it will return first six digits and rest all digits in masked format.
//create skyflow client with loglevel:"DEBUG"
val config = Skyflow.Configuration(vaultID = VAULT_ID, vaultURL = VAULT_URL, tokenProvider = demoTokenProvider, options = Skyflow.Options(logLevel = Skyflow.LogLevel.DEBUG))
val skyflowClient = Skyflow.initialize(config)
val container = skyflowClient.container(type = Skyflow.ContainerType.COLLECT)
// Create a CollectElementInput
val cardNumberInput = Skyflow.CollectElementInput(
table = "cards",
column = "cardNumber",
type = Skyflow.ElementType.CARD_NUMBER,
)
val cardHolderNameInput = Skyflow.CollectElementInput(
table = "cards",
column = "cardHolderName",
type = Skyflow.ElementType.CARDHOLDER_NAME,
)
val cardNumber = container.create(input = cardNumberInput)
val cardHolderName = container.create(input = cardHolderNameInput)
//subscribing to CHANGE event, which gets triggered when element changes
cardNumber.on(eventName = Skyflow.EventName.CHANGE) { state ->
// Your implementation when Change event occurs
log.info("on change", state)
}
cardHolderName.on(eventName = Skyflow.EventName.CHANGE) { state ->
// Your implementation when Change event occurs
log.info("on change", state)
}
{
"elementType": Skyflow.ElementType.CARD_NUMBER,
"isEmpty": false,
"isFocused": true,
"isValid": true,
"value": "4111111111111111"
}
{
"elementType": Skyflow.ElementType.CARDHOLDER_NAME,
"isEmpty": false,
"isFocused": true,
"isValid": true,
"value": "John"
}
{
"elementType": Skyflow.ElementType.CARD_NUMBER,
"isEmpty": false,
"isFocused": true,
"isValid": true,
"value": "41111111XXXXXXXX"
}
{
"elementType": Skyflow.ElementType.CARDHOLDER_NAME,
"isEmpty": false,
"isFocused": true,
"isValid": true,
"value": ""
}
Helps to display custom error messages on the Skyflow Elements through the methods setError
and resetError
on the elements.
setError(error : String)
method is used to set the error text for the element, when this method is trigerred, all the current errors present on the element will be overridden with the custom error message passed. This error will be displayed on the element until resetError()
is trigerred on the same element.
resetError()
method is used to clear the custom error message that is set using setError
.
//create skyflow client with loglevel:"DEBUG"
val config = Skyflow.Configuration(vaultID = VAULT_ID, vaultURL = VAULT_URL, tokenProvider = demoTokenProvider, options = Skyflow.Options(logLevel = Skyflow.LogLevel.DEBUG))
val skyflowClient = Skyflow.initialize(config)
val container = skyflowClient.container(type = Skyflow.ContainerType.COLLECT)
// Create a CollectElementInput
val cardNumberInput = Skyflow.CollectElementInput(
table = "cards",
column = "cardNumber",
type = Skyflow.ElementType.CARD_NUMBER,
)
val cardNumber = container.create(input = cardNumberInput)
//Set custom error
cardNumber.setError("custom error")
//reset custom error
cardNumber.resetError()
}
setValue(value: String)
method is used to set the value of the element. This method will override any previous value present in the element.
clearValue()
method is used to reset the value of the element.
Note:
This methods are only available in DEV env for testing/developmental purposes and MUST NOT be used in PROD env.
//create skyflow client with env DEV
val config = Skyflow.Configuration(vaultID = VAULT_ID, vaultURL = VAULT_URL, tokenProvider = demoTokenProvider, options = Skyflow.Options(env = Skyflow.Env.DEV))
val skyflowClient = Skyflow.initialize(config)
val container = skyflowClient.container(type = Skyflow.ContainerType.COLLECT)
// Create a CollectElementInput
val cardNumberInput = Skyflow.CollectElementInput(
table = "cards",
column = "cardNumber",
type = Skyflow.ElementType.CARD_NUMBER,
)
val cardNumber = container.create(input = cardNumberInput)
//Set a value programatically
cardNumber.setValue("4111111111111111")
//Clear the value
cardNumber.clearValue()
}
- Retrieving data from the vault
- Using Skyflow Elements to reveal data
- UI Error for Reveal Elements
- Set token for Reveal Elements
- Set and clear altText for Reveal Elements
For non-PCI use-cases, retrieving data from the vault and revealing it in the mobile can be done either using the SkyflowID's or tokens as described below
-
For retrieving using tokens, use the
detokenize(records)
method. The records parameter takes a JSON object that containsrecords
to be fetched as shown below.{ "records":[ { "token": "string", // token for the record to be fetched } ] }
An example of a detokenize call:
val getCallback = GetCallback() //Custom callback - implementation of Skyflow.Callback val records = JSONObject() val recordsArray = JSONArray() val recordObj = JSONObject() recordObj.put("token", "45012507-f72b-4f5c-9bf9-86b133bae719") recordsArray.put(recordObj) records.put("records", recordsArray) skyflowClient.detokenize(records = records, callback = getCallback)
The sample response:
{ "records": [ { "token": "131e70dc-6f76-4319-bdd3-96281e051051", "value": "1990-01-01" } ] }
-
For retrieving using SkyflowID's, use the
getById(records)
method.The records parameter takes a JSON object that containsrecords
to be fetched as shown below.{ "records":[ { ids: ArrayList<String>(), // Array of SkyflowID's of the records to be fetched table: "string" // name of table holding the above skyflow_id's redaction: Skyflow.RedactionType //redaction to be applied to retrieved data } ] }
An example of getById call:
val getCallback = GetCallback() //Custom callback - implementation of Skyflow.Callback var recordsArray = JSONArray() var record = JSONObject() record.put("table","cards") record.put("redaction","PLAIN_TEXT") val skyflowIDs = ArrayList<String>() skyflowIDs.add("f8d8a622-b557-4c6b-a12c-c5ebe0b0bfd9") skyflowIDs.add("da26de53-95d5-4bdb-99db-8d8c66a35ff9") record.put("ids",skyflowIDs) var record1 = JSONObject() record1.put("table","cards") record1.put("redaction","PLAIN_TEXT") val recordSkyflowIDs = ArrayList<String>() recordSkyflowIDs.add("invalid skyflow id") // invalid skyflow ID record.put("ids",recordSkyflowIDs) recordsArray.put(record1) val records = JSONObject() records.put("records",recordsArray) skyflowClient.getById(records = records, callback = getCallback)
The sample response:
{ "records": [ { "fields": { "card_number": "4111111111111111", "expiry_date": "11/35", "fullname": "myname", "id": "f8d8a622-b557-4c6b-a12c-c5ebe0b0bfd9" }, "table": "cards" }, { "fields": { "card_number": "4111111111111111", "expiry_date": "10/23", "fullname": "sam", "id": "da26de53-95d5-4bdb-99db-8d8c66a35ff9" }, "table": "cards" } ], "errors": [ { "error": { "code": "404", "description": "No Records Found" }, "ids": ["invalid skyflow id"] } ] }
There are four enum values in Skyflow.RedactionType:
PLAIN_TEXT
MASKED
REDACTED
DEFAULT
Skyflow Elements can be used to securely reveal data in an application without exposing your front end to the sensitive data. This is great for use-cases like card issuance where you may want to reveal the card number to a user without increasing your PCI compliance scope.
To start, create a container using the skyflowClient.container(Skyflow.ContainerType.REVEAL)
method as shown below.
val container = skyflowClient.container(type = Skyflow.ContainerType.REVEAL)
Then define a Skyflow Element to reveal data as shown below.
val revealElementInput = Skyflow.RevealElementInput(
token = "string",
inputStyles = Skyflow.Styles(), //optional, styles to be applied to the element
labelStyles = Skyflow.Styles(), //optional, styles to be applied to the label of the reveal element
errorTextStyles = Skyflow.Styles(), //optional styles that will be applied to the errorText of the reveal element
label = "cardNumber" //optional, label for the element,
altText = "XXXX XXXX XXXX XXXX" //optional, string that is shown before reveal, will show token if altText is not provided
)
Note
:
token
is optional only if it is being used in invokeConnection()
The inputStyles
parameter accepts a styles object as described in the previous section for collecting data but the only state available for a reveal element is the base state.
The labelStyles
and errorTextStyles
fields accept the above mentioned Skyflow.Styles
object as described in the previous section, the only state available for a reveal element is the base state.
The inputStyles
, labelStyles
and errorTextStyles
parameters accepts a styles object as described in the previous section for collecting data but only a single variant is available i.e. base.
An example of a inputStyles object:
var inputStyles = Skyflow.Styles(base = Skyflow.Style(
borderColor = Color.BLUE))
An example of a labelStyles object:
var labelStyles = Skyflow.Styles(base =
Skyflow.Style(font = 12))
An example of a errorTextStyles object:
var labelStyles = Skyflow.Styles(base =
Skyflow.Style(textColor = COLOR.RED))
Once you've defined a Skyflow.RevealElementInput
object, you can use the create()
method of the container to create the Element as shown below:
val element = container.create(context = Context, input = revealElementInput,RevealElementOptions(formateRegex="..$"))
Elements used for revealing data are mounted to the screen the same way as Elements used for collecting data. Refer to Step 3 of the section above.
When the sensitive data is ready to be retrieved and revealed, call the reveal()
method on the container as shown below:
val revealCallback = RevealCallback() //Custom callback - implementation of Skyflow.Callback
container.reveal(callback = revealCallback)
Helps to display custom error messages on the Skyflow Elements through the methods setError
and resetError
on the elements.
setError(error : String)
method is used to set the error text for the element, when this method is trigerred, all the current errors present on the element will be overridden with the custom error message passed. This error will be displayed on the element until resetError()
is trigerred on the same element.
resetError()
method is used to clear the custom error message that is set using setError
.
The setToken(value: String)
method can be used to set the token of the Reveal Element. If no altText is set, the set token will be displayed on the UI as well. If altText is set, then there will be no change in the UI but the token of the element will be internally updated.
The setAltText(value: String)
method can be used to set the altText of the Reveal Element. This will cause the altText to be displayed in the UI regardless of whether the token or value is currently being displayed.
clearAltText()
method can be used to clear the altText, this will cause the element to display the token or actual value of the element. If the element has no token, the element will be empty.
//Initialize skyflow configuration
val config = Skyflow.Configuration(vaultId = <VAULT_ID>, vaultURL = <VAULT_URL>, tokenProvider = demoTokenProvider)
//Initialize skyflow client
val skyflowClient = Skyflow.initialize(config)
//Create a Reveal Container
val container = skyflowClient.container(type = Skyflow.ContainerType.REVEAL)
//Create Skyflow.Styles with individual Skyflow.Style variants
val baseStyle = Skyflow.Style(borderColor = Color.BLUE)
val baseTextStyle = Skyflow.Style(textColor = Color.BLACK)
val inputStyles = Skyflow.Styles(base = baseStyle)
val labelStyles = Skyflow.Styles(base = baseTextStyle)
val errorTextStyles = Skyflow.Styles(base = baseTextStyle)
//Create Reveal Elements
val cardNumberInput = Skyflow.RevealElementInput(
token = "b63ec4e0-bbad-4e43-96e6-6bd50f483f75",
inputStyles = inputStyles,
labelStyles = labelStyles,
errorTextStyles = errorTextStyles,
label = "cardnumber",
altText = "XXXX XXXX XXXX XXXX"
)
val cardNumberElement = container.create(context = Context, input = cardNumberInput)
val nameInput = Skyflow.RevealElementInput(
token = "89024714-6a26-4256-b9d4-55ad69aa4047",
inputStyles = inputStyles,
labelStyles = labelStyles,
errorTextStyles = errorTextStyles,
label = "fullname",
altText = "XXX"
)
val nameElement = container.create(context = Context,input = nameInput)
//set error to the element
nameElement.setError("custom error")
//reset error to the element
nameElement.resetError()
//Can interact with these objects as a normal UIView Object and add to View
//Implement a custom Skyflow.Callback to be called on Reveal success/failure
public class RevealCallback: Skyflow.Callback {
override fun onSuccess(responseBody: Any) {
print(responseBody)
}
override fun onFailure(exception: Exception) {
print(exception)
}
}
//Initialize custom Skyflow.Callback
val revealCallback = RevealCallback()
//Call reveal method on RevealContainer
container.reveal(callback = revealCallback)
The response below shows that some tokens assigned to the reveal elements get revealed successfully, while others fail and remain unrevealed.
{
"success": [
{
"id": "b63ec4e0-bbad-4e43-96e6-6bd50f483f75"
}
],
"errors": [
{
"id": "89024714-6a26-4256-b9d4-55ad69aa4047",
"error": {
"code": 404,
"description": "Tokens not found for 89024714-6a26-4256-b9d4-55ad69aa4047"
}
}
]
}
Currently the skyflow collect elements and reveal elements can't be used in the XML layout definition, we have to add them to the views programatically.