Friday, August 14, 2020

Launch & Persist JSON Documents In The Cloud In 10 Minutes Or Less With Autonomous JSON Database

 This morning at the Developer Live event, our Executive Vice President Juan Loaiza announced the launch of Autonomous JSON Database (AJD). This is an exciting announcement because it represents the very first time we are offering a dedicated solution for JSON document collection persistence in the Oracle Cloud. It symbolizes our commitment to offering developers solutions that fit their needs and solves the problems they face when developing microservices and applications instead of trying to convince them to use a tool that might not be the best fit for them. Of course, like all good solutions, it is scalable and adaptable which means that it can transform into a full-blown Autonomous Transaction Processing database if your needs dictate that in the future. But of course, if you know me then you have likely come to this blog post to learn how to quickly get started using AJD, so let's take a look at how to use it. I promise that you'll be able to get an instance launched and data persisted quickly, so let's not waste any more time and get into the good stuff!

In this post, I'll show you how to create a brand new instance, connect up to it, create a collection and insert, query, and remove data from that collection. We'll do all of these things in less than 10 minutes and all without leaving your browser. 

To get started, log in to your Oracle Cloud console. From any page in the console, click on the 'Cloud Shell' icon in the header of the page to launch a new Cloud Shell.

Your Cloud Shell instance will take a minute or two to launch the first time, but once it launches it will remain open and ready as you navigate around the console. While you're waiting for the Cloud Shell to launch, collect the compartment OCID that you'd like to work with. That's all that we will need before we start creating our instance and performing persistence operations in it.

Once you have that OCID, start by running the following CLI command in Cloud Shell to launch the instance.

One of the (many) awesome things about Cloud Shell is that it comes with the OCI CLI pre-installed and configured so you can immediately start working with it in your tenancy without having to spend time on installation and configuration!

Create the instance, substituting your compartment OCID and a strong password of your choosing. We'll capture the JSON response so that we can grab information from it later on.

$ export ATP=$(oci db autonomous-database create \
    --db-workload AJD \
    --compartment-id [YOUR COMPARTMENT OCID] \
    --admin-password Str0ngPassword_ \
    --cpu-core-count 1 \
    --data-storage-size-in-tbs 1 \
    --db-name testdb \
    --display-name testdb \
    --wait-for-state AVAILABLE)
view rawajd-create.sh hosted with ❤ by GitHub

Note the new value (AJD) for launching an Autonomous JSON Database in the command above.

It will take a few minutes for your AJD instance to be created and provisioned. Next, we'll grab the newly created instance's OCID and store it in a variable.

$ export ATP_ID=$(echo $ATP | jq '.data.id' --raw-output)
view rawset-atp-id.sh hosted with ❤ by GitHub

We'll need to download our wallet to make a connection, so let's do that. Enter your own path and password instead of using mine!

$ oci db autonomous-database generate-wallet \
    --autonomous-database-id $ATP_ID \
    --file /home/todd_sharp/wallet.zip \
    --password Str0ngPassword_
view rawgenerate-wallet.sh hosted with ❤ by GitHub

We're already ready to connect to our instance. You might be wondering how we're going to do that from Cloud Shell? Well, last week we quietly rolled out a new tool added in the form of SQLcl. We'll use that tool to connect and do some basic CRUD operations. Let's get to it.

Launch SQLcl:

$ sql /nolog
view rawlaunch-sqlcl.sh hosted with ❤ by GitHub

Point the tool at our wallet.

SQL> set cloudconfig /home/todd_sharp/wallet.zip
view rawcloud-config.sh hosted with ❤ by GitHub

Connect with the admin user (enter the password when prompted).

SQL> connect admin@testdb_high
view rawconnect.sh hosted with ❤ by GitHub

Note: The next steps are purely optional. If you're just testing things out, feel free to skip them. But if you're planning on working a bit further and connecting up to an application, use these steps to create a schema.

Optionally, create a user and connect with that user. Enter your own username and password for this user.

SQL> CREATE USER sodauser IDENTIFIED BY "Str0ngPassword_";
SQL> GRANT CONNECT, RESOURCE TO sodauser;
SQL> GRANT UNLIMITED TABLESPACE TO sodauser;
SQL> connect sodauser@testdb_high
Password? (**********?) ***************
view rawcreate-schema.sh hosted with ❤ by GitHub

Checkpoint

In just a few minutes we've created and connected to our AJD instance and we're ready to create a collection and perform some CRUD operations. You may be a bit confused right now - wondering how we are going to persist JSON documents in an Autonomous instance. The answer, of course, is SODA. 

There are a number of SODA implementations which means you can work with it natively from your new and existing microservices:

There's one other option for SODA, and that's the SQLcl console that we're already connected to our instance with. Let's work through a full CRUD example very quickly below. You can always refer to the SQLcl documentation for SODA later if you get stuck.

JSON CRUD In 5 Minutes

To create a collection:

SQL> soda create testcollection
Successfully created collection: testcollection

List all collections:

SQL> soda list
List of collections:
testcollection
view rawlist-collections.sh hosted with ❤ by GitHub

Insert a few docs into a collection:

SQL> soda insert testcollection {"name": "todd", "is_cool": true, "age": 43}
Json String inserted successfully.
SQL> soda insert testcollection {"name": "dominic", "is_cool": true, "age": 13}
Json String inserted successfully.
SQL> commit;
Commit complete.

Get all documents from a collection:

SQL> soda get testcollection -all
        KEY                                             Created On
        3FC9160AE62F415996725B199A357FBA                2020-08-04T18:22:36.777638000Z
        470FDF6EF0ED4CF585C63F2CA9B92CA0                2020-08-04T18:25:59.789898000Z
 2 rows selected. 
view rawcollection-get.sh hosted with ❤ by GitHub

Get a document by key:

SQL> soda get testcollection -k 3FC9160AE62F415996725B199A357FBA
Key:             3FC9160AE62F415996725B199A357FBA
Content:         {"name":"todd","is_cool":true,"age":43}
-----------------------------------------
 1 row selected. 
view rawget-by-id.sh hosted with ❤ by GitHub

Search with query by example (QBE):

SQL> soda get testcollection -f {"age":{"$lt":40}}
Key:             470FDF6EF0ED4CF585C63F2CA9B92CA0
Content:         {"name":"dominic","is_cool":true,"age":13}
-----------------------------------------
 1 row selected. 
view rawqbe.sh hosted with ❤ by GitHub

Count docs in collection:

SQL> soda count testcollection
 2 rows selected. 
view rawcount.sh hosted with ❤ by GitHub

Count docs with QBE:

SQL> soda count testcollection {"age":{"$lt":40}}
 1 row selected.
view rawcount-qbe.sh hosted with ❤ by GitHub

Replace (update) an existing doc:

SQL> soda replace testcollection 3FC9160AE62F415996725B199A357FBA {"name": "todd", "is_cool": false, "age": 43}
3FC9160AE62F415996725B199A357FBA
Json String replaced successfully.
SQL> commit;
Commit complete.
view rawupdate.sh hosted with ❤ by GitHub

Confirm the replacement:

SQL> soda get testcollection -k 3FC9160AE62F415996725B199A357FBA
Key:             3FC9160AE62F415996725B199A357FBA
Content:         {"name":"todd","is_cool":false,"age":43}
-----------------------------------------
 1 row selected. 
view rawget.sh hosted with ❤ by GitHub

Remove (delete) a document:

SQL> soda remove testcollection -k 3FC9160AE62F415996725B199A357FBA
Successfully removed 1 record.
view rawremove.sh hosted with ❤ by GitHub

View the table behind the JSON collection:

SQL> describe testcollection;
            Name       Null?             Type 
________________ ___________ ________________ 
ID               NOT NULL    VARCHAR2(255)    
CREATED_ON       NOT NULL    TIMESTAMP(6)     
LAST_MODIFIED    NOT NULL    TIMESTAMP(6)     
VERSION          NOT NULL    VARCHAR2(255)    
JSON_DOCUMENT                BLOB  
view rawdescribe.sh hosted with ❤ by GitHub

Use “vanilla” SQL to query the data (treating the JSON fields as columns):

SQL> select 
  2  tc.id, tc.created_on, tc.last_modified,
  3  tc.json_document.name, tc.json_document.age
  4  from testcollection tc;
                                 ID                         CREATED_ON                      LAST_MODIFIED       NAME    AGE 
___________________________________ __________________________________ __________________________________ __________ ______ 
DBBA0A6CF5AB4093A0819A53CDE3CD99    05-AUG-20 11.15.50.638289000 PM    05-AUG-20 11.15.50.638289000 PM    todd       43     
B464467C808C4CFBA6C0EBD76CF5A73E    05-AUG-20 11.15.57.787655000 PM    05-AUG-20 11.15.57.787655000 PM    dominic    13     
view rawselect.sh hosted with ❤ by GitHub

JSON fields can also be used in your WHERE clause:

SQL> select
2 tc.id, tc.created_on, tc.last_modified,
3 tc.json_document.name, tc.json_document.age
4 from testcollection tc
5 where tc.json_document.name = 'todd';
ID CREATED_ON LAST_MODIFIED NAME AGE
___________________________________ __________________________________ __________________________________ _______ ______
DBBA0A6CF5AB4093A0819A53CDE3CD99 05-AUG-20 11.15.50.638289000 PM 05-AUG-20 11.15.50.638289000 PM todd 43
view rawselect2.sh hosted with ❤ by GitHub

If you want to view the JSON document as a string, use json_serialize():

SQL> select 
  2  json_serialize(tc.json_document) as json
  3  from testcollection tc
  4  where tc.json_document.name = 'todd';
                                      JSON 
__________________________________________ 
{"name":"todd","is_cool":true,"age":43}    
view rawjson_serialize.sh hosted with ❤ by GitHub

That's all it takes to create a JSON collection and insert, update and delete JSON documents to and from that collection. You're now ready to integrate your AJD instance into your new and existing microservices for full JSON document collection persistence in the Oracle Cloud.

Bonus: Create & Test a Node.JS App in Cloud Shell!

Since we're here, we might as well test out one of the SODA client libraries, so let's create a basic Node.JS application in Cloud Shell to work with our collection. Node and NPM are already installed, so we can create a directory and an application straight away at this point. But before we do that, let's do a tiny bit of admin work. We'll need our wallet unzipped and we have to set an environment variable to the location where we unzipped it. We'll also need to update the sqlnet.ora file to point at our wallet directory. We can accomplish all of this like so (again, update the path to the proper path for your Cloud Shell home directory):

$ unzip wallet.zip -d /home/todd_sharp/wallet
$ export TNS_ADMIN=/home/todd_sharp/wallet
$ sed -i 's/?\/network\/admin/\/home\/todd_sharp\/wallet/g' /home/todd_sharp/wallet/sqlnet.ora
view rawunzip-wallet.sh hosted with ❤ by GitHub

Create a directory for the project and switch to it:

$ mkdir node-soda && cd node-soda
view rawmkdir.sh hosted with ❤ by GitHub

Create a new project and install oracledb:

$ npm init && npm install && npm i oracledb
This utility will walk you through creating a package.json file.
# install output removed for brevity
> oracledb@5.0.0 install /home/todd_sharp/node-soda/node_modules/oracledb
> node package/install.js
# install output removed for brevity
added 1 package and audited 1 package in 0.809s
found 0 vulnerabilities
view rawinstall-oracledb.sh hosted with ❤ by GitHub

Install the instantclient into your project directory and set the LD_LIBRARY_PATH:

mkdir -p /opt/oracle
wget https://download.oracle.com/otn_software/linux/instantclient/19800/instantclient-basiclite-linux.x64-19.8.0.0.0dbru.zip
unzip instantclient-basiclite-linux.x64-19.8.0.0.0dbru.zip
rm instantclient-basiclite-linux.x64-19.8.0.0.0dbru.zip
export LD_LIBRARY_PATH=`pwd`/instantclient_19_8:$LD_LIBRARY_PATH

Set some env vars:

$ export DB_USER=sodauser
$ export DB_PASSWORD=Str0ngPassword_
$ export CONNECT_STRING=dbdemo_low
view rawset-vars.sh hosted with ❤ by GitHub
In the project root, create and edit your index.js file (or whatever you chose for your entry point):

const oracledb = require('oracledb');
oracledb.outFormat = oracledb.OBJECT;
oracledb.fetchAsString = [oracledb.CLOB];
oracledb.autoCommit = true;
(async () => {
        console.log(‘Create pool...');
        await oracledb.createPool({
            user: process.env.DB_USER,
            password: process.env.DB_PASSWORD,
            connectString: process.env.CONNECT_STRING,
        });
        console.log('Get connection...');
        const connection = await oracledb.getConnection();
        console.log('Get SODA DB...');
        const soda = connection.getSodaDatabase();
 
console.log('Get collection...');
        const collection = await soda.createCollection('testcollection');
        console.log('Insert doc...');
        const entry = await collection.insertOneAndGet({name: 'ava', is_cool: true, age: 14});
        console.log({id: entry.key, created_on: entry.createdOn});
        console.log('Get new doc...');
        const doc = await collection.find().key(entry.key).getOne();
        console.log(doc.getContent());
        console.log('Close connection...');
        connection.close();
        console.log('Close pool...');
        try {
            await oracledb.getPool().close(10);
            console.log('Pool closed');
        } catch(err) {
            console.error(err);
        }
})();
view rawindex.js hosted with ❤ by GitHub

Save, close, and run it with:

$ node index.js
view rawrun.sh hosted with ❤ by GitHub

Should produce output similar to this:

Creating pool...
Get connection...
Get SODA DB...
Get collection...
Insert doc...
{ id: 'F25C4D41FB514F0ABF62DB76C0AAA99D',
created_on: '2020-08-06T02:23:43.944593Z' }
Get new doc...
{ name: 'ava', is_cool: true, age: 14 }
Close connection...
Close pool...
Pool closed
view rawrun-output.sh hosted with ❤ by GitHub

We can log back in with SQLcl and take a look at our collection to see what happened:

SQL> soda get testcollection -all
        KEY                                             Created On
        03414EC292344FD8BFED22DCE165801E                2020-08-06T02:20:31.716147000Z
        80A0568D85FD4F8EBF954CE5E33D9150                2020-08-06T02:21:02.524842000Z
        F25C4D41FB514F0ABF62DB76C0AAA99D                2020-08-06T02:23:43.944593000Z
 3 rows selected.
view rawget.sh hosted with ❤ by GitHub

We can see that we now have 3 records and the newest record's ID matches the ID that we inserted with Node.JS. Let's get the newest one by ID and confirm that the content matches the content we persisted with Node.JS:

​​​​​​​SQL> soda get testcollection -k F25C4D41FB514F0ABF62DB76C0AAA99D
Key:             F25C4D41FB514F0ABF62DB76C0AAA99D
Content:         {"name":"ava","is_cool":true,"age":14}
-----------------------------------------
 1 row selected.
view rawget-by-id.sh hosted with ❤ by GitHub

In just 5 more minutes, we created a Node.JS application to persist JSON documents in our AJD instance and confirmed those operations with SQLcl. 

Summary & Cleanup

If you'd like to destroy the test instance of Autonomous JSON Database that we created above, run the following:

$ oci db autonomous-database delete --autonomous-database-id $ATP_ID
view rawdestroy.sh hosted with ❤ by GitHub

In this post, we learned about Autonomous JSON Database (AJD), created an AJD instance, and performed CRUD operations using SQLcl and a basic Node.JS application from within Cloud Shell. To learn more about AJD, please refer to the documentation:

Read more about AJD here:


No comments:

Must Watch YouTube Videos for Databricks Platform Administrators

  While written word is clearly the medium of choice for this platform, sometimes a picture or a video can be worth 1,000 words. Below are  ...