Ever pondered how you can keep your secrets out of sight. Me too. I am not promising this is 100% but it can be a good start for you.

This is a quick tutorial on storing your secret keys using a .env file and the dotenv package instead of storing it in your .bash_profile or .zprofile. The reason I am against storing your credentials in those files is that they make it hard to work on multiple projects connecting to different databases/ cloud accounts.

I am going to make it short and sweet. I have taken a leaf out of this blog

TLDR - skip to this section

Folder Structure

  • Create a new project folder and change directory into it.
mkdir project_name
cd project_name
  • Let’s create our .env file If you have a git repo remember to GITIGNORE this file !!!!!
touch .env

This is a sort of hidden file, you probably won’t see it in your folder structure. If in doubt you can always run cat .env to view the file contents or ls -a ./project_name to view the project_name folder contents.

Let’s populate our .env file. This can get a bit interesting. I am assuming you’re a bit comfortable with the command line. So just follow me

vim .env

Don’t panic. It’s just vim.

  • Press i to get into INSERT mode.
  • Copy and paste the below.
DATABASE_URL=postgres://username:password@localhost:5432/dbname
AWS_ACCESS_KEY=myaccesskey
AWS_SECRET_ACCESS_KEY=mysecretkey
OTHER_VARIABLE=zanawazhere
  • Press escape
  • Enter :wq to SAVE and QUIT

Wasn’t too hard was it.

  • Let’s create 2 folders such that our directory looks like.
project_name 
     └── parent_folder
            └── child_folder
  • Install dotenv with pip install dotenv
  • Navigate to child_folder cd parent_folder/child_folder

  • add your config.py file in child_folder. Copy the contents below,
# config.py.py
import os
from dotenv import load_dotenv, find_dotenv

# find .env automagically by walking up directories until it's found
dotenv_path = find_dotenv()

# load up the entries as environment variables
load_dotenv(dotenv_path)

DATABASE_URL = os.environ.get("DATABASE_URL")
OTHER_VARIABLE = os.environ.get("OTHER_VARIABLE")

if __name__ == "__main__":
    print(DATABASE_URL)
    print(OTHER_VARIABLE)

If you run config.py you should get, the following output.

postgres://username:password@localhost:5432/dbname
zanawazhere

The point of me having the child and parent folder was to show that the dotenv package will crawl the directory to extract the secrets in .env.

Take it up a level

Now that we got the basics out of the way.

  • Create a utils folder in the project_name folder. mkdir utils
  • Move your config file to be inside the utils directory
  • You can delete the parent_folder and child_folder

Let’s edit the config.py file to look like,

# config.py
import os
from dotenv import load_dotenv, find_dotenv

# find .env automagically by walking up directories until it's found
dotenv_path = find_dotenv()

# load up the entries as environment variables
load_dotenv(dotenv_path)


class Config:
    APP_NAME = "your_app_name"

class DevelopmentConfig(Config):
    DEBUG = True

    DATABASE_URL = os.environ.get("DATABASE_URL")
    OTHER_VARIABLE = os.environ.get("OTHER_VARIABLE")

    AWS_ACCESS_KEY = os.environ.get("AWS_ACCESS_KEY")
    AWS_SECRET_ACCESS_KEY = os.environ.get("AWS_SECRET_ACCESS_KEY")
    AWS_S3_BUCKET_NAME = 'aws-s3-bucket-name-for-dev'


class TestConfig(Config):
    DEBUG = True
    TESTING = True

    DATABASE_URL = os.environ.get("DATABASE_URL")
    OTHER_VARIABLE = os.environ.get("OTHER_VARIABLE")

    AWS_ACCESS_KEY = os.environ.get("AWS_ACCESS_KEY")
    AWS_SECRET_ACCESS_KEY = os.environ.get("AWS_SECRET_ACCESS_KEY")


class ProductionConfig(Config):
    DEBUG = False

    DATABASE_URL = os.environ.get("DATABASE_URL")
    OTHER_VARIABLE = os.environ.get("OTHER_VARIABLE")

    AWS_ACCESS_KEY = os.environ.get("AWS_ACCESS_KEY")
    AWS_SECRET_ACCESS_KEY = os.environ.get("AWS_SECRET_ACCESS_KEY")


class CIConfig:
    SERVICE = 'travis-ci'
    HOOK_URL = 'web-hooking-url-from-ci-service'

All we have done is create a config class which is inherited by the environment classes. The best thing is that we are not hard coding out environment keys in our code.

  • Create a main.py in the project_name folder. Copy and paste the below,
# main.py
import sys
import utils.config as config

if __name__ == '__main__':
    env = 'dev'

    if env == 'dev':
        app = config.DevelopmentConfig
    elif env == 'test':
        app = config.TestConfig
    elif env == 'prod':
        app = config.ProductionConfig
    else:
        raise ValueError('Invalid environment name')

    app.ci = config.CIConfig

    app = config.DevelopmentConfig()
    print(app.DATABASE_URL)
    print(app.OTHER_VARIABLE)
    print(app.AWS_SECRET_ACCESS_KEY)
    print(app.AWS_ACCESS_KEY)

Run your main.py, you should see the output:

postgres://username:password@localhost:5432/dbname
zanawazhere
mysecretkey
myaccesskey

Take a deep breath. That was longer than I thought. I think you should be able to take these concepts and use them in your personal projects.

I hope if you’re a beginner you got a good idea of what I was trying to show. Any feedback is appreciated.