Node.js read and write from Azure Table Storage

โœ๏ธ

Tutorial teaching you how to connect your Node.js app to Azure Table Storage to read and write data.

2 Oct, 2020 ยท 9 min read

We have been exploring node.js in combination with Azure, and today we will connect to a database!

Not any database, but we will query the Azure Table Storage to read and write data.

We will also finalize our short-url node tool, and it will look like this:

Node.js Azure Table Storage link shortener

Click any of these links if you are looking for the other articles in the Node.js Azure series.

Creating an Azure Table Storage

Azure Table Storage is a NoSQL database where we can store large amounts of data.

In our case, we are storing URLs, which will get a random unique ID to retrieve them.

To create a new Azure Table Storage, we must go to "Storage accounts" and click the Add button.

Azure add new Storage Account

You will have to fill out the following details on the next screen.

  • Resource group: Choose the one we created for the App Service
  • Storage account name: Your unique name for this storage account
  • Location: Same as our App Service
  • Then, we can click Review + create

Storage account settings

Once that is done, we can open our resource and click the tables button.

Azure Storage account ~ table

We will go to another view where we can add a new Table.

Note a Storage account can have multiple tables or other storage.

Azure add new Table

We can then go ahead and open the Storage Explorer to see our table. (This is still a Preview mode)

Azure Table Storage preview

Linking the Azure Table Storage and App Service

If you are testing locally, you can add the following two lines to your routes.js.

process.env.AZURE_STORAGE_ACCOUNT = '{storage_account}';
process.env.AZURE_STORAGE_CONNECTION_STRING = '{connection}';

You can replace these values with the actual values from the Storage Account:

Storage account keys

Once you are ready to publish it to Azure, you can remove the two keys above and visit the App Service in Azure.

Go to Configuration and add these two values as Application Settings.

Azure application setting

Connecting to Azure Table Storage in Node.js

Once we set up our table in Azure, we can modify our Node.js app to connect to this database.

You can find my starter project here on GitHub.

Let's first NPM install the dependencies we need:

npm i -s azure-storage
npm i -s shortid

That will install the azure-storage package and the shortid package.

We can then open up our routes.js file and add these packages.

const azure = require('azure-storage');
const shortid = require('shortid');
const table = 'links';
const entGen = azure.TableUtilities.entityGenerator;

Writing data to Azure Table Storage

Our first objective is to start writing data to our Table Storage.

Let's define our POST route.

The route is called generate and accepts a POST with a JSON object that looks as such:

{
  "url": "https://daily-dev-tips.com"
}
router.route('/generate').post(function (req, res) {
  const { url } = req.body;
  let code = shortid.generate();
  generateCodeUntilSuccess(code, url).then((c) => {
    res.status(200).send('https://dailydevtips.azurewebsites.net/' + c);
  });
});

Once the response body comes in, we generate a unique short-id and call the function generateCodeUntilSuccess. Once that returns something, we send the browser the new short URL!

Lets see that generateCodeUntilSuccess function

async function generateCodeUntilSuccess(code, url) {
  return await addLink(code, url)
    .then((c) => {
      return c;
    })
    .catch((e) => {
      generateCodeUntilSuccess(shortid.generate(), url);
    });
}

Here we use an async...await method since we need to ensure the generated code is unique. If that fails, we let the function call itself.

This means all the magic happens in the addLink function above.

The addLink function accepts a code and a URL. It will first connect to the azure Table Storage and query if this code is already used.

If that is the case, we will reject this call.

If the code is not used before we can go ahead and insert this into our table.

To insert into the table storage, we always need to pass the partitionKey and the rowKey. These are our unique identifiers.

Be aware: It is not an auto-increment field, and we must provide the actual unique values.

Once we insert our row, we resolve the code to show back to the user.

function addLink(code, url) {
  return new Promise((resolve, reject) => {
    try {
      const tableService = azure.createTableService();
      const query = new azure.TableQuery().top(1).where('RowKey eq ?', code);
      tableService.queryEntities(
        table,
        query,
        null,
        function (error, result, response) {
          if (!error) {
            const link = {
              PartitionKey: entGen.String('link_' + code),
              RowKey: entGen.String(code),
              Url: entGen.String(url),
            };
            tableService.insertEntity(
              table,
              link,
              function (error, result, response) {
                if (!error) {
                  resolve(code);
                }
                reject(error);
              }
            );
          }
        }
      );
    } catch (e) {
      reject(e);
    }
  });
}

If we run this in Postman, we should see a return like this.

Postman Result

Read data from Azure Table Storage

After we learned how to write data to the Azure Storage, we now look at how to read data the data. We want to visit the URL we just created and get redirected to the final URL we provided as input.

Let's start by defining the route for our unique code.

router.route('/:uniqueId').get(function (req, res) {
  const uniqueId = req.params.uniqueId;
  getRecord(uniqueId)
    .then((url) => {
      res.redirect(301, url);
    })
    .catch((err) => {
      res.status(400).send('Error: Code not found');
    });
});

We create a "wildcard" route and retrieve our unique code from the URL.

We then call the getRecord function to read from the database. And on success of the query, we redirect the user to the returned URL.

So, what does this getRecord function do?

It's a copy of the above check function but built to read and return the actual URL if the query finds a record.

function getRecord(uniqueId) {
  return new Promise(function (resolve, reject) {
    try {
      const tableService = azure.createTableService();
      const query = new azure.TableQuery()
        .top(1)
        .where('RowKey eq ?', uniqueId);
      tableService.queryEntities(
        table,
        query,
        null,
        function (error, result, response) {
          if (!error) {
            if (result.entries[0] !== undefined) {
              resolve(result.entries[0].Url._);
            } else {
              reject('code not found');
            }
          } else {
            reject(error);
          }
        }
      );
    } catch (e) {
      reject(e);
    }
  });
}

If we visit our unique URL, we get redirected to the defined link at which we want to end up.

You can find the complete code for this project on GitHub.

Thank you for reading, and let's connect!

Thank you for reading my blog. Feel free to subscribe to my email newsletter and connect on Facebook or Twitter

Spread the knowledge with fellow developers on Twitter
Tweet this tip
Powered by Webmentions - Learn more

Read next ๐Ÿ“–

GitHub automated deployments to Azure

26 Sep, 2020 ยท 3 min read

GitHub automated deployments to Azure

Deploying a Node app to Azure

25 Sep, 2020 ยท 3 min read

Deploying a Node app to Azure