Concurrency Labs - AWS Price Calculator tool
This repository uses the AWS Price List API to implement price calculation utilities.
- Dynamo DB
Visit the following URLs for more details:
https://www.concurrencylabs.com/blog/aws-pricing-lambda-realtime-calculation-function/ https://www.concurrencylabs.com/blog/aws-lambda-cost-optimization-tools/ https://www.concurrencylabs.com/blog/calculate-near-realtime-pricing-serverless-applications/
The code is structured in the following way:
awspricecalculator. The modules in this package search data within the AWS Price List API index files. They take price dimension parameters as inputs and return results in JSON format. This package is called by Lambda functions or other Python scripts.
functions. This is where our Lambda functions live. Functions are packaged using the Serverless framework.
scripts. Here are some Python scripts to help with management and price optimizations. See README.md in the scripts folder for more details.
Available Lambda functions:
This function is called by a schedule configured using CloudWatch Events. The function receives a JSON object configured in the schedule. The JSON object supports the following format:
The function finds resources with the corresponding tag, gets current usage using CloudWatch metrics,
projects usage into a longer time period (a month), calls pricecalculator to calculate price
and puts results in CloudWatch metrics under the namespace
Supported services are EC2, EBS, ELB, RDS and Lambda. Not all price dimensions are supported for all services, though.
You can configure as many CloudWatch Events as you want, each one with a different tag.
- The function only considers for price calculation those resources that are tagged. For example, if there is an untagged ELB with tagged EC2 instances, the function will only consider the EC2 instances for the calculation. If there is a tagged ELB with untagged EC2 instances, the function will only calculate price for the ELB.
- The behavior described above is intended for simplicity, otherwise the function would have to cover a number of combinations that might or might not be suitable to all users of the function.
- To keep it simple, if you want a resource to be included in the calculation, then tag it. Otherwise leave it untagged.
Limitations The function doesn't support cost estimations for the following:
- EC2 data transfer for instances not registered to an ELB
- EC2 Reserved Instances
- EBS Snapshots
- RDS data trasfer
- Lambda data transfer
- Kinesis PUT Payload Units are partially calculated based on CloudWatch metrics (there's no 100% accuracy for this price dimension)
- Dynamo DB storage
- Dynamo DB data transfer
Install - using CloudFormation (recommended)
I created a CloudFormation template that deploys the Lambda function, as well as the CloudWatch Events schedule. All you have to do is specify the tag key and value you want to calculate pricing for. For example: TagKey:stack, TagValue:mywebapp
Click here to get started:
The function publishes a metric named
EstimatedCharges to CloudWatch, under namespace
ConcurrencyLabs/Pricing/NearRealTimeForecast and it uses
the following dimensions:
- Currency: USD
- ForecastPeriod: monthly
- ServiceName: ec2, rds, lambda, kinesis, dynamodb
- Tag: mykey=myvalue
Updating to the latest version using CloudFormation
This function will be updated regularly in order to fix bugs, update AWS Price List data and also to add more functionality. This means you will likely have to update the function at some point. I recommend installing the function using the CloudFormation template, since it will simplify the update process.
To update the function, just go to the CloudFormation console, select the stack you've created and click on Actions -> Update Stack:
Then select "Specify an Amazon S3 template URL" and enter the following value:
And that's it. CloudFormation will update the function with the latest code.
Install Locally (if you want to modify it) - Manual steps
If you only want to install the Lambda function, you don't need to follow the steps below, just follow the instructions in the "Install - Using CloudFormation" section above.
If you want to setup a dev environment, run a local copy, make some modifications and then install in your AWS account, then keep reading...
Clone the repo
git clone https://github.com/concurrencylabs/aws-pricing-tools aws-pricing-tools
(Optional, but recommended) Create an isolated Python environment using virtualenv
It's always a good practice to create an isolated environment so we have greater control over the dependencies in our project, including the Python runtime.
If you don't have virtualenv installed, run:
pip install virtualenv
For more details on virtualenv, click here.
Now, create a Python 2.7 virtual environment in the location where you cloned the repo into. If you want to name your project aws-pricing-tools, then run (one level up from the dir, use the same local name you used when you cloned the repo):
virtualenv aws-pricing-tools -p python2.7
After your environment is created, it's time to activate it. Go to the recently created folder of your project (i.e. aws-pricing-tools) and from there run:
From your project root folder, run:
This will install the following dependencies to the
- tinydb - The code in this repo queries the Price List API csv records using the tinydb library.
- numpy - Used for statistics in the lambda optimization script
... and the following dependencies in your default site-packages location:
- python-local-lambda - lets me test my Lambda functions locally using test events in my workstation.
- boto3 - AWS Python SDK to call AWS APIs.
Install the Serverless Framework
Since the pricing tool runs on AWS Lambda, I decided to use the Serverless Framework. This framework enormously simplifies the development, configuration and deployment of Function as a Service (a.k.a. FaaS, or "serverless") code into AWS Lambda.
You should follow the instructions described here, which can be summarized in the following steps:
- Make sure you have Node.js installed in your workstation.
- Install the Serverless Framework
npm install -g serverless
- Confirm Serverless has been installed
The steps in this post were tested using version
Serverless needs access to your AWS account, so it can create and update AWS Lambda functions, among other operations. Therefore, you have to make sure Serverless can access a set of IAM User credentials. Follow these instructions. In the long term, you should make sure these credentials are limited to only the API operations Serverless requires - avoid Administrator access, which is a bad security and operational practice.
Checkout the code from this repo into your virtualenv folder.
Set environment variables
export AWS_DEFAULT_PROFILE=<your-aws-cli-profile> export AWS_DEFAULT_REGION=<us-east-1|us-west-2|etc.>
How to test the function locally
Download the latest AWS Price List API Index file
The code needs a local copy of the the AWS Price List API index file. The GitHub repo doesn't come with the index file, therefore you have to download it the first time you run your code and every time AWS publishes a new Price List API index.
Also, this index file is constantly updated by AWS. I recommend subscribing to the AWS Price List API change notifications.
In order to download the latest index file, go to the ```scripts```` folder and run:
python get-latest-index.py --service=all
The script takes a few seconds to execute since some index files are a little heavy (like the EC2 one).
Run a test
Once you have the virtualenv activated, all dependencies installed, environment variables set and the latest AWS Price List index file, it's time to run a test.
test/events/constant-tag.json with a tag key/value pair that exists in your AWS account.
Then run, from the root location in the local repo and replace and with actual values:
python-lambda-local functions/calculate-near-realtime.py test/events/constant-tag.json -l lib/ -l . -f handler -t 30 -a arn:aws:lambda:<your-region>:<your-aws-account-id>:function:calculate-near-realtime
Deploy the Serverless Project
From your project root folder, run: