ZL
About Articles Contact
Published on Nov 13, 2019
Filed under:
#node

Protecting and syncing secret keys

You should never expose API keys or secrets. If you expose them, you might get into trouble. Once, I almost had to pay an excessive amount because my friend leaked my Amazon API key by accident.

What’s the amount? I can’t remember, but I think somewhere between $20,000 to $60,000. Thankfully, Amazon waived the charges.

It’s big trouble if you expose your API keys. So don’t expose them.

The best way to protect your API keys is to use environment variables.

Setting environment variables

An environment variable is a variable that’s set outside of the code you’re running. Environment variables can be set on a Machine level.

If you use Bash, you can set environment variables in .bash_profile. If you use ZSH, you can set environment variables in .zshrc.

Terminal window
# Exports a variable called helloworld
export helloworld="Hello world!"

After setting the environment variable, update your shell. You can do this by:

  1. Running source on the file you changed. (like source ~/.zshrc)
  2. Restarting the terminal

Either way works.

After you sourced the file (or restarted the terminal), type echo $helloworld in your Terminal. You should see this:

Terminal window
echo $helloworld
Echoing the environment variable.

Using environment variables

In Node, you can use the environment variable by writing process.env.VARIABLE_NAME.

// This is located in a Node file called server.js
const variable = process.env.helloworld
console.log(variable)
Terminal shows Hello world! when we run the server.js file.

A better way to use environment variables

It can be a hassle to setup environment variables on your computer, on the server, and on computers for all your team members.

The easy way to sync environment variables across all computers is to use a dotenv.

Setting up dotenv

First, you’ll need to install dotenv from npm:

Terminal window
npm install dotenv --save

dotenv lets you save environment variables into a .env file. I like to put this .env file in the secrets folder. This can go along with all my secret files (like gcreds.json).

Placed the variables.env file in the secrets folder.

Here’s the syntax to create an environment variable in a .env file. (Note: DO NOT write the export keyword!).

Terminal window
# Creates a environment variable called "variable".
variable="value"

Example:

Terminal window
helloworld="Hello world!"

Using variables from dotenv

First, you need to load the .env file. If you placed .env in ./secrets, you can load the .env file this way:

const dotenv = require('dotenv').config({
path: './secrets/.env'
})

Then, you use the environment variable like before:

const variable = process.env.helloworld
console.log(variable)
Logged helloworld environment variable in the console.

Syncing the .env file into the server

You can use rsync to sync the .env file into your server. (I use Digital Ocean if you’re curious. Use this link to get $50 credit).

To use rsync, you can run a command like this:

Terminal window
rsync -avzr ./secrets/ user@host:/path-to-destination

This command syncs everything in the ./secrets folder into your destination folder. In the example above, I had a greds.json file. This gcreds.json file gets synced as well.

Unfortunately, you need to enter user and host into the rsync command. This means the user and hostname of your server gets exposed (if you synced the command).

A better way is to run rsync in Node.

Node Rsync

First, you need to install rsync from npm:

Terminal window
npm install rsync --save-dev

Then, you need to require rsync.

const Rsync = require('rsync')

Then, you create an rsync object with the options you want to include. Here’s what I use:

const rsync = new Rsync()
.shell('ssh') // Tells rsync to use SSH
.set('stats') // Tells rysnc to display stats from the
.flags('avz') // Tells rsync to use `a`, `v`, and `z` options. (Archive, Verbose, and Compress).
.flags('n') // This is for dryrun. Test before syncing! :)
.source('./secrets') // The folder you want to sync
.destination(
`${process.env.SSH_USER}@${process.env.SSH_HOST}:/path-to-destination`
) // The destination

Notice I used SSH_USER and SSH_HOST environment variables in the rsyrc object? This allows me to access the server on any computer via SSH. (Provided the computer has a valid SSH private key).

This also means I need to include dotenv before rsync.

const dotenv = require('dotenv').config({
path: './secrets/.env'
})
const rsync = new Rsync()
// ...

After setting up the rsync object, you can pipe the outputs from rsync into the terminal. You can do it with this command.

Note: You only do this if you want to see the results from rsync in your terminal.

rsync.output(
function (data) {
// do things like parse progress
const string = Buffer.from(data).toString()
console.log(string)
},
function (data) {
// do things like parse error output
console.log(data)
}
)

Finally, you execute rsync with this:

// Execute the command
rsync.execute(function (error, code, cmd) {
if (error) console.error(error)
console.log(cmd)
})

I put all the code above into a file called sync.js. When I want to sync my secrets, I run this sync.js file.

Terminal window
node sync.js

To make things easier for me, I put this command as a script in my package.json file.

"scripts": {
"sync": "node sync.js"
}

Updating environment variables

dotenv does not overwrite environment variables that are already set. If you need to overwrite environment variables, you can run this code:

const Rsync = require('rsync')
const fs = require('fs')
const dotenv = require('dotenv')
const updateEnv = pathToConfig => {
const envConfig = dotenv.parse(fs.readFileSync(pathToConfig))
for (const k in envConfig) {
process.env[k] = envConfig[k]
}
}
updateEnv('./secrets/.env')

That’s it!

Previous Designing Learn JavaScript's course portal (Part 2) Next Overcoming my panic towards accessibility

Join My Newsletter

I share what I’m learning on this newsletter: code, building businesses, and living well.

Sometimes I write about technical deep-dives, product updates, musings on how to live, and sometimes my struggles and how I’m breaking through.

Regardless of the type of content, I do my best to send you an update every week.

If you’re into making things and growing as a person, you’ll probably feel at home here.

“

Zell’s writing is very accessible to newcomers because he shares his learning experience. He covers current topics and helps all readers level up their web development skills. Must subscribe.

Chen Hui Jing
Chen Hui Jing — Web Developer
The Footer

General

Home About Contact Testimonials Tools I Use

Projects

Magical Dev School Splendid Labz

Socials

Youtube Instagram Tiktok Github Bluesky X

Follow Along

Email RSS
© 2013 - 2025 Zell Liew / All rights reserved / Terms