Monday, March 14, 2016

Keep Your Database Permissions Small

Update: March 15, 2016
There were several interesting bits of legitimate criticism on this post on Reddit.  The idea of the Principle of Least Privilege that this post communicates is right.  Some of the implementation details in the examples fell a little short though.  A thousand thanks to all who commented.  I learned a lot from this exchange.  The original post below........
 

A typical web application needs an account to access a database in order to do it's job.  These applications often face the outside world.  As such, direct access to your database is only a few degrees of separation from your users.  How does one secure this?  Start with minimalism.


Make A Minimal Database Account

Start by setting up a database account.  When doing this, err to too few permissions rather than too many.  It is inconvenient in the short term.  However, over the life of your project, you'll be glad you did.  Things will be more secure and easier to maintain.

An example.  There is a feature in one of my projects where the user can delete learning resources.  Here is a piece of Ansible configuration that sets up the MySQL database account.


- name: Set up the public user
mysql_user:
login_user: root
login_password: "exampleRootPassword"
name: public
password: "examplePublicPassword"
priv: "mydatabase.*: SELECT,INSERT"

Notice that the account does not offer any delete permissions at all.  So how is deletion possible?


Coding Within Constraints

The trick is this.  When the user "deletes" a learning resource, they don't directly delete anything at all.  Instead, they put in a request to delete it.  This much only requires insert privileges.  Here is an example using Python and SQLAlchemy.


conn = eng.connect()
resource_parm = bindparam("resource_id", type_=Integer)
now = datetime.now()
query = resource_deletion_table.insert()\
.values(resource_id=resource_parm, deletion_time=now)
conn.execute(query, resource_id=resource_id)

As far as the user is concerned, the resource is "deleted" right away.  This is thanks to a subquery used to help with the lookup.


conn = eng.connect()
user_id_parm = bindparam("user_id")
query = select([resource_table])\
.where(and_(resource_table.c.user_id == user_id_parm,
text("""resources.id not in
(select resource_id from resource_deletions)""")))
result = conn.execute(query, user_id=user_id)

Doing Real Deletions Safely

To be sure, that can take up a good bit of space over time.  A real deletion of data is ultimately called for.  Here's a snippet from a script that does that job.


# set up the connection to an account that has delete permissions.
root_pw = parser["myapplication"]["root_password"]
ct_string = "mysql+pymysql://root:{}@localhost/mydatabase".format(root_pw)
eng = create_engine(ct_string)
conn = eng.connect()
# clear the learning resources
query_string = """delete from resources where id in (
select resource_id from resource_deletions)"""
conn.execute(query_string)
# clear the deletion queue for resources
query_string = "delete from resource_deletions"
conn.execute(query_string)

Not that this script does use a database account with delete permissions.  It is also a task run through a cron scheduler on an Ubuntu Linux server.  This is important because it is not invoked by the main web application at all.  The system is more secure because actual deletion is treated as an internal detail that exists outside of public access.  Below is one way to set up a cron scheduled job to run every other hour.


- name: Set up the cron job
cron:
name: Purge Deleted Files
state: present
job: /var/app/myapp/purge_deletions.py
user: www-data
hour: "*/2"

The takeaway is this.  A good security mindset can lead to solutions that provide features to user while also keeping data safer.  So the next time there comes a situation where you think you need more permissions, stop and reconsider.  You might be able to get by with less than you think.

No comments:

Post a Comment