‘Smart contracts’ is a misnomer. Despite its name, smart contracts on Ethereum are not self-executing digital agreements. Smart contract code only run when triggered by an external account. In other words, you need an external process to trigger the smart contract.
In this article, we’ll build a solution to this problem. You’ll learn:
Finally, we’ll go through serverless-ethers, a fully-functional smart contract automation service that you can run and deploy out-of-the box. You can use this project as a base for building custom smart contract automation that fit your needs.
Thesample application is open source and available on Github. Just clone and hit deploy!serverless-ethers
Read on to learn why we need automation and how it works.
Imagine that we want to implement a smart contract with a function that should be automatically executed every 1 hour. ⏳⚙️
How can you accomplish this?
You can’t. This is not possible with plain Solidity smart contracts. Despite its name, ‘smart’ contracts in Ethereum are not self-executing. You need an external source (either human or machine) to call the smart contract and execute its code.
The most a contract can do is enforce a 1-hour interval between executions, for example:
function runMe() public {
require(block.timestamp >= lastTriggeredAt + 1 hour);
...
}
The above require() statement ensures that there is at least an hour in between executions. Otherwise, the transaction reverts.
However, somebody still needs to call the smart contract for the code to run in the first place.
Technically, it is possible to use function modifiers to automatically execute certain operations. One example of this is Compound Governance’s COMP distribution. Once an address has earned 0.001 COMP, any Compound transaction (e.g. supplying an asset, or transferring a cToken) will automatically transfer COMP to their wallet.
You can implement the above logic in a function modifier (a decorator), wrap the modifier around a function, and have the logic automatically executed whenever the function is called. The caller will pay the gas required for the additional logic.
However, not all smart contract systems follow this approach. One reason is that it can lead to unpredictable gas usage, since these modifiers may only run under certain conditions. It also forces additional gas fees onto a random subset of users, who just happened to be the unlucky few selected to ‘rebalance’ the contract.
Finally, somebody still needs to call the smart contract for the code to run.
DeFi protocols already rely on some kind of off-chain smart contract automation. MakerDAO relies on third party Keepers to monitor the collateralization ratios of debt positions and liquidate any undercollateralized position. Other DeFi protocols have similar needs.
There are two, often overlapping use cases around off-chain smart contract automation:
You often want to execute a contract periodically or under certain conditions. For example:
You want to know if certain conditions are met. For example:
The above use cases sounds like a good fit for a serverless function. By going serverless, we can deploy code without provisioning anything beforehand, or managing anything afterward. It’s easier than ever to make your idea live.
The Serverless Framework gives you everything you need to develop, deploy, monitor and secure serverless applications. We’ll be using it to speed up our development and reduce mental overhead.
> npm install -g serverless
> serverless -v
x.x.x
Let’s quickly go through how the Serverless Framework operates.
You can skip ahead if you’re just interested in seeing things working. Read on to learn more about the Serverless framework.
All of the Lambda functions and events in your Serverless service can be found in a configuration file called the serverless.yml. It defines a service with Functions and Events.
service: serverless-ethers
provider:
name: aws
runtime: nodejs12.x
environment:
CHAIN_ID: 3
DEFAULT_GAS_PRICE: 60000000000
functions:
myFunc:
handler: functions/myFunc.handler
events:
- schedule: rate(2 hours)
Under the
functions
property, you define your serverless functions. In the above example:myFunc
. handler
property points to the file and module containing the code you want to run in your function.events
property specifies Event triggers for the function to be executed.You can have multiple functions in a single service.
A Function is an AWS Lambda function. It’s an independent unit of deployment, like a microservice. It’s merely code, deployed in the cloud, that is most often written to perform a single job.
// functions/myFunc.js
exports.handler = async function(event, context) {
// Do anything
};
Functions are just normal JS functions. They can take an event object as payload.
Events are the things that trigger your functions to run. Events belong to each Function and can be found in the events property in
serverless.yml
.You can use the Scheduled Events trigger to automatically execute functions periodically. For example, to run the myFunc function every 2 hours we specify:
# serverless.yml
functions:
myFunc:
handler: functions/myFunc.handler
events:
- schedule: rate(2 hours)
You can also specify the schedule using cron schedule expressions:
# serverless.yml
events:
- schedule: cron(0 12 * * ? *) # 12PM UTC
If you are using AWS as your provider, all events in the service are anything in AWS that can trigger an AWS Lambda function, like:
That’s all you need to know for now.
To learn more about the Serverless framework, check out the docs.
With the Serverless Framework basics out of the way, let’s jump into the
service.serverless-ethers
is a fully-functional Serverless service that you can deploy and run out-of-the box.serverless-ethers
git clone [email protected]:yosriady/serverless-ethers.git
cd serverless-ethers
nvm use
npm install
You can use this project as a base for building custom smart contract automation. It comes preconfigured for AWS, but can be modified to work with other cloud providers such as GCP, Azure, and many others.
The
serverless-ethers
project is structured as follows:├── contracts/
│ ├── abis/
│ ├── abis.js
│ └── addresses.js
├── functions/
│ └── exec.js
└── serverless.yml
contracts/
contain smart contract ABIs and addresses.functions/
contain JS functions that implement the business logic. serverless.yml
describe the service’s configuration.Let’s look at each section in detail.
I’ve written and deployed a sample smart contract for testing purposes:
// SPDX-License-Identifier: GPL-3.0
pragma solidity ^0.6.10;
contract DummyStorage {
event Write(address indexed source, uint256 value);
uint internal _currentValue;
function get() public view returns (uint) {
return _currentValue;
}
function put(uint value) public {
emit Write(msg.sender, value);
_currentValue = value;
}
}
The
DummyStorage
smart contract has the following functions:get
is a read-only function that returns the contract’s current value.put
is a write function that updates the contract’s current value.The sample contract is verified and live on Ropsten. Feel free to use it to test your functions!
The contracts directory contains the ABIs of contracts the functions interact with. In the sample project, it contains the ABI for the DummyStorage contract.
├── contracts/
│ ├── abis/
│ │ └── DummyStorage.json
│ ├── abis.js
│ └── addresses.js
You can think of an ABI as a smart contract’s public API specification, kind of like an OpenAPI specification. You need the ABI to call a contract’s functions.
The contracts/ directory structure lets us import both the contract ABI and address like so:
// functions/exec.js
const { abis, addresses } = require('../contracts');
const DummyStorageABI = abis.DummyStorage;
const DummyStorageAddress = addresses.DummyStorage;
We’ll need these in our function.
The exec function uses Ethers to load contract ABIs and call a smart contract:
// Initialize contract
const contract = new ethers.Contract(
DummyStorageAddress,
DummyStorageABI,
wallet,
)
// Call smart contract function `put(uint)`
const RANDOM_INTEGER = Math.floor(Math.random() * 100); // returns a random integer from 0 to 99
const tx = await contract.put(RANDOM_INTEGER)
Loading the contract ABI and address gives us an
ethers.Contract
abstraction with all the functions of our smart contract, including get()
and put()
.In the sample
exec
function, we call contract.put()
with a random integer.Before you can run the exec function, you’ll need to specify some environment variables in your serverless.yml:
# serverless.yml
service: serverless-ethers
provider:
name: aws
runtime: nodejs12.x
region: ap-southeast-1
timeout: 30
environment:
DEFAULT_GAS_PRICE: 60000000000
MNEMONIC: ...
SLACK_HOOK_URL: ...
serverless-ethers
uses the following environment variables:DEFAULT_GAS_PRICE
: Default gas price used when making write transactions.MNEMONIC
: 12-word mnemonic used to derive an Ethereum address. Make sure it’s funded with Ether if you intend to write data to Ethereum!SLACK_HOOK_URL
: The example sends messages to Slack using Incoming Webhooks. You can get this URL from your Slack dashboard. (Optional)You can change your deployed function’s environment variables on the fly from the AWS Lambda console.
Important Note: make sure you do not store keys in plaintext in production. Use a secure parameter store such as AWS Secrets Manager when storing credentials such as mnemonics and API keys. Since every project has its own security requirements and setup, we leave it up to readers to decide how they want to approach storing secrets.
You can use the serverless CLI command invoke local to run your functions locally. This is great for testing!
> serverless invoke local -f exec
Starting...
Contract ABIs loaded
Ethers wallet loaded
Contract loaded
Sending transaction...
:white_check_mark: Transaction sent https://ropsten.etherscan.io/tx/0x72204f07911a319b4e5f7eb54ad15ed666cfc1403b53def40c9d60188b176383
Completed
true
Deploying is as easy as
serverless deplo
y:> serverless deploy
Serverless: Packaging service...
Serverless: Excluding development dependencies...
Serverless: Creating Stack...
Serverless: Checking Stack create progress...
........
Serverless: Stack create finished...
Serverless: Uploading CloudFormation file to S3...
Serverless: Uploading artifacts...
Serverless: Uploading service serverless-ethers.zip file to S3 (2.95 MB)...
Serverless: Validating template...
Serverless: Updating Stack...
Serverless: Checking Stack update progress...
.....................
Serverless: Stack update finished...
Service Information
service: serverless-ethers
stage: dev
region: ap-southeast-1
stack: serverless-ethers-dev
resources: 8
api keys:
None
endpoints:
None
functions:
exec: serverless-ethers-dev-exec
layers:
None
That’s it! You now have a live serverless function that you can use to automate and monitor your smart contracts. You can use this project as a base for building your own smart contract automation.
Congratulations! You learned about:
Feel free to let me know if you found this useful, or if you have any questions! I’d be interested to know what your automation use cases are.
The serverless-ethers sample application is open source and available on Github. Star the repo if you found it useful!
Included with serverless-ethers is a postToSlack function to help you integrate with Slack.
const successMessage = `:white_check_mark: Transaction sent https://ropsten.etherscan.io/tx/${tx.hash}`;
await postToSlack(successMessage);
The
postToSlack
function makes use of a SLACK_HOOK_URL
environment variable that you can get from your Slack console. Once set up, you’ll be able to notify Slack whenever a transaction was sent successfully:It’s a nice and simple way to monitor your functions.
So far, we’ve only implemented the ‘Automated Trigger’ use case. What about monitoring smart contract state and events?
You can use the Ethers v5 Events API to periodically monitor for certain events. In your function, you can do the following:
// Given the following Event:
// event Transfer(bytes32 indexed node, address owner)
// Get the filter (the second null could be omitted)
const filter = contract.filters.Transfer(userAccount, null);
// Query the filter
const logs = contract.queryFilter(filter, 0, "latest"); // from block 0 to latest block
// Print out all the values:
logs.forEach((log) => {
console.log(log.args._to, log.args._value);
});
Assuming that you want to have this function execute periodically (e.g. every 5 minutes), you’ll also need to store a flag that keeps track of the last block the function has seen since the last execution.
Event monitoring is especially useful if you have an Access Control whitelist that you need to keep an eye on. With an event monitoring function, you can notify a Slack channel whenever new addresses are whitelisted for admin roles. Very handy!
Previously published at https://yos.io/2020/07/04/serverless-smart-contract-automation/