Yoti Java SDK
Welcome to the Yoti Java SDK. This repo contains the tools and step by step instructions you need to quickly integrate your Java back-end with Yoti so that your users can share their identity details with your application in a secure and trusted way.
Table of Contents
-  An Architectural view - High level overview of integration 
-  Requirements - Everything you need to get started 
-  Building From Source - Everything you need to build from source 
-  Enabling the SDK - Description on importing your SDK 
-  Client Initialisation - Description on setting up your SDK 
-  Profile Retrieval - Description on setting up profile 
-  Handling Users - Description on handling user logons 
-  Connectivity Requirements - Description of network connectivity requirements 
-  Modules - The Modules above explained 
-  Spring Boot Auto Configuration - Description of utilising Spring Boot 
-  Running the Examples - Running the Spring example projects 
-  Breaking changes and enhancements made in v2.0.0 - Things you need to know when migrating from 1.x to 2.x 
-  Spring Security Integration - Integrating Yoti Authentication with Spring Boot. 
-  Misc - Miscellaneous items 
-  Known Issues - Known issues using the libraries 
-  Support - Please feel free to reach out 
An Architectural View
To integrate your application with Yoti, your back-end must expose a GET endpoint that Yoti will use to forward tokens. The endpoint can be configured in Yoti Hub when you create/update your application.
The image below shows how your application back-end and Yoti integrate in the context of a Login flow. Use the Yoti SDK to complete steps 6 through 9 for you, including profile decryption and communication with backend services.
Yoti also allows you to enable user details verification from your mobile app by means of the Android (TBA) and iOS (TBA) SDKs. In that scenario, your Yoti-enabled mobile app is playing both the role of the browser and the Yoti app. By the way, your back-end doesn't need to handle these cases in a significantly different way. You might just decide to handle the User-Agent header in order to provide different responses for web and mobile clients.
Requirements
- Java 1.8 or higher
- SLF4J
Building From Source
Building from source is not necessary since artifacts are published in Maven Central. However, if you want to build from source use the Maven Wrapper bundled with this distribution. For those familiar with Gradle this is much like the Gradle Wrapper and ensures that the correct version of Maven is being used.
From the top level directory:
./mvnw clean install -Dgpg.skip=true 
Notable flags that you may wish to use to skip certain static analysis/code quality tools are listed below.
- -Dfindbugs.skip=true: skips findbugs and the findbugs security extension.
- -Ddependency-check.skip=true: skips the OWASP dependency scanner.
Example Usage
./mvnw clean install -Dfindbugs.skip=true -Ddependency-check.skip=true 
Enabling the SDK
To include the Yoti SDK in your project, you can use your favourite dependency management system. If you are using Maven, you need to add the following dependency:
<dependency>
    <groupId>com.yoti</groupId>
    <artifactId>yoti-sdk-api</artifactId>
    <version>3.0.0</version>
</dependency> 
If you are using Gradle, here is the dependency to add:
compile group: 'com.yoti', name: 'yoti-sdk-api', version: '3.0.0'
You will find all classes packaged under com.yoti.api
Client Initialisation
The entry point of the SDK is com.yoti.api.client.YotiClient. To initialise it you need to include the following snippet:
import com.yoti.api.client.YotiClient;
import static com.yoti.api.client.FileKeyPairSource.fromFile;
YotiClient client = YotiClient.builder()
    .withClientSdkId(<YOUR_CLIENT_SDK_ID>)
    .withKeyPair(fromFile(new java.io.File("<PATH/TO/YOUR/APPLICATION/KEY_PAIR.pem>")))
    .build(); 
Where:
- YOUR_CLIENT_SDK_IDis the identifier generated by Yoti Hub when you create your app.
- PATH/TO/YOUR/APPLICATION/KEY_PAIR.pemis the path to the pem file your browser generates for you, when you create your app on Yoti Hub.
Profile Retrieval
Callback URl
To allow sharing a Users profile with your application, you must expose a GET endpoint over HTTPS that Yoti will use to push a one time use token to your back end.
You can use any public domain name or localhost, as well as any path or port, but the resulting url must be served over HTTPS. Your endpoint must consume a query string parameter named 'token' - that's the value you will need for retrieving the shared profile.
You configure the callback URL for your application on the Integration page of the Yoti Hub.
Important: these tokens only allow a single use. You cannot retrieve a profile using the same token more than once.
ActivityDetails
When your application receives a one time use token from Yoti, you retrieve details of the share with the following:
import com.yoti.api.client.ActivityDetails;
ActivityDetails activityDetails = client.getActivityDetails(encryptedToken); 
or, for a more detailed example:
import com.yoti.api.client.YotiClient;
import com.yoti.api.client.ActivityDetails;
import com.yoti.api.client.HumanProfile;
import com.yoti.api.client.ProfileException;
try {
    final ActivityDetails activityDetails = client.getActivityDetails(token);
    final HumanProfile profile = activityDetails.getUserProfile();
    //use the profile to extract attributes
} catch (ProfileException e) {
    LOG.info("Could not get profile", e);
    // do something meaningful
} 
com.yoti.api.client.ActivityDetails gives you access to the com.yoti.api.client.HumanProfile of the user that has shared with you.
Handling Users
Profiles
User profiles are returned to applications with a user ID unique to that application. This user ID will be fixed for your app, so can use it to determine whether a user is new to you. For example:
ActivityDetails activityDetails;
try {
    activityDetails = client.getActivityDetails(token);
    Optional<YourAppUserClass> user = yourUserSearchMethod(activityDetails.getRememberMeId());
    if (user.isPresent()) {
        String rememberMeId = activityDetails.getRememberMeId();
        Attribute<Image> selfie = profile.getSelfie();
        if (selfie != null) {
            Image selfieValue = selfie.getValue();
            String base64Selfie = selfieValue.getBase64Content();
        }
        Attribute<String> fullName = profile.getFullName();
        Attribute<String> givenNames = profile.getGivenNames();
        Attribute<String> familyName = profile.getFamilyName();
        Attribute<String> phoneNumber = profile.getPhoneNumber();
        Attribute<String> emailAddress = profile.getEmailAddress();
        AgeVerification over18Verification = profile.findAgeOverVerification(18);
        if (over18Verification != null) {
            boolean isAgedOver18 = over18Verification.getResult();
        }
        Attribute<Date> dateOfBirth = profile.getDateOfBirth();
        Attribute<String> gender = profile.getGender();
        Attribute<String> postalAddress = profile.getPostalAddress();
        Attribute<String> nationality = profile.getNationality();
        Attribute<Map<?, ?>> structuredPostalAddress = profile.getStructuredPostalAddress();
        Attribute<DocumentDetails> documentDetails = profile.getDocumentDetails();
  } else {
      // handle registration
  }
} catch (ProfileException e) {
  LOG.info("Could not get profile", e);
  return "error";
} 
Where yourUserSearchMethod is a method in your app that finds the user for the given rememberMeId. Regardless of whether the user is new to your app, Yoti will always provide the profile. So you don't necessarily need to store it.
Attributes
User details are provided by a list of attributes on the com.yoti.api.client.HumanProfile class. Whether a given attribute is present or not depends on the settings you have applied on Yoti Hub.
Attributes are returned as an instance of com.yoti.api.client.Attribute<T>. Since v2.0.0 of the SDK, Attribute is a generic class where represents the type of the Attribute value returned.
Anchors, Sources and Verifiers
com.yoti.api.client.Anchor represents how a given Attribute<T> has been sourced or verified. These values are created and signed whenever a Profile Attribute is created, or verified with an external party.
For example, an attribute value that was sourced from a Passport might have the following values:
| Anchorproperty | Example value | 
|---|---|
| type | SOURCE | 
| value | PASSPORT | 
| subType | OCR | 
| signedTimestamp | 2017-10-31, 19:45:59.123789 | 
Similarly, an attribute verified against the data held by an external party will have an Anchor of type VERIFIER, naming the party that verified it.
Attribute Anchors are returned as List<Anchor> from Attribute<T>.getSources and Attribute<T>.getVerifiers.
Connectivity Requirements
Using the com.yoti.api.client.YotiClient to get com.yoti.api.client.ActivityDetails requires your app to establish an outbound TCP connection to port 443 on the Yoti servers at https://api.yoti.com (by default - see the Misc section).
By default the Yoti Client will block indefinitely when connecting to the remote server or reading data. Consequently it is possible that your application thread could be blocked.
Since version 1.1 of the yoti-sdk-api you can set the following two system properties to control this behaviour:
- yoti.client.connect.timeout.ms- the number of milliseconds that you are prepared to wait for the connection to be established. Zero is interpreted as an infinite timeout.
- yoti.client.read.timeout.ms- the number of milliseconds that you are prepared to wait for data to become available to read in the response stream. Zero is interpreted as an infinite timeout.
Modules
The SDK is split into a number of modules for easier use and future extensibility.
yoti-sdk-api
Being the only interface, you need to explicitly couple your code to this module. Exposes the core classes:
| Class | Description | 
|---|---|
| HumanProfile | The set of attributes the user has configured for the transaction. | 
| YotiClientBuilder | Builds a YotiClient instance by automatically selecting the available implementations on the class path. | 
| YotiClient | Allows your app to retrieve a user profile, given an encrypted token. | 
| KeyPairSource and its implementations | A set of classes responsible for working with different sources (e.g. files, classpath resources, URLs) to load the private/public keypair. | 
yoti-sdk-spring-boot-auto-config
A module that can be used in Spring Boot applications to automatically configure the YotiClient and KeyPairSource with standard application properties.
yoti-sdk-spring-security
A module that can be used in Spring applications that use Spring Security to add Yoti authentication.
Spring Boot Auto Configuration
As a convenience, if your application happens to use Spring Boot, you can utilise the Spring Boot auto configuration module that will take care of configuring the Yoti Client and Key Pair for you based on standard application properties.
For more information and to see an example of this in use take a look at the Spring Boot Auto Configuration module.
Running the Examples
Instructions on how to run the Spring example projects can be found at the following:
Breaking changes and enhancements made in v3.0.0
This major update does not have any major updates to the API, but instead builds upon and standardizes our implementation.
Dropped support for Java 7
Minimum supported Java version is now 8.
Builder are now in-line classes
Most builders for the request objects are now in-line classes, and can be accessed with a static .builder() method (instead of using factories to instantiate new builders)
Spring Security Integration
If you use Spring Security you can use the yoti-sdk-spring-security module to make integration easier. You are provided with some classes that fit into Spring Security's existing authentication model.
Combining this with the Spring Boot Auto Configuration can make integration very easy with very little code needing to be written.
Misc
- By default, Yoti SDKs fetch profiles from https://api.yoti.com/api/v1. If necessary, this can be overridden by setting the yoti.api.urlsystem property.
- This SDK uses AES-256 encryption. If you are using the Oracle JDK, this key length is not enabled by default. The following stack overflow question explains how to fix this: http://stackoverflow.com/questions/6481627/java-security-illegal-key-size-or-default-parameters
- To find out how to set up your Java project in order to use this SDK, you can check the Spring Boot example in this repo.
- Windows users - if you see unmappable character for encoding Cp1252when runningmvn clean install, you need to set the default encoding to be UTF-8 before proceeding. This can be done by setting theJAVA_TOOL_OPTIONSvariable from the Command Prompt:set JAVA_TOOL_OPTIONS=-Dfile.encoding=UTF8
Known Issues
Age Verification
Affects
- Versions prior to 1.4.2
Description
A bug in the isAgeVerified() property of the profile meant that isAgeVerified() would always return null (i.e. not checked).
How to fix
Use SDK version 1.4.2 or later.
Loading Private Keys
Affects
- Version 1.1 onwards.
Description
There was a known issue with the encoding of RSA private key PEM files that were issued in the past by Yoti Hub (most likely where you downloaded the private key for your application).
Some software is more accepting that others and will have been able to cope with the incorrect encoding, whereas some stricter libraries will not accept this encoding.
At version 1.1 of this client the Java Security Provider that we use (Bouncy Castle) was upgraded from 1.51 -> 1.57. This upgrade appears to have made the key parser more strict in terms of encoding since it no longer accepts these incorrectly encoded keys.
Symptoms
This error usually manifests itself when constructing and instance of the Yoti Client to read the private key.
Generally you'll encounter an exception with an message and stack trace as follows:
com.yoti.api.client.InitialisationException: Cannot load key pair
    at com.yoti.api.client.spi.remote.SecureYotiClient.loadKeyPair(SecureYotiClient.java:99)
    at com.yoti.api.client.spi.remote.SecureYotiClient.<init>(SecureYotiClient.java:73)
    at com.yoti.api.client.spi.remote.SecureYotiClientFactory.getInstance(SecureYotiClientFactory.java:25)
    at com.yoti.api.client.ServiceLocatorYotiClientBuilder.build(ServiceLocatorYotiClientBuilder.java:40)
    at com.yoti.api.spring.YotiClientAutoConfiguration.yotiClient(YotiClientAutoConfiguration.java:48)
Caused by: org.bouncycastle.openssl.PEMException: problem creating RSA private key: java.lang.IllegalArgumentException: failed to construct sequence from byte[]: corrupted stream detected
    at org.bouncycastle.openssl.PEMParser$KeyPairParser.parseObject(Unknown Source)
    at org.bouncycastle.openssl.PEMParser.readObject(Unknown Source)
    at com.yoti.api.client.spi.remote.SecureYotiClient$KeyStreamVisitor.findKeyPair(SecureYotiClient.java:269)
    at com.yoti.api.client.spi.remote.SecureYotiClient$KeyStreamVisitor.accept(SecureYotiClient.java:260)
    at com.yoti.api.spring.SpringResourceKeyPairSource.getFromStream(SpringResourceKeyPairSource.java:28)
    at com.yoti.api.client.spi.remote.SecureYotiClient.loadKeyPair(SecureYotiClient.java:97)
    ... 52 common frames omitted
Caused by: org.bouncycastle.openssl.PEMException: problem creating RSA private key: java.lang.IllegalArgumentException: failed to construct sequence from byte[]: corrupted stream detected
    at org.bouncycastle.openssl.PEMParser$RSAKeyPairParser.parse(Unknown Source)
    ... 58 common frames omitted
Caused by: java.lang.IllegalArgumentException: failed to construct sequence from byte[]: corrupted stream detected
    at org.bouncycastle.asn1.ASN1Sequence.getInstance(Unknown Source)
    ... 59 common frames omitted 
How To Fix
You can re-encode the badly encoded PEM file using some software that is more accepting of the incorrect encoding and saving the new key.
An example of software able to do this is OpenSSL versions 1.0.2g and 1.1.0 using the command:
openssl rsa -in input-file.pem -out fixed-input-file.pem 
Using the new (correctly encoded) file should now be compatible with versions 1.1 onwards (as well as older versions like 1.0 prior to this).
API Coverage
- Activity Details 
  -  Remember Me Id getRememberMeId()
-  Parent Remember Me Id getParentRememberMeId()
-  Timestamp getTimestamp()
-  Receipt ID getReceiptId()
-  User Profile getUserProfile()-  Selfie getSelfie()
-  Given Names getGivenNames()
-  Family Name getFamilyName()
-  Full Name getFullName()
-  Mobile Number getPhoneNumber()
-  Email Address getEmailAddress()
-  Date of Birth getDateOfBirth()
-  Postal Address getPostalAddress()
-  Structured Postal Address getStructuredPostalAddress()
-  Gender getGender()
-  Nationality getNationality()
-  Age Verifications getAgeVerifications()
-  Age Over Verification findAgeOverVerification(int age)
-  Age Under Verification findAgeUnderVerification(int age)
-  Document Details getDocumentDetails()
 
-  Selfie 
-  Application Profile getApplicationProfile()-  Name getApplicationName()
-  URL getApplicationUrl()
-  Logo getApplicationLogo()
-  Receipt Background Color getApplicationReceiptBgColor()
 
-  Name 
 
-  Remember Me Id 
Support
For any questions or support please email sdksupport@yoti.com. Please provide the following to get you up and working as quickly as possible:
- Computer type
- OS version
- Version of Java being used
- Screenshot
Once we have answered your question we may contact you again to discuss Yoti products and services. If you’d prefer us not to do this, please let us know when you e-mail.
 JarCasting
 JarCasting
