56👍
Heroku pushes the use of environment variables for settings and secret keys:
The traditional approach for handling such config vars is to put them under source – in a properties file of some sort. This is an error-prone process, and is especially complicated for open source apps which often have to maintain separate (and private) branches with app-specific configurations.
A better solution is to use environment variables, and keep the keys out of the code. On a traditional host or working locally you can set environment vars in your bashrc. On Heroku, you use config vars.
With Foreman and .env
files Heroku provide an enviable toolchain to export, import and synchronise environment variables.
Personally, I believe it’s wrong to save secret keys alongside code. It’s fundamentally inconsistent with source control, because the keys are for services extrinsic to the the code. The one boon would be that a developer can clone HEAD and run the application without any setup. However, suppose a developer checks out a historic revision of the code. Their copy will include last year’s database password, so the application will fail against today’s database.
With the Heroku method above, a developer can checkout last year’s app, configure it with today’s keys, and run it successfully against today’s database.
102👍
You’re exactly right to want to encrypt your sensitive settings file while still maintaining the file in version control. As you mention, the best solution would be one in which Git will transparently encrypt certain sensitive files when you push them so that locally (i.e. on any machine which has your certificate) you can use the settings file, but Git or Dropbox or whoever is storing your files under VC does not have the ability to read the information in plaintext.
Tutorial on Transparent Encryption/Decryption during Push/Pull
This gist https://gist.github.com/873637 shows a tutorial on how to use the Git’s smudge/clean filter driver with openssl to transparently encrypt pushed files. You just need to do some initial setup.
Summary of How it Works
You’ll basically be creating a .gitencrypt
folder containing 3 bash scripts,
clean_filter_openssl
smudge_filter_openssl
diff_filter_openssl
which are used by Git for decryption, encryption, and supporting Git diff. A master passphrase and salt (fixed!) is defined inside these scripts and you MUST ensure that .gitencrypt is never actually pushed.
Example clean_filter_openssl
script:
#!/bin/bash
SALT_FIXED=<your-salt> # 24 or less hex characters
PASS_FIXED=<your-passphrase>
openssl enc -base64 -aes-256-ecb -S $SALT_FIXED -k $PASS_FIXED
Similar for smudge_filter_open_ssl
and diff_filter_oepnssl
. See Gist.
Your repo with sensitive information should have a .gitattribute file (unencrypted and included in repo) which references the .gitencrypt directory (which contains everything Git needs to encrypt/decrypt the project transparently) and which is present on your local machine.
.gitattribute
contents:
* filter=openssl diff=openssl
[merge]
renormalize = true
Finally, you will also need to add the following content to your .git/config
file
[filter "openssl"]
smudge = ~/.gitencrypt/smudge_filter_openssl
clean = ~/.gitencrypt/clean_filter_openssl
[diff "openssl"]
textconv = ~/.gitencrypt/diff_filter_openssl
Now, when you push the repository containing your sensitive information to a remote repository, the files will be transparently encrypted. When you pull from a local machine which has the .gitencrypt directory (containing your passphrase), the files will be transparently decrypted.
Notes
I should note that this tutorial does not describe a way to only encrypt your sensitive settings file. This will transparently encrypt the entire repository that is pushed to the remote VC host and decrypt the entire repository so it is entirely decrypted locally. To achieve the behavior you want, you could place sensitive files for one or many projects in one sensitive_settings_repo. You could investigate how this transparent encryption technique works with Git submodules http://git-scm.com/book/en/Git-Tools-Submodules if you really need the sensitive files to be in the same repository.
The use of a fixed passphrase could theoretically lead to brute-force vulnerabilities if attackers had access to many encrypted repos/files. IMO, the probability of this is very low. As a note at the bottom of this tutorial mentions, not using a fixed passphrase will result in local versions of a repo on different machines always showing that changes have occurred with ‘git status’.
- [Django]-Missing Table When Running Django Unittest with Sqlite3
- [Django]-Switching to PostgreSQL fails loading datadump
- [Django]-How do I filter query objects by date range in Django?
17👍
The cleanest way in my opinion is to use environment variables. You won’t have to deal with .dist files for example, and the project state on the production environment would be the same as your local machine’s.
I recommend reading The Twelve-Factor App‘s config chapter, the others too if you’re interested.
- [Django]-Django models: mutual references between two classes and impossibility to use forward declaration in python
- [Django]-Sending HTML email in django
- [Django]-Is there a way to filter a queryset in the django admin?
10👍
An option would be to put project-bound credentials into an encrypted container (TrueCrypt or Keepass) and push it.
Update as answer from my comment below:
Interesting question btw. I just found this: github.com/shadowhand/git-encrypt which looks very promising for automatic encryption
- [Django]-How to set environment variables in PyCharm?
- [Django]-Data Mining in a Django/Postgres application
- [Django]-What are the differences between django-tastypie and djangorestframework?
10👍
I suggest using configuration files for that and to not version them.
You can however version examples of the files.
I don’t see any problem of sharing development settings. By definition it should contain no valuable data.
- [Django]-Django Rest Framework – Could not resolve URL for hyperlinked relationship using view name "user-detail"
- [Django]-What is a django.utils.functional.__proxy__ object and what it helps with?
- [Django]-Django datetime issues (default=datetime.now())
9👍
Since asking this question I have settled on a solution, which I use when developing small application with a small team of people.
git-crypt uses GPG to transparently encrypt files when their names match certain patterns. For intance, if you add to your .gitattributes
file…
*.secret.* filter=git-crypt diff=git-crypt
…then a file like config.secret.json
will always be pushed to remote repos with encryption, but remain unencrypted on your local file system.
If I want to add a new GPG key (a person) to your repo which can decrypt the protected files then run git-crypt add-gpg-user <gpg_user_key>
. This creates a new commit. The new user will be able to decrypt subsequent commits.
- [Django]-Access web server on VirtualBox/Vagrant machine from host browser?
- [Django]-Django: How to manage development and production settings?
- [Django]-Where should signal handlers live in a django project?
7👍
BlackBox was recently released by StackExchange and while I have yet to use it, it seems to exactly address the problems and support the features requested in this question.
From the description on https://github.com/StackExchange/blackbox:
Safely store secrets in a VCS repo (i.e. Git or Mercurial). These
commands make it easy for you to GPG encrypt specific files in a repo
so they are “encrypted at rest” in your repository. However, the
scripts make it easy to decrypt them when you need to view or edit
them, and decrypt them for for use in production.
- [Django]-How do I use an UpdateView to update a Django Model?
- [Django]-.filter() vs .get() for single object? (Django)
- [Django]-MySQL "incorrect string value" error when save unicode string in Django
6👍
I ask the question generally, but in my specific instance I would like
to store secret keys and passwords for a Django/Python site using git
and github.
No, just don’t, even if it’s your private repo and you never intend to share it, don’t.
You should create a local_settings.py put it on VCS ignore and in your settings.py do something like
from local_settings import DATABASES, SECRET_KEY
DATABASES = DATABASES
SECRET_KEY = SECRET_KEY
If your secrets settings are that versatile, I am eager to say you’re doing something wrong
- [Django]-Multiple ModelAdmins/views for same model in Django admin
- [Django]-Iterate over model instance field names and values in template
- [Django]-Django – "Incorrect type. Expected pk value, received str" error
4👍
EDIT: I assume you want to keep track of your previous passwords versions – say, for a script that would prevent password reusing etc.
I think GnuPG is the best way to go – it’s already used in one git-related project (git-annex) to encrypt repository contents stored on cloud services. GnuPG (gnu pgp) provides a very strong key-based encryption.
- You keep a key on your local machine.
- You add ‘mypassword’ to ignored files.
- On pre-commit hook you encrypt the mypassword file into the mypassword.gpg file tracked by git and add it to the commit.
- On post-merge hook you just decrypt mypassword.gpg into mypassword.
Now if your ‘mypassword’ file did not change then encrypting it will result with same ciphertext and it won’t be added to the index (no redundancy). Slightest modification of mypassword results in radically different ciphertext and mypassword.gpg in staging area differs a lot from the one in repository, thus will be added to the commit. Even if the attacker gets a hold of your gpg key he still needs to bruteforce the password. If the attacker gets an access to remote repository with ciphertext he can compare a bunch of ciphertexts, but their number won’t be sufficient to give him any non-negligible advantage.
Later on you can use .gitattributes to provide an on-the-fly decryption for quit git diff of your password.
Also you can have separate keys for different types of passwords etc.
- [Django]-Why doesn't django's model.save() call full_clean()?
- [Django]-Django get the static files URL in view
- [Django]-Django count RawQuerySet
3👍
Usually, i seperate password as a config file. and make them dist.
/yourapp
main.py
default.cfg.dist
And when i run main.py
, put the real password in default.cfg
that copied.
ps. when you work with git or hg. you can ignore *.cfg
files to make .gitignore
or .hgignore
- [Django]-Django – Clean permission table
- [Django]-Django-way for building a "News Feed" / "Status update" / "Activity Stream"
- [Django]-Django stops working with RuntimeError: populate() isn't reentrant
3👍
Provide a way to override the config
This is the best way to manage a set of sane defaults for the config you checkin without requiring the config be complete, or contain things like hostnames and credentials. There are a few ways to override default configs.
Environment variables (as others have already mentioned) are one way of doing it.
The best way is to look for an external config file that overrides the default config values. This allows you to manage the external configs via a configuration management system like Chef, Puppet or Cfengine. Configuration management is the standard answer for the management of configs separate from the codebase so you don’t have to do a release to update the config on a single host or a group of hosts.
FYI: Encrypting creds is not always a best practice, especially in a place with limited resources. It may be the case that encrypting creds will gain you no additional risk mitigation and simply add an unnecessary layer of complexity. Make sure you do the proper analysis before making a decision.
- [Django]-Why is __init__ module in django project loaded twice
- [Django]-Missing Table When Running Django Unittest with Sqlite3
- [Django]-How to pass information using an HTTP redirect (in Django)
2👍
Encrypt the passwords file, using for example GPG. Add the keys on your local machine and on your server. Decrypt the file and put it outside your repo folders.
I use a passwords.conf, located in my homefolder. On every deploy this file gets updated.
- [Django]-Disable a method in a ViewSet, django-rest-framework
- [Django]-Find object in list that has attribute equal to some value (that meets any condition)
- [Django]-Django – "Incorrect type. Expected pk value, received str" error
2👍
No, private keys and passwords do not fall under revision control. There is no reason to burden everyone with read access to your repository with knowing sensitive service credentials used in production, when most likely not all of them should have access to those services.
Starting with Django 1.4, your Django projects now ship with a project.wsgi
module that defines the application
object and it’s a perfect place to start enforcing the use of a project.local
settings module that contains site-specific configurations.
This settings module is ignored from revision control, but it’s presence is required when running your project instance as a WSGI application, typical for production environments. This is how it should look like:
import os
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "project.local")
# This application object is used by the development server
# as well as any WSGI server configured to use this file.
from django.core.wsgi import get_wsgi_application
application = get_wsgi_application()
Now you can have a local.py
module who’s owner and group can be configured so that only authorized personnel and the Django processes can read the file’s contents.
- [Django]-MySQL "incorrect string value" error when save unicode string in Django
- [Django]-FileUploadParser doesn't get the file name
- [Django]-Redirect to named url pattern directly from urls.py in django?
2👍
If you need VCS for your secrets you should at least keep them in a second repository seperated from you actual code. So you can give your team members access to the source code repository and they won’t see your credentials. Furthermore host this repository somewhere else (eg. on your own server with an encrypted filesystem, not on github) and for checking it out to the production system you could use something like git-submodule.
- [Django]-Django queries – id vs pk
- [Django]-Django Rest Framework pagination extremely slow count
- [Django]-Error when using django.template
2👍
This is what I do:
- Keep all secrets as env vars in $HOME/.secrets (go-r perms) that $HOME/.bashrc sources (this way if you open .bashrc in front of someone, they won’t see the secrets)
- Configuration files are stored in VCS as templates, such as config.properties stored as config.properties.tmpl
-
The template files contain a placeholder for the secret, such as:
my.password=##MY_PASSWORD##
-
On application deployment, script is ran that transforms the template file into the target file, replacing placeholders with values of environment variables, such as changing ##MY_PASSWORD## to the value of $MY_PASSWORD.
- [Django]-Django: Open uploaded file while still in memory; In the Form Clean method?
- [Django]-CommandError: You must set settings.ALLOWED_HOSTS if DEBUG is False
- [Django]-You are trying to add a non-nullable field 'new_field' to userprofile without a default
1👍
Another approach could be to completely avoid saving secrets in version control systems and instead use a tool like vault from hashicorp, a secret storage with key rolling and auditing, with an API and embedded encryption.
- [Django]-How to submit form without refreshing page using Django, Ajax, jQuery?
- [Django]-Why does DEBUG=False setting make my django Static Files Access fail?
- [Django]-Bypass confirmation prompt for pip uninstall
0👍
You could use EncFS if your system provides that. Thus you could keep your encrypted data as a subfolder of your repository, while providing your application a decrypted view to the data mounted aside. As the encryption is transparent, no special operations are needed on pull or push.
It would however need to mount the EncFS folders, which could be done by your application based on an password stored elsewhere outside the versioned folders (eg. environment variables).
- [Django]-TransactionManagementError "You can't execute queries until the end of the 'atomic' block" while using signals, but only during Unit Testing
- [Django]-Django.db.migrations.exceptions.InconsistentMigrationHistory
- [Django]-How to completely uninstall a Django app?