Functions for sending shell commands to Auto Scaling Groups or single instances.
The ASG version works like this:
- Ask which instances belong to an ASG.
- Randomly pick one Instance ID.
- Send the supplied command(s) to that instance.
Tip! You can use the php /path/artisan schedule:run
command for autoscaled Laravel projects. The scheduler will then just run on whichever instance gets to run the command, without having to select/dedicate a server for it.
These commands will run as root so prefixing the command with sudo -Hu apache
(or similar) will probably make sense in some deployments.
You should have the SSM agent installed on any instance you're sending commands to. Please refer to AWS documentation on SSM for instructions.
The functions use the Linux-specific AWS-RunShellScript
command document, so these will not work with Windows as-is. But it's easy to just change the document in the code.
When a Lambda function is run every minute it will accumulate 44640 requests for a 31-day month. Since the minimum memory of 128 MB is plenty for this and the average duration seems to be 500 ms, we can calculate 44640 requests * (128 / 1024 GB) * 0,5 seconds = 2790 GB-seconds.
Running Lambda at these volumes is free when your account is still under the free tier limit of 1 million requests and 400,000 GB-seconds. If we ignore the free tier, these would cost $0.0000002 per request and $0.00000104 per 500 ms. So a total of $0.009486 per month.
See https://aws.amazon.com/lambda/pricing/ for details.
For target-auto-scaling-group.js:
{"autoScalingGroup": "some-asg", "commands": [ "uptime" ]}
For target-instance.js:
{"instanceId": "i-11111111111111111", "commands": [ "uptime" ]}
These steps will assume you have not set up anything yet and are using AWS Management Console. This process consists of setting up the Lamdba function, an IAM role and a CloudWatch Events rule.
- Go to Lambda > Create a Lambda function
- In the Select blueprint step, pick Blank Function.
- In the Configure triggers step, click on the dashed box and pick CloudWatch Events.
- For Rule, select Create a new rule.
- Fill in a Rule name, eg.
my-app-cron
. - Fill in a Schedule expression, eg.
rate(1 minute)
. - Click Enable trigger (or don't if you want to test thoroughly).
- In the Configure function step...
- Fill in a Name, eg.
commander
. - For Runtime, pick Node.js 6.10 or above.
- Provide the function code however you wish. Editing code inline and copy-pasting the content of either
target-auto-scaling-group.js
ortarget-instance.js
is a quick way. - Keep Handler as the default
index.handler
. - For Role, pick Create a custom role. A new tab opens for IAM.
- Fill in a Role Name, eg.
commander-role
. - Click on View Policy Document, then Edit.
- Copy-paste the content of
iam-role.json
. This will allow two additional actions for these functions. - After you click Allow, the new tab will close and you'll get back to defining the Lambda function.
- Fill in a Role Name, eg.
- Under Advanced settings set Timeout to 10 seconds. Node.js functions can sometimes go past the default 3.
- Fill in a Name, eg.
- After creating the function you should define its parameters at CloudWatch Events, then enabled it.
- At the function's Triggers tab, click on the CloudWatch Event name to jump to its configuration.
- At the top right, click Actions and Edit.
- In the Targets section, click Add target.
- The Lambda function should be preselected, so pick the function name for Function.
- Expand Configure input.
- Pick Constant (JSON text).
- Copy-paste suitable parameters from the Parameters section above. Edit according to your needs.
- Continue to the next step by clicking Configure details.
- State should already be checked, so click on Update rule to get things going.
Licensed under the Apache 2.0 License.