1. Introduction
In this codelab you will learn how to use Spanner Data Boost to query Spanner data from BigQuery using zero-ETL federated queries and without impacting the Spanner database.
Spanner Data Boost is a fully managed, serverless service that provides independent compute resources for supported Spanner workloads. Data Boost lets you execute analytics queries and data exports with near-zero impact to existing workloads on the provisioned Spanner instance using a serverless on-demand usage model.
When paired with BigQuery external connections, Data Boost allows you to easily query data from Spanner into your data analytics platform without complex ETL data movement.
Prerequisites
- A basic understanding of the Google Cloud, Console
- Basic skills in command line interface and Google shell
What you'll learn
- How to deploy a Spanner instance
- How to load data to create a Spanner database
- How to access Spanner data from BigQuery without Data Boost
- How to access Spanner data from BigQuery with Data Boost
What you'll need
- A Google Cloud Account and Google Cloud Project
- A web browser such as Chrome
2. Setup and Requirements
Self-paced environment setup
- Sign-in to the Google Cloud Console and create a new project or reuse an existing one. If you don't already have a Gmail or Google Workspace account, you must create one.
- The Project name is the display name for this project's participants. It is a character string not used by Google APIs. You can always update it.
- The Project ID is unique across all Google Cloud projects and is immutable (cannot be changed after it has been set). The Cloud Console auto-generates a unique string; usually you don't care what it is. In most codelabs, you'll need to reference your Project ID (typically identified as
PROJECT_ID
). If you don't like the generated ID, you might generate another random one. Alternatively, you can try your own, and see if it's available. It can't be changed after this step and remains for the duration of the project. - For your information, there is a third value, a Project Number, which some APIs use. Learn more about all three of these values in the documentation.
- Next, you'll need to enable billing in the Cloud Console to use Cloud resources/APIs. Running through this codelab won't cost much, if anything at all. To shut down resources to avoid incurring billing beyond this tutorial, you can delete the resources you created or delete the project. New Google Cloud users are eligible for the $300 USD Free Trial program.
Start Cloud Shell
While Google Cloud can be operated remotely from your laptop, in this codelab you will be using Google Cloud Shell, a command line environment running in the Cloud.
From the Google Cloud Console, click the Cloud Shell icon on the top right toolbar:
It should only take a few moments to provision and connect to the environment. When it is finished, you should see something like this:
This virtual machine is loaded with all the development tools you'll need. It offers a persistent 5GB home directory, and runs on Google Cloud, greatly enhancing network performance and authentication. All of your work in this codelab can be done within a browser. You do not need to install anything.
3. Create a Spanner instance and database
Enable the Spanner API
Inside Cloud Shell, make sure that your project ID is setup:
gcloud config set project [YOUR-PROJECT-ID]
PROJECT_ID=$(gcloud config get-value project)
Configure your default region to us-central1
. Feel free to change this to a different region supported by Spanner regional configurations.
gcloud config set compute/region us-central1
Enable the Spanner API:
gcloud services enable spanner.googleapis.com
Create the Spanner instance
In this step we set up our Spanner instance for the codelab. To do this, open up Cloud Shell and run this command:
export SPANNER_INSTANCE_ID=codelab-demo
export SPANNER_REGION=regional-us-central1
gcloud spanner instances create $SPANNER_INSTANCE_ID \
--config=$SPANNER_REGION \
--description="Spanner Codelab instance" \
--nodes=1
Command output:
$ gcloud spanner instances create $SPANNER_INSTANCE_ID \
--config=$SPANNER_REGION \
--description="Spanner Codelab instance" \
--nodes=1
Creating instance...done.
Create the database
Once your instance is running, you can create the database. Spanner allows for multiple databases on a single instance.
The database is where you define your schema. You can also control who has access to the database, set up custom encryption, configure the optimizer, and set the retention period.
To create the database, again use the gcloud command line tool:
export SPANNER_DATABASE=codelab-db
gcloud spanner databases create $SPANNER_DATABASE \
--instance=$SPANNER_INSTANCE_ID
Command output:
$ gcloud spanner databases create $SPANNER_DATABASE \
--instance=$SPANNER_INSTANCE_ID
Creating database...done.
4. Load data
Before you can use Data Boost, you need to have some data in the database. To do this, you will create a Cloud Storage bucket, upload an avro import into the bucket, and start a Dataflow import job to load the Avro data into Spanner.
Enable APIs
To do that, open a Cloud Shell prompt if the previous one was closed.
Make sure to enable the Compute, Cloud Storage, and Dataflow APIs.
gcloud services enable compute.googleapis.com storage.googleapis.com dataflow.googleapis.com
Expected console output:
$ gcloud services enable compute.googleapis.com storage.googleapis.com dataflow.googleapis.com
Operation "operations/acat.*snip*" finished successfully.
Stage import files on Cloud Storage
Now, create the bucket to store the avro files:
export GCS_BUCKET=spanner-codelab-import_$(date '+%Y-%m-%d_%H_%M_%S')
gcloud storage buckets create gs://$GCS_BUCKET
Expected console output:
$ gcloud storage buckets create gs://$GCS_BUCKET
Creating gs://spanner-codelab-import/...
Next, download the tar file from github and extract it.
wget https://github.com/dtest/spanner-databoost-tutorial/releases/download/v0.1/spanner-chat-db.tar.gz
tar -xzvf spanner-chat-db.tar.gz
Expected console output:
$ wget https://github.com/dtest/spanner-databoost-tutorial/releases/download/v0.1/spanner-chat-db.tar.gz
*snip*
*snip*(123 MB/s) - ‘spanner-chat-db.tar.gz' saved [46941709/46941709]
$
$ tar -xzvf spanner-chat-db.tar.gz
spanner-chat-db/
spanner-chat-db/users.avro-00000-of-00002
spanner-chat-db/user_notifications-manifest.json
spanner-chat-db/interests-manifest.json
spanner-chat-db/users-manifest.json
spanner-chat-db/users.avro-00001-of-00002
spanner-chat-db/topics-manifest.json
spanner-chat-db/topics.avro-00000-of-00002
spanner-chat-db/topics.avro-00001-of-00002
spanner-chat-db/user_interests-manifest.json
spanner-chat-db/spanner-export.json
spanner-chat-db/interests.avro-00000-of-00001
spanner-chat-db/user_notifications.avro-00000-of-00001
spanner-chat-db/user_interests.avro-00000-of-00001
And now upload the files to the bucket you created.
gcloud storage cp spanner-chat-db gs://$GCS_BUCKET --recursive
Expected console output:
$ gcloud storage cp spanner-chat-db gs://$GCS_BUCKET --recursive
Copying file://spanner-chat-db/users.avro-00000-of-00002 to gs://spanner-codelab-import/spanner-chat-db/users.avro-00000-of-00002
Copying file://spanner-chat-db/user_notifications-manifest.json to gs://spanner-codelab-import/spanner-chat-db/user_notifications-manifest.json
Copying file://spanner-chat-db/interests-manifest.json to gs://spanner-codelab-import/spanner-chat-db/interests-manifest.json
Copying file://spanner-chat-db/users-manifest.json to gs://spanner-codelab-import/spanner-chat-db/users-manifest.json
Copying file://spanner-chat-db/users.avro-00001-of-00002 to gs://spanner-codelab-import/spanner-chat-db/users.avro-00001-of-00002
Copying file://spanner-chat-db/topics-manifest.json to gs://spanner-codelab-import/spanner-chat-db/topics-manifest.json
Copying file://spanner-chat-db/topics.avro-00000-of-00002 to gs://spanner-codelab-import/spanner-chat-db/topics.avro-00000-of-00002
Copying file://spanner-chat-db/topics.avro-00001-of-00002 to gs://spanner-codelab-import/spanner-chat-db/topics.avro-00001-of-00002
Copying file://spanner-chat-db/user_interests-manifest.json to gs://spanner-codelab-import/spanner-chat-db/user_interests-manifest.json
Copying file://spanner-chat-db/spanner-export.json to gs://spanner-codelab-import/spanner-chat-db/spanner-export.json
Copying file://spanner-chat-db/interests.avro-00000-of-00001 to gs://spanner-codelab-import/spanner-chat-db/interests.avro-00000-of-00001
Copying file://spanner-chat-db/user_notifications.avro-00000-of-00001 to gs://spanner-codelab-import/spanner-chat-db/user_notifications.avro-00000-of-00001
Copying file://spanner-chat-db/user_interests.avro-00000-of-00001 to gs://spanner-codelab-import/spanner-chat-db/user_interests.avro-00000-of-00001
Completed files 13/13 | 54.6MiB/54.6MiB
Average throughput: 46.4MiB/s
Import Data
With the files in Cloud Storage, you can start a dataflow import job to load the data into Spanner.
gcloud dataflow jobs run import_chatdb \
--gcs-location gs://dataflow-templates-us-central1/latest/GCS_Avro_to_Cloud_Spanner \
--region us-central1 \
--staging-location gs://$GCS_BUCKET/tmp \
--parameters \
instanceId=$SPANNER_INSTANCE_ID,\
databaseId=$SPANNER_DATABASE,\
inputDir=gs://$GCS_BUCKET/spanner-chat-db
Expected console output:
$ gcloud dataflow jobs run import_chatdb \
> --gcs-location gs://dataflow-templates-us-central1/latest/GCS_Avro_to_Cloud_Spanner \
> --region us-central1 \
> --staging-location gs://$GCS_BUCKET/tmp \
> --parameters \
> instanceId=$SPANNER_INSTANCE_ID,\
> databaseId=$SPANNER_DATABASE,\
> inputDir=gs://$GCS_BUCKET/spanner-chat-db
createTime: '*snip*'
currentStateTime: '*snip*'
id: *snip*
location: us-central1
name: import_chatdb
projectId: *snip*
startTime: '*snip*'
type: JOB_TYPE_BATCH
You can check the status of the import job with this command.
gcloud dataflow jobs list --filter="name=import_chatdb" --region us-central1
Expected console output:
$ gcloud dataflow jobs list --filter="name=import_chatdb"
`--region` not set; getting jobs from all available regions. Some jobs may be missing in the event of an outage. https://cloud.google.com/dataflow/docs/concepts/regional-endpoints
JOB_ID NAME TYPE CREATION_TIME STATE REGION
*snip* import_chatdb Batch 2024-04-*snip* Done us-central1
Verify data in Spanner
Now, go to the Spanner Studio and ensure the data is there. First, expand the topics table to view the columns.
Now, run the following query to ensure data is available:
SELECT COUNT(*) FROM topics;
Expected output:
5. Read data from BigQuery
Now that you have data in Spanner, it's time to access it from within BigQuery. To do this, you will set up an external connection to Spanner in BigQuery.
Assuming you have the right permissions, create an external connection to Spanner with the following steps.
Click the ‘Add' button at the top of the BigQuery console, and select the ‘Connections to eternal data sources' option.
You can now run a query to read data from Spanner. Run this query in the BigQuery console, making sure to substitute the value for your ${PROJECT_ID}:
SELECT *
FROM (
SELECT * FROM EXTERNAL_QUERY("projects/${PROJECT_ID}/locations/us-central1/connections/codelab-demo-chat_no-databoost", "SELECT users.userUUID, SHA256(users.email) as hashed_email, COUNT(*) num_topics, m.last_posted from users HASH JOIN (select MAX(t.created) last_posted, t.userUUID FROM topics t GROUP BY 2) m USING (userUUID)HASH JOIN topics USING (userUUID) GROUP BY users.userUUID, users.email, m.last_posted")
)
ORDER BY num_topics DESC;
Example output:
You can see information about the job, such as how long it took to run and how much data was processed in the ‘Job Information' tab.
Next, you will add a Data Boost connection to Spanner, and compare the results.
6. Read data using Data Boost
To use Spanner Data Boost, you need to create a new external connection from BigQuery to Spanner. Click ‘Add' in the BigQuery console and select ‘Connections from external data sources
' again.
Fill in the details with the same connection URI to Spanner. Change the ‘Connection ID' and check the ‘use data boost' box.
With the Data Boost connection created, you can run the same query but with the new connection name. Again, substitute your project_id in the query.
SELECT *
FROM (
SELECT * FROM EXTERNAL_QUERY("projects/${PROJECT_ID}/locations/us-central1/connections/codelab-demo-chat_use-databoost", "SELECT users.userUUID, SHA256(users.email) as hashed_email, COUNT(*) num_topics, m.last_posted from users HASH JOIN (select MAX(t.created) last_posted, t.userUUID FROM topics t GROUP BY 2) m USING (userUUID)HASH JOIN topics USING (userUUID) GROUP BY users.userUUID, users.email, m.last_posted")
)
ORDER BY num_topics DESC;
You should get the same result set as before. Did the timing change?
7. Understanding Data Boost
Spanner Data Boost allows you to use resources unrelated to your Spanner instance resources. This primarily reduces the impact of your analytical workloads on your operational workloads.
You can see this if you run the query to not use Data Boost a few times over two or three minutes. Remember to substitute ${PROJECT_ID}
.
SELECT *
FROM (
SELECT * FROM EXTERNAL_QUERY("projects/${PROJECT_ID}/locations/us-central1/connections/codelab-demo-chat_no-databoost", "SELECT users.userUUID, SHA256(users.email) as hashed_email, COUNT(*) num_topics, m.last_posted from users HASH JOIN (select MAX(t.created) last_posted, t.userUUID FROM topics t GROUP BY 2) m USING (userUUID)HASH JOIN topics USING (userUUID) GROUP BY users.userUUID, users.email, m.last_posted")
)
ORDER BY num_topics DESC;
Then, wait a few more minutes, and run the query to use Data Boost a few more times. Remember to substitute ${PROJECT_ID}
.
SELECT *
FROM (
SELECT * FROM EXTERNAL_QUERY("projects/${PROJECT_ID}/locations/us-central1/connections/codelab-demo-chat_use-databoost", "SELECT users.userUUID, SHA256(users.email) as hashed_email, COUNT(*) num_topics, m.last_posted from users HASH JOIN (select MAX(t.created) last_posted, t.userUUID FROM topics t GROUP BY 2) m USING (userUUID)HASH JOIN topics USING (userUUID) GROUP BY users.userUUID, users.email, m.last_posted")
)
ORDER BY num_topics DESC;
Now, go back to Spanner studio in the Cloud Console, and go to System Insights
Here, you can see the CPU metrics. The queries run without Data Boost are using CPU for the ‘executesql_select_withpartitiontoken' operations. Even though the query is the same, the Data Boost execution does not show up on your instance CPU utilization.
In many cases, the performance of the analytic query will improve when using Data Boost. The data set in this tutorial is small and there are no other workloads competing for resources. Therefore, this tutorial does not expect to showcase performance improvements.
Feel free to play around with the queries and workloads and see how Data Boost works. When you're done, proceed to the next section to clean up the environment.
8. Clean up environment
If you created your project specifically for this codelab, you can simply delete the project to clean it up. If you want to keep the project, and clean up the individual components then proceed in the following steps.
Remove BigQuery connections
To remove both connections click on the three dots next to the connection name., Select ‘Delete', then follow the instructions to delete the connection.
Delete Cloud Storage bucket
gcloud storage rm --recursive gs://$GCS_BUCKET
Delete Spanner instance
To clean up, just go into the Cloud Spanner section of the Cloud Console and delete the ‘codelab-demo
' instance we created in the codelab.
9. Congratulations
Congratulations for completing the codelab.
What we've covered
- How to deploy a Spanner instance
- How to load data into Spanner using Dataflow
- How to access Spanner data from BigQuery
- How to use Spanner Data Boost to avoid impact on your Spanner instance for analytical queries from BigQuery
10. Survey
Output: