Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/GoogleCloudPlatform/auto-data-tokenize

Identify and tokenize sensitive data automatically using Cloud DLP and Dataflow
https://github.com/GoogleCloudPlatform/auto-data-tokenize

cloud-migration data-governance data-loss-prevention dataflow deidentification tokenization

Last synced: 2 months ago
JSON representation

Identify and tokenize sensitive data automatically using Cloud DLP and Dataflow

Awesome Lists containing this project

README

        

# Automatic data tokenizing pipelines

[![codecov](https://codecov.io/gh/GoogleCloudPlatform/auto-data-tokenize/branch/main/graph/badge.svg?token=2BUEEXNC1H)](https://codecov.io/gh/GoogleCloudPlatform/auto-data-tokenize)

This document discusses how to identify and tokenize data with an automated data transformation pipeline to detect sensitive data like personally identifiable information (PII), using Cloud Data Loss Prevention [(Cloud DLP)](https://cloud.google.com/dlp) and [Cloud KMS](https://cloud.google.com/kms). De-identification techniques like encryption lets you preserve the utility of your data for joining or analytics while reducing the risk of handling the data by obfuscating the raw sensitive identifiers.

To minimize the risk of handling large volumes of sensitive data, you can use an automated data transformation pipeline to create de-identified datasets that can be used for migrating from on-premise to cloud or keep a de-identified replica for Analytics. Cloud DLP can inspect the data for sensitive information when the dataset has not been characterized, by using [more than 100 built-in classifiers](https://cloud.google.com/dlp/docs/infotypes-reference).

One of the daunting challenges during data migration to cloud is to manage sensitive data. The sensitive data can be in structured forms like analytics tables or unstructured like chat history or transcription records. One needs to use Cloud DLP to identify sensitive data from both of these kinds of sources, followed by tokenizing the sensitive parts.

Tokenizing structured data can be optimized for cost and speed by using representative samples for each of the columns to categorize the kind of information, followed by bulk encryption of sensitive columns. This approach reduces cost of using Cloud DLP, by limiting the use to classification of a small representative sample, instead of all the records. The throughput and cost of tokenization can be optimized by using envelope encryption for columns classified as sensitive.

This document demonstrates a reference implementation of tokenizing structured data through two tasks: _sample and identify_, followed by _bulk tokenization_ using encryption.

This document is intended for a technical audience whose responsibilities include data security, data processing, or data analytics. This guide assumes that you're familiar with data processing and data privacy, without the need to be an expert.

Watch the video to learn how the tool works:

[![Understand code Youtube video](https://img.youtube.com/vi/S6fYkWvUBDo/default.jpg)](https://www.youtube.com/watch?v=S6fYkWvUBDo)

## Architecture

The solution comprises two pipelines (one for each of the tasks):
1. Sample + Identify
1. Tokenize

### Sample & Identify Pipeline

![Identify pipeline](sampling_dlp_identify_catalog_architecture.svg)

The __sample & identify pipeline__ extracts a few sample records from the source files. The *identify* part of pipeline then decomposes each sample record into columns to categorize them into one of the [in-built infotypes](https://cloud.google.com/dlp/docs/infotypes-reference) or [custom infotypes](https://cloud.google.com/dlp/docs/creating-custom-infotypes) using Cloud DLP. The sample & identify pipeline outputs following files to Cloud Storage:
* Avro schema of the file
* Detected info-types for each of the input columns.

### Tokenization Pipeline

![Tokenization pipeline](AutoDLP_Encryption_Catalog_Architecture.svg)

The __tokenize pipeline__ then encrypts the user-specified source columns using the schema information from the _sample & identify pipeline_ and user provided enveloped data encryption key. The tokenizing pipeline performs following of transforms on each of the records in the source file:
1. Unwrap data-encryption key using Cloud KMS
1. un-nest each record into a flat record.
1. tokenize required values using [deterministic AEAD](https://github.com/google/tink/blob/master/docs/PRIMITIVES.md#deterministic-authenticated-encryption-with-associated-data) encryption.
1. re-nest the flat record into Avro record
1. Write Avro file with encrypted fields

### Concepts
* [Envelope Encryption](https://cloud.google.com/kms/docs/envelope-encryption) is a form of multi-layer encryption, involving use of multiple layers of keys for encrypting data. It is the process of encrypting the actual data encryption key with another key to secure the data-encryption key.
* [Cloud KMS](https://cloud.google.com/kms) provides easy management of encryption keys at scale.
* [Tink](https://github.com/google/tink) is an open-source library that provides easy and secure APIs for handling encryption/decryption. It
reduces common crypto pitfalls with user-centered design, careful implementation and code reviews, and extensive
testing. At Google, Tink is one of the standard crypto libraries, and has been deployed in hundreds of products and
systems. Tink natively integrates with Cloud KMS for use with envelope encryption technique.
* [Deterministic AEAD encryption](https://github.com/google/tink/blob/master/docs/PRIMITIVES.md#deterministic-authenticated-encryption-with-associated-data) is used by to serve following purposes:
1. Permits use of the cipher-text as join keys. The deterministic property of the cipher ensures that cipher-text for the same plain-text is always the same. Using this property one can safely use the encrypted data for performing statistical analysis like cardinality analysis, frequency analysis etc.
1. store signed plain-text within the cipher to assert authenticity.
1. reversibility, use of 2-way encryption algorithm permits reversing the algorithm to obtain original plain-text. Hashing does not permit such operations.

* [Cloud Data Loss Prevention](https://cloud.google.com/dlp) is a Google Cloud service providing data classification, de-identification and re-identification features, allowing you to easily manage sensitive data in your enterprise.

* __Record Flattening__ is the process of converting nested/repeated records as flat table. Each leaf-node of the record gets a unique identifier. This flattening process enables sending data to DLP for identification purposes as the DLP API supports a simple [data-table](https://cloud.google.com/dlp/docs/examples-deid-tables).

Consider a contact record, for **Jane Doe**, it has a nested and repeated field `contacts`.
```json
{
"name": "Jane Doe",
"contacts": [
{
"type": "WORK",
"number": 2124567890
},
{
"type": "HOME",
"number": 5304321234
}
]
}
```

Flattening this record yields a [FlatRecord](https://github.com/GoogleCloudPlatform/auto-data-tokenize/blob/master/proto-messages/src/main/resources/proto/google/cloud/autodlp/auto_tokenize_messages.proto#L103) with following data. Notice the `values` map, which demonstrates that each leaf node of the contact record is mapped using a [JsonPath](https://goessner.net/articles/JsonPath/) notation.
The `keySchema` shows a mapping between the leaf value's key's to a schema key to demonstrate that leaf-nodes of same type share the same key-schema, for example: `$.contacts[0].contact.number` is logically same as `$.contacts[1].contact.number` as both of them have the same key-schema `$.contacts.contact.number`.

```json
{
"values": {
"$.name": "Jane Doe",
"$.contacts[0].contact.type": "WORK",
"$.contacts[0].contact.number": 2124567890,
"$.contacts[1].contact.type": "WORK",
"$.contacts[1].contact.number": 5304321234
},

"keySchema": {
"$.name": "$.name",
"$.contacts[0].contact.type": "$.contacts.contact.type",
"$.contacts[0].contact.number": "$.contacts.contact.number",
"$.contacts[1].contact.type": "$.contacts.contact.type",
"$.contacts[1].contact.number": "$.contacts.contact.number"
}
}
```

## Prerequisites

This tutorial assumes some familiarity with shell scripts and basic knowledge of Google Cloud.

## Objectives

1. Understand record sampling and identify sensitive columns using DLP.
1. Use of symmetric encryption to tokenize data using KMS wrapped data-encryption key.

## Costs

This tutorial uses billable components of Google Cloud, including the following:

* [Dataflow](https://cloud.google.com/dataflow/pricing)
* [Cloud Storage](https://cloud.google.com/storage/pricing)
* [Cloud Data Loss Prevention](https://cloud.google.com/dlp/pricing)
* [Cloud KMS](https://cloud.google.com/kms/pricing)

Use the [pricing calculator](https://cloud.google.com/products/calculator) to generate a cost estimate based on your
projected usage.

## Before you begin

For this tutorial, you need a Google Cloud [project](https://cloud.google.com/resource-manager/docs/cloud-platform-resource-hierarchy#projects). To make
cleanup easiest at the end of the tutorial, we recommend that you create a new project for this tutorial.

1. [Create a Google Cloud project](https://console.cloud.google.com/projectselector2/home/dashboard).
1. Make sure that [billing is enabled](https://support.google.com/cloud/answer/6293499#enable-billing) for your Google
Cloud project.
1. [Open Cloud Shell](https://console.cloud.google.com/?cloudshell=true).

At the bottom of the Cloud Console, a [Cloud Shell](https://cloud.google.com/shell/docs/features) session opens and
displays a command-line prompt. Cloud Shell is a shell environment with the Cloud SDK already installed, including
the [gcloud](https://cloud.google.com/sdk/gcloud/) command-line tool, and with values already set for your current
project. It can take a few seconds for the session to initialize.

1. Enable APIs for Cloud DLP, Cloud KMS, Compute Engine, Cloud Storage, Dataflow and BigQuery services with the following command:
```shell script
gcloud services enable \
dlp.googleapis.com \
cloudkms.googleapis.com \
compute.googleapis.com \
storage.googleapis.com \
dataflow.googleapis.com \
bigquery.googleapis.com
```

## Setting up your environment

1. In Cloud Shell, clone the source repository and go to the directory for this tutorial:
```shell script
git clone https://github.com/GoogleCloudPlatform/auto-data-tokenize.git
cd auto-data-tokenize/
```

1. Use a text editor of your choice to modify the `set_variables.sh` file to set required environment variables.
```shell script
# The Google Cloud project to use for this tutorial
export PROJECT_ID=""

# The Compute Engine region to use for running Dataflow jobs and create a
# temporary storage bucket
export REGION_ID=""

# define the GCS bucket to use as temporary bucket for Dataflow
export TEMP_GCS_BUCKET=""

# Name of the service account to use (not the email address)
export DLP_RUNNER_SERVICE_ACCOUNT_NAME=""

# Name of the GCP KMS key ring name
export KMS_KEYRING_ID=""

# name of the symmetric Key encryption kms-key-id
export KMS_KEY_ID=""

# The JSON file containing the TINK Wrapped data-key to use for encryption
export WRAPPED_KEY_FILE=""
````

1. Run the script to set the environment variables:
```shell script
source set_variables.sh
```

## Creating resources

The tutorial uses following resources
* _Service account_ to run data flow pipelines enabling fine-grain access control
* A symmetric Cloud KMS managed _Key Encryption key_, this is used to wrap the actual data encryption key
* _Cloud Storage bucket_ for temporary data storage and test data

### Create service accounts

We recommend that you run pipelines with fine-grained access control to improve access partitioning. If
your project does not have a user-created service account, create one using following instructions.

You can use your browser by going to [**Service
accounts**](https://console.cloud.google.com/projectselector/iam-admin/serviceaccounts?supportedpurview=project)
in the Cloud Console.

1. Create a service account to use as the user-managed controller service account for Dataflow:
```shell script
gcloud iam service-accounts create ${DLP_RUNNER_SERVICE_ACCOUNT_NAME} \
--project="${PROJECT_ID}" \
--description="Service Account for Tokenizing pipelines." \
--display-name="Tokenizing pipelines"
```
1. Create a custom role with required permissions for accessing DLP, Dataflow and KMS:
```shell script
export TOKENIZING_ROLE_NAME="tokenizing_runner2"

gcloud iam roles create ${TOKENIZING_ROLE_NAME} \
--project=${PROJECT_ID} \
--file=tokenizing_runner_permissions.yaml
```

1. Apply the custom role to the service account:
```shell script
gcloud projects add-iam-policy-binding ${PROJECT_ID} \
--member="serviceAccount:${DLP_RUNNER_SERVICE_ACCOUNT_EMAIL}" \
--role=projects/${PROJECT_ID}/roles/${TOKENIZING_ROLE_NAME}
```
1. Assign the `dataflow.worker` role to allow the service account to allow it to run as a Dataflow worker:
```shell script
gcloud projects add-iam-policy-binding ${PROJECT_ID} \
--member="serviceAccount:${DLP_RUNNER_SERVICE_ACCOUNT_EMAIL}" \
--role=roles/dataflow.worker
```

### Create Key encryption key

The data would be encrypted using a Data Encryption Key (DEK). You will
use [envelope encryption](https://cloud.google.com/kms/docs/envelope-encryption) technique to encrypt the DEK using a
Key in [Cloud KMS](https://cloud.google.com/kms), this ensures that the DEK can be safely stored without compromising
it.

1. Create KMS Key-ring
```shell script
gcloud kms keyrings create --project ${PROJECT_ID} --location ${REGION_ID} ${KMS_KEYRING_ID}
```
1. Create KMS symmetric key to use for encrypting your data encryption key.
```shell script
gcloud kms keys create --project ${PROJECT_ID} --keyring=${KMS_KEYRING_ID} --location=${REGION_ID} --purpose="encryption" ${KMS_KEY_ID}
```
1. Download and unpack the latest version of [Tinkey](https://github.com/google/tink/blob/master/docs/TINKEY.md). Tinkey
is an open source utility to create wrapped encryption keys.
```shell script
mkdir tinkey/
tar zxf tinkey-.tar.gz -C tinkey/
export TINKEY="${PWD}/tinkey/tinkey"
alias tinkey="${TINKEY}"
```

1. Create a new wrapped data encryption key.
```shell script
tinkey create-keyset \
--master-key-uri="${MAIN_KMS_KEY_URI}" \
--key-template=AES256_SIV \
--out="${WRAPPED_KEY_FILE}" \
--out-format=json
```

### Create Cloud Storage bucket

Create a Cloud Storage bucket for storing test data and Dataflow staging location.

```shell script
gsutil mb -p ${PROJECT_ID} -l ${REGION_ID} "gs://${TEMP_GCS_BUCKET}"
```

### Copy test data to Cloud Storage

You can use your own file datasets or copy the included demo dataset (`userdata.avro` or `userdata.parquet`).

```shell script
gsutil cp userdata.avro gs://${TEMP_GCS_BUCKET}
```

## Compile modules

### Prerequisites

The solution uses[Testcontainers](https://www.testcontainers.org/) for database dependent unit-tests.
If you wish to skip running tests add `-x test` flag to `gradle` command.

Install Docker:

1. Follow [distro specific steps](https://docs.docker.com/engine/install/)
1. For security purpose [setup Docker in Root-less mode](https://docs.docker.com/engine/security/rootless/#prerequisites)
1. Install Docker Compose using [steps](https://docs.docker.com/compose/install/)

Check the [supported](https://www.testcontainers.org/supported_docker_environment/) Docker versions for TestContainers before installing.

### Compile

You need to compile all the modules to build executables for deploying the _sample & identify_ and _bulk tokenize_ pipelines.

```shell script
./gradlew clean buildNeeded shadowJar
```

## Run Sample and Identify pipeline

Run the sample & identify pipeline to identify sensitive columns in the data you need to tokenize.

The pipeline extracts `sampleSize` number of records, flattens the record and identifies sensitive columns
using [Data Loss Prevention (DLP)](https://cloud.google.com/dlp). Cloud DLP provides functionality
to [identify](https://cloud.google.com/dlp/docs/inspecting-text) sensitive information-types. The DLP identify methods
supports only flat tables, hence the pipeline flattens the Avro/Parquet records as they can contain nested and/or repeated fields.

### Launch sample & identify pipeline

```shell script
sample_and_identify_pipeline --project="${PROJECT_ID}" \
--region="${REGION_ID}" \
--runner="DataflowRunner" \
--serviceAccount=${DLP_RUNNER_SERVICE_ACCOUNT_EMAIL} \
--gcpTempLocation="gs://${TEMP_GCS_BUCKET}/temp" \
--stagingLocation="gs://${TEMP_GCS_BUCKET}/staging" \
--tempLocation="gs://${TEMP_GCS_BUCKET}/bqtemp" \
--workerMachineType="n1-standard-1" \
--sampleSize=500 \
--sourceType="AVRO" \
--inputPattern="gs://${TEMP_GCS_BUCKET}/userdata.avro" \
--reportLocation="gs://${TEMP_GCS_BUCKET}/dlp_report/"
```

> **Note:** Use `sampleSize=0` to process all records.

The pipeline supports multiple **Source Types**, use the following table to use the right combination of `sourceType` and `inputPattern` arguments.

| Data source | sourceType | inputPattern |
|------------------------------------------------------|------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------|
| **Avro** file in Cloud Storage | `AVRO` | `gs://:.` |
| Query results in BigQuery | `BIGQUERY_QUERY` | BigQuery SQL statement in StandardSQL dialect. |
| Relational Databases (using JDBC) | `JDBC_TABLE` | `[TABLE_NAME]`, use parameters `jdbcConnectionUrl` and `jdbcDriverClass` to specify the JDBC connection details. ([Details](sample_identify_and_tag.md)) |
| Query Results from Relational Databases (using JDBC) | `JDBC_QUERY` | `SELECT` query, use parameters `jdbcConnectionUrl` and `jdbcDriverClass` to specify the JDBC connection details. |

The pipeline detects all the [standard infotypes](https://cloud.google.com/dlp/docs/infotypes-reference) supported by DLP.
Use `--observableInfoTypes` to provide additional custom info-types that you need.

Pipeline options for Sample, Identify and Tag pipeline (`DlpSamplerPipelineOptions`):

| Sampling Pipeline Options | Description |
|--------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| `sourceType` | The data source to analyse/inspect. One of: \[AVRO, PARQUET, BIGQUERY_TABLE, BIGQUERY_QUERY, JDBC_TABLE\] |
| `inputPattern` | The location of the datasource: for AVRO or PARQUET, the GCS file pattern to use as input.
For BIGQUERY_TABLE: Fully Qualified table name as {projectId}:{datasetId}.{tableId} format, for JDBC_TABLE, the name of the table. |
| `sampleSize` | (Optional) The sample size to send to DLP. (Default:1000) |
| `reportLocation` | (Optional) The GCS location to write the aggregated inspection results and the datasource's AVRO Schema. Atleast one of `reportLocation` or `reportBigQueryTable` must be specified. |
| `reportBigQueryTable` | (Optional) The BigQuery table ({projectId}:{datasetId}.{tableId}) to write the aggregated inspection results, the table must exist. Atleast one of reportLocation or reportBigQueryTable must be specified. |
| `observableInfoTypes` | (Optional) Provide a list of DLP InfoTypes to inspect the data with. Keeping EMPTY uses all DLP supported InfoTypes. |
| `jdbcConnectionUrl` | The Connection URL used for connecting to a SQL datasource using JDBC. (Required when `sourceType=JDBC_TABLE`) |
| `jdbcDriverClass` | The JDBC driver to use for reading from SQL datasource. (Required when `sourceType=JDBC_TABLE`) |
| `jdbcFilterClause` | When using JDBC source, it is highly recommended to use a sampling filter to select random records, instead of fetching all the records from a relational database. The provided string is set as the WHERE clause of the query. (Optional when `sourceType=JDBC_TABLE`) |
| `csvHeaders` | (Optional) Provide column names when using `CSV_FILE` sourceType. |
| `csvFirstRowHeaders` | (Optional) Direct to omit first row of each CSV file and use as header. (Default: `false`) |
| `csvCharset` | (Optional) Specify the charset used for the CSV file. (Default: `UTF-8`) |
| `csvColumnDelimiter` | (Optional) Specify the character(s) used for delimiting column in a row. (Default: `,`) |
| `csvFormatType` | (Optional) Specify the CSV format based on Apache Commons CSV. Choose from one of the [CSVFormat#Predefined](https://github.com/apache/commons-csv/blob/f6cdeac129665cf6f131b00678c9b4e824d758e5/src/main/java/org/apache/commons/csv/CSVFormat.java#L679) types. (Default: `Default`) |
| `dataCatalogEntryGroupId` | The Entry Group Id (/projects/{projectId}/locations/{locationId}/entryGroups/{entryGroupId}) to create a new Entry for inspected datasource. Provide to enable pipeline to create new entry in DataCatalog with schema. (Not used for `sourceType=BIGQUERY_TABLE`) |
| `dataCatalogInspectionTagTemplateId` | The Datacatalog TempalteId to use for creating the sensitivity tags. |
| `dataCatalogForcedUpdate` | Force updates to Data Catalog Tags/Entry based on execution of this pipeline. (Default: `false`) |
| `dlpRegion` | The DLP [processing location](https://cloud.google.com/dlp/docs/locations#regions) to use. (Default: `global`) |

### Sample & Identify pipeline DAG

The Dataflow execution DAG would look like following:

![Sample and Identify Pipeline DAG](sampling_pipeline_with_catalog_jdbc.png)

### Retrieve report

The sample & identify pipeline outputs the Avro schema (or converted for Parquet) of the files and one file for each of the columns detected to contain sensitive information. Retrieve the report to your local machine to have a look.

```shell script
mkdir -p dlp_report/ && rm dlp_report/*.json
gsutil -m cp "gs://${TEMP_GCS_BUCKET}/dlp_report/*.json" dlp_report/
```

List all the column names that have been identified.

```shell script
cat dlp_report/col-*.json | jq .columnName
```
The output will match the following list.

```text
"$.kylosample.birthdate"
"$.kylosample.cc"
"$.kylosample.email"
"$.kylosample.first_name"
"$.kylosample.ip_address"
"$.kylosample.last_name"
```

You can view the details of the identified column by issuing `cat` command for the file.

```shell script
cat dlp_report/col-kylosample-cc-00000-of-00001.json
```

Following is a snippet of the `cc` column.
```json
{
"columnName": "$.kylosample.cc",
"infoTypes": [
{
"infoType": "CREDIT_CARD_NUMBER",
"count": "394"
}
]
}
```

> The `"count"` value will vary based on the randomly selected samples during execution.

## Launch bulk tokenize pipeline

The sample & identify pipeline used few samples from the original dataset to identify sensitive information using
DLP. The bulk tokenize pipeline processes the entire dataset and encrypts the desired columns using the provided Data
Encryption Key (DEK).

```shell script
tokenize_pipeline --project="${PROJECT_ID}" \
--region="${REGION_ID}" \
--runner="DataflowRunner" \
--tempLocation="gs://${TEMP_GCS_BUCKET}/bqtemp" \
--serviceAccount=${DLP_RUNNER_SERVICE_ACCOUNT_EMAIL} \
--workerMachineType="n1-standard-1" \
--schema="$(


Encryption Mode
Parameter
Parameter information




Fixed encryption
--tinkEncryptionKeySetJson
The wrapped encryption key details

--mainKmsKeyUri
The KMS key used to wrap the tinkEncryption key.
In following format:
gcp-kms://projects/${PROJECT_ID}/locations/${REGION_ID}/keyRings/${KMS_KEYRING_ID}/cryptoKeys/${KMS_KEY_ID}

--tokenizeColumns
One or more logical column names in JSONPath format


DLP De-identify
--dlpEncryptConfigJson
DlpEncryptConfig
JSON to provide PrimitiveTransformation
for each tokenizedColumn.


Use the sample configuration JSON for reference, and the transformation reference to understand each of the transformations.



The pipeline supports following destinations for storing the tokenized output.

| Destination | Description | Pipeline parameter |
| --- | --- | --- |
| File in Cloud Storage | Stores as an AVRO file | `--outputDirectory=gs://:.` |
You can use one or both of them simultaneously.

The pipeline executes asynchronously on Dataflow. You can check the progress by following the JobLink printed in the following format:
```
INFO: JobLink: https://console.cloud.google.com/dataflow/jobs/?project=
```
Additional Pipeline options for Bulk Tokenize pipeline (`EncryptionPipelineOptions`):

| Encryption Key parameters | Description |
|----------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| `dlpEncryptConfigJson` | Provide a valid JSON object as per DlpEncryptCongig Proto defining the DLP configuration |
| `tinkEncryptionKeySetJson` | When using TINK based encryption, provide the wrapped KeySet generated by `tinkey` |
| `keyMaterial` | The encryption key to use for custom encryption module as String |
| `keyMaterialType` | One of `TINK_GCP_KEYSET_JSON`, `RAW_BASE64_KEY`, `RAW_UTF8_KEY`, `GCP_KMS_WRAPPED_KEY`. Default: `TINK_GCP_KEYSET_JSON` |
| `mainKmsKeyUri` | The Google Cloud KMS key to use for decrypting the TINK Key or GCP_KMS_WRAPPED_KEY |
| `valueTokenizerFactoryFullClassName` | The value tokenization class to use for non-DLP based tokenization. Default: `com.google.cloud.solutions.autotokenize.encryptors.DaeadEncryptingValueTokenizer$DaeadEncryptingValueTokenizerFactory` |
**NOTE:** Provide only one of dlpEncryptConfigJson, tinkEncryptionKeySetJson or keyMaterial

The tokenize pipeline's DAG will look like following:
![Encrypting Pipeline DAG](encryption_pipeline_dag.png)

### Verify encrypted result

Load the bulk tokenize pipeline's output file(s) into BigQuery to verify that all the columns specified using `tokenizeColumns` flag have been encrypted.

1. Create a BigQuery dataset for tokenized data
Replace with a region of your choice. Ensure that BigQuery dataset region/multi-region is in the same region as the Cloud Storage bucket. You can read about [considerations for batch loading data](https://cloud.google.com/bigquery/docs/batch-loading-data) for more information.

```shell script
bq --location= \
--project_id="${PROJECT_ID}" \
mk --dataset tokenized_data
```

1. Load tokenized data to a BigQuery table.
```shell script
bq load \
--source_format=AVRO \
--project_id="${PROJECT_ID}" \
"tokenized_data.TokenizedUserdata" \
"gs://${TEMP_GCS_BUCKET}/encrypted/*"
```
1. Check some records to confirm that `email` and `cc` fields have been encrypted.
```shell script
bq query \
--project_id="${PROJECT_ID}" \
"SELECT first_name, encrypted_email, encrypted_cc FROM tokenized_data.TokenizedUserdata LIMIT 10"
```

## Cleaning up

To avoid incurring charges to your Google Cloud account for the resources used in this tutorial, you can delete the project:

1. In the Cloud Console, go to the [**Manage resources** page](https://console.cloud.google.com/iam-admin/projects).
1. In the project list, select the project that you want to delete and then click **Delete** ![delete](bin_icon.png).
1. In the dialog, type the project ID and then click **Shut down** to delete the project.

## What's next

* Learn more about [Cloud DLP](https://cloud.google.com/dlp)
* Learn more about [Cloud KMS](https://cloud.google.com/kms)
* Learn about [Inspecting storage and databases for sensitive data](https://cloud.google.com/dlp/docs/inspecting-storage)
* Handling [De-identification and re-identification of PII in large-scale datasets using DLP](https://cloud.google.com/solutions/de-identification-re-identification-pii-using-cloud-dlp)

## Disclaimer
**License**: Apache 2.0

This is not an official Google product.