Skip to content

Commit

Permalink
Merge pull request #251 from hackoregon/ssm-upload-script
Browse files Browse the repository at this point in the history
Script to enable bulk import of AWS SSM parameters
  • Loading branch information
MikeTheCanuck authored Jul 20, 2019
2 parents defc73d + 631e1f9 commit d0fbe36
Show file tree
Hide file tree
Showing 3 changed files with 107 additions and 0 deletions.
27 changes: 27 additions & 0 deletions bin/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
* **create-db.sh**: after installing PostgreSQL database service, configured and running, run this script from an SSH shell on the VM to create each new database instance (note: if your database instance requires PostGIS, run `build-and-install-PostGIS.sh` before restoring the database or generating the database schema)
* **create-ec2-machine-database.sh**: run this script from a local \*NIX (Mac, Linux or Ubuntu on Win10) shell, where `awscli` is installed and AWS credentials are available, to create a new EC2 VM from the input `profile.sh` configuration
* **ec2-profile-database-development.sh**: use this script as the `profile.sh` input configuration for the `create-ec2-machine-database.sh` script, to create the central PostgreSQL database host for Hack Oregon's 2018 project season
* **ssm-parameters-upload.py**: run this script to upload a bulk array of new SSM Parameter Store parameters
* **upload-and-ssh-into-ec2.sh**: run this script to `scp` all files in the current directory and `ssh` into the designated SSH-enabled host

Order of operations, from scratch:
Expand All @@ -14,3 +15,29 @@ Order of operations, from scratch:
4. **build-and-install-PostGIS.sh** - adds PostGIS functionality to PostgreSQL
5. **create-db.sh** - creates individual database instances on the PostgreSQL/PostGIS service
6. **grant-readonly-access-for-PostgreSQL-user.sh** - takes an additional read-only psql user and grants it only read-only access to the specified database

## ssm-parameters-upload.py

Imports an arbitrary set of SSM parameters into AWS Parameter Store, both String and SecureString parameters. Does not overwrite existing SSM parameters.

Script expects comma delimiter by default, but can be trivially changed by modifying the `delimiter_character` variable.

### Prerequisites

* the `awscli` must be installed locally on the host
* user must have AWS credentials sufficient to write to the AWS SSM parameter store

### Required Inputs

* `--file` = path and name of CSV file
* `--keyid` = the key-id for encrypting SecureString SSM parameters
* `--region` = the AWS region in which the parameters will be stored

### CSV file requirements

File must contain a header row with four columns (in any order):

* NamespacePrefix (e.g. `/staging/2019-sandbox`)
* Parameter (e.g. `POSTGRES_HOST`)
* Type (either `String` or `SecureString`)
* Value (e.g. `rdshostname1.randomsubdomain.us-west-2.rds.amazonaws.com`)
31 changes: 31 additions & 0 deletions bin/ssm-parameters-upload.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
## Assumes an input CSV with four columns headed with a header row whose values are:
## 1. NamespacePrefix
## 2. Parameter
## 3. Type (SecureString or String)
## 4. Value
##
## Where the SSM parameter being created is the concatenation of NamespacePrefix + '/' + Parameter

from csv import DictReader
from operator import itemgetter
import subprocess
import argparse

delimiter_character = ','

parser = argparse.ArgumentParser()
parser.add_argument("--file", type=argparse.FileType('r', encoding='UTF-8'), help="name of CSV file with SSM parameters", required=True, dest="csv_file", action="store")
parser.add_argument("--key", help="keyid for encrypting SecureString SSM parameters", required=True, dest="key_id", action="store")
parser.add_argument("--region", help="AWS region in which to store SSM parameters", required=True, dest="AWSRegion", action="store")
args = parser.parse_args()

with args.csv_file as params:
params_reader = DictReader(params, delimiter = delimiter_character)
for row in params_reader:
if row['Type'] == "SecureString":
subprocess.run(['aws', 'ssm', 'put-parameter', '--name', row['NamespacePrefix'] + "/" + row['Parameter'], '--type', row['Type'], '--value', row['Value'], '--region', args.AWSRegion, '--key-id', args.key_id])
# subprocess.run(['aws', 'ssm', 'put-parameter', '--name', row['NamespacePrefix'] + "/" + row['Parameter'], '--type', row['Type'], '--value', row['Value'], '--region', AWSRegion, '--key-id', key_id])
elif row['Type'] == "String":
subprocess.run(['aws', 'ssm', 'put-parameter', '--name', row['NamespacePrefix'] + "/" + row['Parameter'], '--type', row['Type'], '--value', row['Value'], '--region', args.AWSRegion])
else:
print('\n' + "Input record has invalid Type field - param = " + row['Parameter'])
49 changes: 49 additions & 0 deletions docs/database/2019/HOWTO: Build RDS instances.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
# HOWTO Build RDS instances (for 2019 project season)

Summary: these are the manual RDS-instantiation instrux that will form the basis for CloudFormation automation of each 2019 project's databases.

## Background

In the 2019 Hack Oregon project season, we will be using per-project RDS instances for hosting both the staging and production databases.

Per-project - rather than a centralized database instance shared by all databases - because at the outset of the 2019 project season, there was more uncertainty than usual about the size or compute-resource requirements for each 2019 project.

Staging vs. Production - rather than a single layer for full-season usage - because we assume that we'll get around to the laborious and expensive task of standing up a production-ready, well-protected API-to-DB layer - a less-free-for-all infrastructure for Django-to-DB compute than we've had in the past (despite all good intentions).

## Procedure

1. Create an RDS instance with the following shared characteristics (suitable for staging aka "development" work):
* engine = PostgreSQL
* Use Case = dev/test
* DB engine version = PostgreSQL 11.2-R1
* DB instance class = db.t2.small
* Multi-AZ deployment = No
* Storage type = General purpose (SSD)
* Allocated storage = 100 GB
* Virtual Private Cloud = Default VPC
* Public accessibility = yes (note: will be "no" for production DBs, production databases can only be accessed via the related Django container in ECS production deploy)
* Availability zone = no preference
* VPC security groups = hacko-public-database (note: will be more restrictive for production DBs)
* IAM DB authentication = Disable
* Encryption = Enable
* Master key = (default) aws/rds
* Backup retention period = 7 days
* Backup window = no preference
* Enhanced monitoring = Enable enhanced monitoring
* Performance insights = Enable performance insights, retention period = 7 days, Master key = (default) aws/rds
* Log exports = Postgresql log
* Auto minor version upgrade = Enable auto minor version upgrade
* Maintenance window = No preference
* Deletion protection = Enable delete protection
2. Create a login role according to the naming conventions e.g. `transportation2019` - which has following privileges: Can login, Create databases, Inherit rights from the parent roles
3. Create the database according to the https://docs.google.com/spreadsheets/d/147thL899Bf8IL3ma1S9XBrNXL2xYsIRM5mE-3fceIcQ/ naming scheme
4. Assign the created login role as the Owner of the DB
* note: in AWS RDS, the role creating the database must have the role which will be owner of the database being created: https://stackoverflow.com/a/34898033
5. Execute the following command in the new DB using the Query Tool or other SQL automation:
`CREATE EXTENSION postgis;`
* Even better: add the `postgis` extension to the `template1` database, so that all new databases will automatically get it: _(TBD)_
6. If you need add another role with read-only privileges to the DB, use the script in *HOWTO-enable-readonly-database-user* or similar.

## Other troubleshooting notes

* Check which extensions are enabled for each database - connect to the database and run `SELECT * FROM pg_extension;`

0 comments on commit d0fbe36

Please sign in to comment.