diff --git a/.gitignore b/.gitignore index 7a8efa88..98204149 100644 --- a/.gitignore +++ b/.gitignore @@ -18,6 +18,7 @@ /_yardoc/ /doc/ /rdoc/ +/yardoc/ ## Environment normalization: /.bundle/ diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 45ff023a..13cfa768 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -51,7 +51,7 @@ Verifies the documentation with [Yardstick][] and generates the `measurement/rep bundle exec rake yard ``` -Generates [YARD][] documentation in the doc folder. +Generates [YARD][] documentation in the `./yardoc` folder. ### Git diff --git a/README.md b/README.md index 17a7916a..416480a4 100644 --- a/README.md +++ b/README.md @@ -10,39 +10,16 @@ Welcome to the Yoti Ruby SDK. This repository contains the tools you need to qui ## Table of Contents -1. [An Architectural view](#an-architectural-view) - High level overview of integration 1. [Requirements](#requirements) - Everything you need to get started -1. [Installing the SDK](#installing-the-sdk) - How to install our SDK -1. [Configuration](#configuration) - Configuring the SDK -1. [Profile Retrieval](#profile-retrieval) - How to retrieve a Yoti profile using the one time use token -1. [AML Integration](#aml-integration) - How to integrate with Yoti's AML (Anti Money Laundering) service -1. [Running the Examples](#running-the-examples) - How to run the example projects provided -1. [API Coverage](#api-coverage) - Attributes defined -1. [Support](#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 is configured in the [Yoti Hub](https://hub.yoti.com) where you create/update your application. To see an example of how this is configured, see the [Running the Examples](#running-the-examples) section. -The image below shows how your application back-end and Yoti integrate into the context of a Login flow. -Yoti SDK carries out for you steps 6, 7, 8 and the profile decryption in step 9. - -![alt text](login_flow.png "Login flow") - -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. Your back-end doesn't need to handle these cases in a significantly different way, but you might decide to handle the `User-Agent` header in order to provide different responses for desktop and mobile clients. +1. [Installing the SDK](#installing-the-sdk) - How to install our SDK -## References +1. [Setup](#setup) - Setup required before using the Yoti services -* [AES-256 symmetric encryption][] -* [RSA pkcs asymmetric encryption][] -* [Protocol buffers][] -* [Base64 data][] +1. [Products](#products) - +Links to more information about the products offered by the Yoti SDK -[AES-256 symmetric encryption]: https://en.wikipedia.org/wiki/Advanced_Encryption_Standard -[RSA pkcs asymmetric encryption]: https://en.wikipedia.org/wiki/RSA_(cryptosystem) -[Protocol buffers]: https://en.wikipedia.org/wiki/Protocol_Buffers -[Base64 data]: https://en.wikipedia.org/wiki/Base64 +1. [Support](#support) - Please feel free to reach out ## Requirements @@ -76,7 +53,7 @@ bundle install Or simply run the following command from your terminal: ```shell -[sudo] gem install yoti +gem install yoti ``` ## SDK Project Import @@ -89,7 +66,12 @@ rails generate yoti:install The generated initialisation file can be found in `config/initializers/yoti.rb`. -## Configuration +## Setup + +For each service you will need: + +* Your Client SDK ID, generated by [Yoti Hub](https://hub.yoti.com) when you create (and then publish) your app. +* Your .pem file. This is your own unique private key which your browser generates from the [Yoti Hub](https://hub.yoti.com) when you create an application. A minimal Yoti client initialisation looks like this: @@ -101,10 +83,8 @@ end ``` Make sure the following environment variables can be accessed by your app: - -`YOTI_CLIENT_SDK_ID` - found on the Key settings page on your Yoti Hub - -`YOTI_KEY_FILE_PATH` - the full path to your security key downloaded from the *Keys* settings page (e.g. /Users/developer/access-security.pem) +- `YOTI_CLIENT_SDK_ID` +- `YOTI_KEY_FILE_PATH` The following options are available: @@ -144,197 +124,12 @@ heroku config:add YOTI_KEY ="$(cat your-access-security.pem)" [Heroku Command Line]: https://devcenter.heroku.com/articles/heroku-command-line -## Profile Retrieval - -When your application receives a one time use token via the exposed endpoint (it will be assigned to a query string parameter named `token`), you can easily retrieve the user profile: - -```ruby -one_time_use_token = params[:token] -yoti_activity_details = Yoti::Client.get_activity_details(one_time_use_token) -``` - -Before you inspect the user profile, you might want to check whether the user validation was successful. This is done as follows: - -```ruby -if yoti_activity_details.outcome == 'SUCCESS' - profile = yoti_activity_details.profile - given_names = profile.given_names.value - family_name = profile.family_name.value -else - # handle unhappy path -end -``` - -The `profile` object provides a set of attributes corresponding to user attributes. Whether the attributes are present or not depends on the settings you have applied to your app on Yoti Hub. - -### Handling Users - -When you retrieve the user profile, you receive a user ID generated by Yoti exclusively for your application. -This means that if the same individual logs into another app, Yoti will assign her/him a different ID. -You can use this ID to verify whether (for your application) the retrieved profile identifies a new or an existing user. -Here is an example of how this works: - -```ruby -if yoti_activity_details.outcome == 'SUCCESS' - user = your_user_search_function(yoti_activity_details.user_id) - profile = yoti_activity_details.profile - - if user - # handle login - email = profile.email_address.value - else - # handle registration - given_names = profile.given_names.value - family_name = profile.family_name.value - email = profile.email_address.value - end -else - # handle unhappy path -end -``` - -Where `your_user_search_function` is a piece of logic in your app that is supposed to find a user, given a user_id. Regardless of whether the user is a new or an existing one, Yoti will always provide their profile, so you don't necessarily need to store it. - -You can retrieve the sources and verifiers for each attribute as follows: - -```ruby -given_names_sources = profile.given_names.sources # list of anchors -given_names_verifiers = profile.given_names.verifiers # list of anchors -given_names_anchors = profile.given_names.anchors # list of anchors -``` -You can also retrieve further properties from these respective anchors in the following way: - -```ruby -# Retrieving properties of the first anchor -type = given_names_sources[0].type # string -value = given_names_sources[0].value # string -sub_type = given_names_sources[0].sub_type # string -time_stamp = given_names_sources[0].signed_time_stamp.time_stamp # DateTime object -origin_server_certs = given_names_sources[0].origin_server_certs # list of X509 certificates -``` - -In case you want to prove the sources and verifiers of the helper`ActivityDetails.age_verified` on `Age Over 18` set as age derivation, please retrieve it's original attribute from the profile as follow: - -```ruby -age_attribute = profile.get_attribute('age_over:18') -sources = age_attribute.sources -verifiers = age_attribute.verifiers -anchors = age_attribute.anchors -``` - -## AML Integration - -Yoti provides an AML (Anti Money Laundering) check service to allow a deeper KYC process to prevent fraud. This is a chargeable service, so please contact [sdksupport@yoti.com](mailto:sdksupport@yoti.com) for more information. - -Yoti will provide a boolean result on the following checks: - -* PEP list - Verify against Politically Exposed Persons list -* Fraud list - Verify against US Social Security Administration Fraud (SSN Fraud) list -* Watch list - Verify against watch lists from the Office of Foreign Assets Control - -To use this functionality you must ensure your application is assigned to your organisation in the Yoti Hub - please see [here](https://developers.yoti.com/yoti-app-integration/yoti-app-integration#step-1-creating-an-organisation) for further information. - -For the AML check you will need to provide the following: - -* Data provided by Yoti (please ensure you have selected the Given name(s) and Family name attributes for your scenario on the Yoti Hub) - * Given name(s) - * Family name -* Data that must be collected from the user: - * Country of residence (must be an ISO 3166 3-letter code) - * Social Security Number (US citizens only) - * Postcode/Zip code (US citizens only) - -### Consent - -Performing an AML check on a person *requires* their consent. -**You must ensure you have user consent *before* using this service.** - -### Code Example - -Given a YotiClient initialised with your SDK ID and KeyPair (see [Client Initialisation](#client-initialisation)) performing an AML check is a straightforward case of providing basic profile data. - -```ruby -require 'yoti' - -Yoti.configure do |config| - config.client_sdk_id = ENV['YOTI_CLIENT_SDK_ID'] - config.key_file_path = ENV['YOTI_KEY_FILE_PATH'] -end - -aml_address = Yoti::AmlAddress.new('GBR') -aml_profile = Yoti::AmlProfile.new('Edward Richard George', 'Heath', aml_address) - -puts Yoti::Client.aml_check(aml_profile) -``` +## Products -## Running the Examples - -The examples can be found in the [examples folder](examples). - -### Ruby on Rails - -1. Create your application in the [Yoti Hub](https://hub.yoti.com) -1. Set the application domain of your app to `localhost:3001` -1. Set the scenario callback URL to `/profile` -1. Rename the [.env.example](examples/rails/.env.example) file to `.env` -1. Fill in the environment variables in this file with the ones specific to your application (mentioned in the [Configuration](#configuration) section) -1. Install the dependencies by running the following commands - ```ruby - $ bundle install - $ gem install foreman # We are doing this as it's not recommended to include foreman in your Gemfile - ``` -1. Start the server `foreman start` - -Visiting `https://localhost:3001/` should show a Yoti Connect button - -### Sinatra - -1. Create your application in the [Yoti Hub](https://hub.yoti.com) -1. Set the application domain of your app to `localhost:4567` -1. Set the scenario callback URL to `/profile` -1. Rename the [.env.example](examples/sinatra/.env.example) file to `.env` -1. Fill in the environment variables in this file with the ones specific to your application (mentioned in the [Configuration](#configuration) section) -1. Install the dependencies by running the following commands - ```ruby - $ bundle install - $ gem install foreman # We are doing this as it's not recommended to include foreman in your Gemfile - ``` -1. Start the server `foreman start` - -Visiting `https://localhost:4567/` should show a Yoti Connect button - -### AML Check - -* rename the [.env.example](examples/aml_check/.env.example) file to `.env` and fill in the required configuration values -* install the dependencies with `bundle install` -* run the script with `ruby ./app.rb` - -## API Coverage - -* Activity Details - * [X] Remember Me ID `remember_me_id` - * [X] Parent Remember Me ID `parent_remember_me_id` - * [X] Receipt ID `receipt_id` - * [X] Timestamp `timestamp` - * [X] Base64 Selfie URI `base64_selfie_uri` - * [X] Age verified `age_verified` - * [X] Profile `profile` - * [X] Selfie `selfie` - * [X] Full Name `full_name` - * [X] Given Names `given_names` - * [X] Family Name `family_name` - * [X] Mobile Number `phone_number` - * [X] Email Address `email_address` - * [X] Age / Date of Birth `date_of_birth` - * [X] Address `postal_address` - * [X] Structured Postal Address `structured_postal_address` - * [X] Gender `gender` - * [X] Nationality `nationality` - * [X] Application Profile `application_profile` - * [X] Name `name` - * [X] URL `url` - * [X] Logo `logo` - * [X] Receipt Background Color `receipt_bgcolor` +The Yoti SDK can be used for the following products, follow the links for more information about each: +1) [Yoti app integration](/docs/PROFILE.md) - Connect with already-verified customers. +1) [Yoti Doc Scan](/docs/DOCSCAN.md) - Identity verification embedded in your website or app. +1) [Yoti AML](/docs/AML.md) - Anti-Money Laundering check service to allow a deeper KYC process, preventing fraud ## Support diff --git a/Rakefile b/Rakefile index 15dc2958..1c7d8b9a 100644 --- a/Rakefile +++ b/Rakefile @@ -31,6 +31,7 @@ end require 'yard' YARD::Rake::YardocTask.new do |t| + t.options = ['--output-dir', './yardoc'] t.stats_options = ['--list-undoc'] end diff --git a/docs/AML.md b/docs/AML.md new file mode 100644 index 00000000..90733453 --- /dev/null +++ b/docs/AML.md @@ -0,0 +1,51 @@ +# AML Integration + +## About + +Yoti provides an AML (Anti Money Laundering) check service to allow a deeper KYC process to prevent fraud. This is a chargeable service, so please contact [sdksupport@yoti.com](mailto:sdksupport@yoti.com) for more information. + +Yoti will provide a boolean result on the following checks: + +* PEP list - Verify against Politically Exposed Persons list +* Fraud list - Verify against US Social Security Administration Fraud (SSN Fraud) list +* Watch list - Verify against watch lists from the Office of Foreign Assets Control + +To use this functionality you must ensure your application is assigned to your Organisation in the Yoti Hub - please see [here](https://developers.yoti.com/yoti-app/web-integration#step-1-creating-an-organisation) for further information. + +For the AML check you will need to provide the following: + +* Data provided by Yoti + * Given name(s) + * Family name + * Postcode/Zip code (US citizens only) - this can be retrieved from the Structured Postal Address attribute + +* Data that must be collected from the user: + * Country of residence (must be an ISO 3166 3-letter code) + * Social Security Number (US citizens only) + +## Consent + +Performing an AML check on a person *requires* their consent. +**You must ensure you have user consent *before* using this service.** + +## Performing a Check + +Performing an AML check is a straightforward case of providing basic profile data. + +```ruby +require 'yoti' + +Yoti.configure do |config| + config.client_sdk_id = ENV['YOTI_CLIENT_SDK_ID'] + config.key_file_path = ENV['YOTI_KEY_FILE_PATH'] +end + +aml_address = Yoti::AmlAddress.new('GBR') +aml_profile = Yoti::AmlProfile.new('Edward Richard George', 'Heath', aml_address) + +puts Yoti::Client.aml_check(aml_profile) +``` + +## Running the example + +- See the [AML Example](../examples/aml_check/README.md) folder for instructions on how to run the AML Example project diff --git a/docs/DOCSCAN.md b/docs/DOCSCAN.md new file mode 100644 index 00000000..26eba9a3 --- /dev/null +++ b/docs/DOCSCAN.md @@ -0,0 +1,11 @@ +# Yoti Doc Scan + +## About + +Yoti Doc Scan can be seamlessly integrated with your website, app or custom product so you can perform secure identity checks. You'll be able to request specific ID documents from users directly from your website or app. + +See the the [Developer Docs](https://developers.yoti.com/yoti/getting-started-docscan) for more information. + +## Running the example + +- See the [Doc Scan Example](../examples/doc_scan/README.md) folder for instructions on how to run the Doc Scan Example project diff --git a/docs/PROFILE.md b/docs/PROFILE.md new file mode 100644 index 00000000..42dbf6eb --- /dev/null +++ b/docs/PROFILE.md @@ -0,0 +1,140 @@ +Ruby Yoti App Integration +============================= + +1) [An Architectural View](#an-architectural-view) - +High level overview of integration + +1) [Profile Retrieval](#profile-retrieval) - +Description on setting up profile + +1) [Handling Users](#handling-users) - +Description on handling user details + +1) [Running the Examples](#running-the-examples) +How to run the example project provided + +1) [API Coverage](#api-coverage) - +Attributes defined + +## 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. +Yoti SDK carries out for you steps 6, 7 ,8 and the profile decryption in step 9. + +![alt text](login_flow.png "Login flow") + +Yoti also allows you to enable user details verification from your mobile app by means of the [Android](https://github.com/getyoti/android-sdk-button) and [iOS](https://github.com/getyoti/ios-sdk-button) 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. + +## Profile Retrieval + +When your application receives a one time use token via the exposed endpoint (it will be assigned to a query string parameter named `token`), you can easily retrieve the user profile: + +```ruby +one_time_use_token = params[:token] +yoti_activity_details = Yoti::Client.get_activity_details(one_time_use_token) +``` + +Before you inspect the user profile, you might want to check whether the user validation was successful. This is done as follows: + +```ruby +if yoti_activity_details.outcome == 'SUCCESS' + profile = yoti_activity_details.profile + given_names = profile.given_names.value + family_name = profile.family_name.value +else + # handle unhappy path +end +``` + +The `profile` object provides a set of attributes corresponding to user attributes. Whether the attributes are present or not depends on the settings you have applied to your app on Yoti Hub. + +### Handling Users + +When you retrieve the user profile, you receive a user ID generated by Yoti exclusively for your application. +This means that if the same individual logs into another app, Yoti will assign her/him a different ID. +You can use this ID to verify whether (for your application) the retrieved profile identifies a new or an existing user. +Here is an example of how this works: + +```ruby +if yoti_activity_details.outcome == 'SUCCESS' + user = your_user_search_function(yoti_activity_details.user_id) + profile = yoti_activity_details.profile + + if user + # handle login + email = profile.email_address.value + else + # handle registration + given_names = profile.given_names.value + family_name = profile.family_name.value + email = profile.email_address.value + end +else + # handle unhappy path +end +``` + +Where `your_user_search_function` is a piece of logic in your app that is supposed to find a user, given a user_id. Regardless of whether the user is a new or an existing one, Yoti will always provide their profile, so you don't necessarily need to store it. + +You can retrieve the sources and verifiers for each attribute as follows: + +```ruby +given_names_sources = profile.given_names.sources # list of anchors +given_names_verifiers = profile.given_names.verifiers # list of anchors +given_names_anchors = profile.given_names.anchors # list of anchors +``` +You can also retrieve further properties from these respective anchors in the following way: + +```ruby +# Retrieving properties of the first anchor +type = given_names_sources[0].type # string +value = given_names_sources[0].value # string +sub_type = given_names_sources[0].sub_type # string +time_stamp = given_names_sources[0].signed_time_stamp.time_stamp # DateTime object +origin_server_certs = given_names_sources[0].origin_server_certs # list of X509 certificates +``` + +In case you want to prove the sources and verifiers of the helper`ActivityDetails.age_verified` on `Age Over 18` set as age derivation, please retrieve it's original attribute from the profile as follow: + +```ruby +age_attribute = profile.get_attribute('age_over:18') +sources = age_attribute.sources +verifiers = age_attribute.verifiers +anchors = age_attribute.anchors +``` + +## Running the Examples + +Follow the below links for instructions on how to run the example projects: +- [Rails](../examples/rails/README.md) +- [Sinatra](../examples/sinatra/README.md) + +## API Coverage + +* Activity Details + * [X] Remember Me ID `remember_me_id` + * [X] Parent Remember Me ID `parent_remember_me_id` + * [X] Receipt ID `receipt_id` + * [X] Timestamp `timestamp` + * [X] Base64 Selfie URI `base64_selfie_uri` + * [X] Age verified `age_verified` + * [X] Profile `profile` + * [X] Selfie `selfie` + * [X] Full Name `full_name` + * [X] Given Names `given_names` + * [X] Family Name `family_name` + * [X] Mobile Number `phone_number` + * [X] Email Address `email_address` + * [X] Age / Date of Birth `date_of_birth` + * [X] Address `postal_address` + * [X] Structured Postal Address `structured_postal_address` + * [X] Gender `gender` + * [X] Nationality `nationality` + * [X] Application Profile `application_profile` + * [X] Name `name` + * [X] URL `url` + * [X] Logo `logo` + * [X] Receipt Background Color `receipt_bgcolor` diff --git a/login_flow.png b/docs/login_flow.png similarity index 100% rename from login_flow.png rename to docs/login_flow.png diff --git a/examples/aml_check/README.md b/examples/aml_check/README.md new file mode 100644 index 00000000..ecf83f9f --- /dev/null +++ b/examples/aml_check/README.md @@ -0,0 +1,5 @@ +# AML Example + +* rename the [.env.example](examples/aml_check/.env.example) file to `.env` and fill in the required configuration values +* install the dependencies with `bundle install` +* run the script with `ruby ./app.rb` diff --git a/examples/doc_scan/README.md b/examples/doc_scan/README.md index 100d26ad..860f94e5 100644 --- a/examples/doc_scan/README.md +++ b/examples/doc_scan/README.md @@ -1,4 +1,4 @@ -### Doc Scan Example - Ruby on Rails +# Doc Scan Example - _Ruby on Rails_ 1. Create your application in the [Yoti Hub](https://hub.yoti.com) 1. Set the application domain of your app to `localhost:3002` diff --git a/examples/rails/README.md b/examples/rails/README.md new file mode 100644 index 00000000..eb7716d6 --- /dev/null +++ b/examples/rails/README.md @@ -0,0 +1,15 @@ +# Profile Example - _Ruby on Rails_ + +1. Create your application in the [Yoti Hub](https://hub.yoti.com) +1. Set the application domain of your app to `localhost:3001` +1. Set the scenario callback URL to `/profile` +1. Rename the [.env.example](examples/rails/.env.example) file to `.env` +1. Fill in the environment variables in this file with the ones specific to your application, generated in the Yoti Hub when you create (and then publish) your application +1. Install the dependencies by running the following commands + ```ruby + $ bundle install + $ gem install foreman # We are doing this as it's not recommended to include foreman in your Gemfile + ``` +1. Start the server `foreman start` + +Visiting `https://localhost:3001/` should show a Yoti Connect button diff --git a/examples/sinatra/README.md b/examples/sinatra/README.md new file mode 100644 index 00000000..e51bc363 --- /dev/null +++ b/examples/sinatra/README.md @@ -0,0 +1,15 @@ +# Profile Example - _Sinatra_ + +1. Create your application in the [Yoti Hub](https://hub.yoti.com) +1. Set the application domain of your app to `localhost:4567` +1. Set the scenario callback URL to `/profile` +1. Rename the [.env.example](examples/sinatra/.env.example) file to `.env` +1. Fill in the environment variables in this file with the ones specific to your application, generated in the Yoti Hub when you create (and then publish) your application +1. Install the dependencies by running the following commands + ```ruby + $ bundle install + $ gem install foreman # We are doing this as it's not recommended to include foreman in your Gemfile + ``` +1. Start the server `foreman start` + +Visiting `https://localhost:4567/` should show a Yoti Connect button diff --git a/lib/yoti/http/request.rb b/lib/yoti/http/request.rb index 70c73cc7..de1234f6 100644 --- a/lib/yoti/http/request.rb +++ b/lib/yoti/http/request.rb @@ -1,4 +1,5 @@ require 'securerandom' +require 'cgi' module Yoti # Manage the API's HTTPS requests diff --git a/lib/yoti/version.rb b/lib/yoti/version.rb index 9b04cf8d..8cac6795 100644 --- a/lib/yoti/version.rb +++ b/lib/yoti/version.rb @@ -1,4 +1,4 @@ module Yoti # @return [String] the gem's current version - VERSION = '1.7.0'.freeze + VERSION = '1.7.1'.freeze end diff --git a/sonar-project.properties b/sonar-project.properties index 17fc999e..59538cb8 100644 --- a/sonar-project.properties +++ b/sonar-project.properties @@ -3,10 +3,11 @@ sonar.organization = getyoti sonar.projectKey = getyoti:ruby sonar.projectName = Ruby SDK -sonar.projectVersion = 1.7.0 +sonar.projectVersion = 1.7.1 -sonar.language=ruby +sonar.language = ruby sonar.exclusions = **/protobuf/**.rb, coverage/**, spec/sample-data/** +sonar.coverage.exclusions = lib/yoti/version.rb, lib/generators/yoti/install/templates/yoti.rb sonar.links.scm = https://github.com/getyoti/yoti-ruby-sdk sonar.sources = ./lib sonar.ruby.coverage.reportPaths = coverage/.resultset.json diff --git a/yoti.gemspec b/yoti.gemspec index ebfbd629..a2293c89 100644 --- a/yoti.gemspec +++ b/yoti.gemspec @@ -5,8 +5,8 @@ require 'yoti/version' Gem::Specification.new do |spec| spec.name = 'yoti' spec.version = Yoti::VERSION - spec.authors = ['Sebastian Zaremba'] - spec.email = ['tech@yoti.com'] + spec.authors = ['Yoti'] + spec.email = ['websdk@yoti.com'] spec.summary = 'Yoti Ruby SDK for back-end integration.' spec.description = <<-DESC @@ -18,7 +18,7 @@ Gem::Specification.new do |spec| spec.homepage = 'https://github.com/getyoti/yoti-ruby-sdk' spec.license = 'MIT' - spec.files = `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features|examples)/|^sonar-project.properties$|^.dependabot/config.yml$|^.travis.yml$}) } + spec.files = `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features|examples|docs)/|^sonar-project.properties$|^.dependabot/config.yml$|^.travis.yml$}) } spec.require_paths = ['lib'] spec.required_ruby_version = '>= 2.4'