Security Audit
March 13th, 2023
Version 1.0.0
Presented by 0xMacro
This document includes the results of the security audit for thirdweb's smart contract code as found in the section titled ‘Source Code’. The security audit was performed by the Macro security team from Feburary 6th, 2022 to Feburary 15th, 2023.
The purpose of this audit is to review the source code of certain thirdweb Solidity contracts, and provide feedback on the design, architecture, and quality of the source code with an emphasis on validating the correctness and security of the software in its entirety.
Disclaimer: While Macro’s review is comprehensive and has surfaced some changes that should be made to the source code, this audit should not solely be relied upon for security, as no single audit is guaranteed to catch all possible bugs.
The following is an aggregation of issues found by the Macro Audit team:
Severity | Count | Acknowledged | Won't Do | Addressed |
---|---|---|---|---|
High | 1 | - | - | 1 |
Medium | 5 | - | - | 5 |
Low | 1 | - | - | 1 |
Code Quality | 2 | - | - | 2 |
Gas Optimization | 4 | - | 2 | 2 |
thirdweb was quick to respond to these issues.
Our understanding of the specification was based on the following sources:
The following source code was reviewed during the audit:
fc068ba9ed7d034544d76deb07c8296046b2f26d
8e67c4e3406f1826461c939c2a2f2413e6574860
We audited the following contracts specificially for the Airdrop audit:
Contract | SHA256 |
---|---|
contracts/airdrop/AirdropERC1155.sol |
|
contracts/airdrop/AirdropERC20.sol |
|
contracts/airdrop/AirdropERC721.sol |
|
contracts/interfaces/airdrop/IAirdropERC1155.sol |
|
contracts/interfaces/airdrop/IAirdropERC20.sol |
|
contracts/interfaces/airdrop/IAirdropERC721.sol |
|
contracts/lib/CurrencyTransferLib.sol |
|
contracts/openzeppelin-presets/token/ERC20/utils/SafeERC20.sol |
|
contracts/extension/Ownable.sol |
|
contracts/extension/PermissionsEnumerable.sol |
|
contracts/extension/interface/IPermissionsEnumerable.sol |
|
We audited the following contracts specificially for the Multichain Registry audit:
Contract | SHA256 |
---|---|
contracts/plugin/TWRouter.sol |
|
contracts/plugin/interface/IDefaultPluginSet.sol |
|
contracts/plugin/interface/IPlugin.sol |
|
contracts/plugin/interface/IPluginRegistry.sol |
|
contracts/plugin/interface/ITWRouter.sol |
|
contracts/registry/TWMultichainRegistry.sol |
|
contracts/extension/Initializable.sol |
|
contracts/extension/interface/IPermissionsEnumerable.sol |
|
contracts/extension/PermissionsEnumerable.sol |
|
contracts/openzeppelin-presets/utils/EnumerableSet.sol |
|
contracts/lib/TWStringSet.sol |
|
contracts/extension/interface/IMulticall.sol |
|
contracts/extension/Multicall.sol |
|
contracts/lib/TWAddress.sol |
|
contracts/plugin/DefaultPluginSet.sol |
|
contracts/plugin/PluginRegistry.sol |
|
contracts/plugin/PluginState.sol |
|
contracts/plugin/Router.sol |
|
contracts/interfaces/ITWMultichainRegistry.sol |
|
contracts/extension/interface/IPermissions.sol |
|
contracts/extension/interface/IERC2771Context.sol |
|
contracts/registry/plugin/MultichainRegistryCore.sol |
|
Note: This document contains an audit solely of the Solidity contracts listed above. Specifically, the audit pertains only to the contracts themselves, and does not pertain to any other programs or scripts, including deployment scripts.
Click on an issue to jump to it, or scroll down to see them all.
processPayments()
can be griefed
We quantify issues in three parts:
This third part – the severity level – is a summary of how much consideration the client should give to fixing the issue. We assign severity according to the table of guidelines below:
Severity | Description |
---|---|
(C-x) Critical |
We recommend the client must fix the issue, no matter what, because not fixing would mean significant funds/assets WILL be lost. |
(H-x) High |
We recommend the client must address the issue, no matter what, because not fixing would be very bad, or some funds/assets will be lost, or the code’s behavior is against the provided spec. |
(M-x) Medium |
We recommend the client to seriously consider fixing the issue, as the implications of not fixing the issue are severe enough to impact the project significantly, albiet not in an existential manner. |
(L-x) Low |
The risk is small, unlikely, or may not relevant to the project in a meaningful way. Whether or not the project wants to develop a fix is up to the goals and needs of the project. |
(Q-x) Code Quality |
The issue identified does not pose any obvious risk, but fixing could improve overall code quality, on-chain composability, developer ergonomics, or even certain aspects of protocol design. |
(I-x) Informational |
Warnings and things to keep in mind when operating the protocol. No immediate action required. |
(G-x) Gas Optimizations |
The presented optimization suggestion would save an amount of gas significant enough, in our opinion, to be worth the development cost of implementing it. |
AirdropERC721.sol, AirdropERC20.sol, and AirdropERC1155.sol, each have the function addRecipients
, which with some variance adds airdrop recipients to a mapping, that will be used to issue airdrops. Since there is a lot of data being stored, multiple calls to this function may be required to set all the desired airdrops.
However, on calls after the first, the loop that adds recipients to the mapping is not properly setup, and may not be executed.
(Provided code is from AirdropERC721.sol, but the same issue is present in AirdropERC20.sol and AirdropERC1155.sol)
function addRecipients(AirdropContent[] calldata _contents) external onlyRole(DEFAULT_ADMIN_ROLE) {
uint256 len = _contents.length;
require(len > 0, "No payees provided.");
uint256 currentCount = payeeCount;
payeeCount += len;
for (uint256 i = currentCount; i < len; i += 1) {
airdropContent[i] = _contents[i];
}
emit RecipientsAdded(_contents);
}
Since after the first call to addRecipients
the payeeCount
will be non-zero, and potentially greater than the length of _contents
. At the start of the loop i
is set to currentCount
but the condition i < len
expects i
to be zero in order to loop through all of _contents
.
This leads to some or all of the airdrop recipients to not be included in the drop, while also increasing the payeeCount
by number of recipients that were supposed to be added to the airdropContent
mapping.
A couple issues arise from this bug:
getAllAirdropPayments
, getAllAirdropPaymentsPending
, getAllAirdropPaymentsProcessed
, getAllAirdropPaymentsFailed
, or getAllAirdropPaymentsCancelled
. This could lead to problems for contracts that query and use this data, as well as effect front end applications that use it.processPayments
would revert when attempting to transfer as the tokenAddress of these missed recipients would be the zero address. The first recipients added can receive their airdrop, if processPayments
is called with paymentsToProcess
equal to the number of recipients initially added, but after that processPayments
will no longer be able to be called.Remediations to Consider
At the start of the loop set i to be equal to zero, and determine the index to store the recipient in the airdropContent mapping to be currentCount + i.
Some of the well known “ERC20” tokens - such as USDT - don’t fully comply to the ERC-20 standard and don’t return a boolean on e.g. transfer
and transferFrom
. For such tokens, AirdropERC20::_transferCurrencyWithReturnVal
will fail on the following line:
try IERC20(_currency).transferFrom(_from, _to, _amount) returns (bool success_) {
Remediations to Consider
In such cases, we usually recommend using safeTransferFrom
from SafeERC20
library.
However, in this specific case, library calls can’t be wrapped around try/catch and thus making it
difficult to implement a failsafe using try/catch. Instead of using safeTransferFrom
, consider
using the low level .call
method and implement error handling according to your needs.
In AirdropERC721.sol, AirdropERC20.sol, and AirdropERC1155.sol, resetRecipients()
is called to cancel pending airdrops, and does so by setting processedCount
to payeeCount
, preventing any unprocessed airdrops from being processed. It also loops through each item canceled and sets its index in the isCancelled
mapping to true, allowing getters to determine if an airdrop was canceled.
Looping through and setting the isCancelled
mapping for each item is quite costly as it is a separate storage write for each item. In the case where a large airdrop was added using addRecipients()
and contained an error, it would cost the owner of the contract a lot of gas to cancel all the items.
It is also possible that if the amount of unprocessed airdrop recipients is very large (~1350 pending recipients), it may be impossible to call resetRecipients()
, as it will try to cancel all pending airdrops, causing the transaction cost to exceed the block gas limit.
Remediations to Consider
processPayments
, processing up to the airdrop causing issues, then call resetRecipients()
with a value of 1 to cancel the airdrop, then continue processing afterwords, thus preventing having to cancel all airdrops after the issue, and then having to reinsert the old airdrops.isCancelled
, store a range of indexes that represent the canceled items, as cancelation is always done in a sequence. Doing so would allow resetRecipients()
to only require 2-3 storage writes, in some cases substantially reducing gas costs.processPayments()
can be griefed
In AirdropERC721.sol, AirdropERC20.sol, and AirdropERC1155.sol, processPayments()
airdrops tokens to recipients set prior to the call. For AirdropERC721.sol, and AirdropERC1155.sol, safeTransferFrom
is used to transfer their respective tokens, which if the recipient is a contract it calls either onERC721Received
, or onERC1155Received
, ensuring the contract can properly handle owning tokens, but as a side effect it gives execution control over to the contract, which can execute malicious code. For AirdropERC20.sol, execution control is handed over if the token being sent is the native token of the blockchain, and the recipient is a contract that has implemented a receive()
fallback function.
In the case where tokens are sent to a contract that implements one of these hooks maliciously, there is little the contract can do other than consume all the gas provided. If this is done, then calls to processPayments()
could cost substantially more than expected, or exceed the block gas limit, preventing calls from executing if processing the malicious recipient.
Currently the resolution would be to call resetRecipients()
to clear the pending airdrops, then submit the recipients again by calling addRecipients()
with the unprocessed recipients, omitting the griefer. Doing this is rather costly, especially if the griefer is queued before many other recipients.
Remediations to Consider
PluginState.sol::_addPlugin
explicitly checks that a new function selector being added doesn’t already exist in the pluginMetadata
mapping - thus avoiding function clashes between different plugins.
require(
data.pluginMetadata[_plugin.functions[i].functionSelector].implementation == address(0),
"PluginState: plugin already exists for function."
);
However, this check is not being done in the _updatePlugin
function, allowing to override existing function selectors.
Consider the following scenario:
d()
e()
d()
.The last step overrides the mapping for d()
pointing to PluginD, so that d()
now points to PluginE. As a consequence, TWRouter
forwards any calls targeting function d()
to PluginE.
Moreover, deleting PluginA at this state would remove the d()
function selector from the pluginMetadata
mapping, also making the d()
function of PluginE unreachable.
Remediations to Consider
Consider adding above require check to the _updatePlugin
function as well.
MultichainRegistryCore.sol
should allow trusted forwarders to call add(…)
and remove(…)
on behalf of the msg.sender
; as implemented here:
function _msgSender() internal view returns (address sender) {
if (IERC2771Context(address(this)).isTrustedForwarder(msg.sender)) {
// The assembly code is more direct than the Solidity version using `abi.decode`.
assembly {
sender := shr(96, calldataload(sub(calldatasize(), 20)))
}
} else {
return msg.sender;
}
}
In order to support trusted forwarders - and as seen in the above code - ERC2771Context
has to be implemented by TWMultichainRegistry
. This can be achieved either via directly inheriting from ERC2771Context
, or in the current ThirdWeb setup, by adding ERC2771Context
implementation as a plugin.
The current setup involves:
ERC2771Context
contract with trusted forwardersERC2771Context
contract as a plugin to TWMultichainRegistry
.In step 1, when the ERC2771Context
plugin is being created, the constructor stores the forwarders array into storage - using the storage layout of ERC2771Context
contract. In step 2, the plugin is added to TWMultichainRegistry
, thus forwarding all isTrustedForwarder()
calls to the plugin.
When e.g. an add(..)
call (which forwards to MultichainRegistryCore.add(..)
) is now initiated via the proxy, it runs under the proxy’s execution context and uses the proxy’s storage layout. However, the forwarders haven’t been set in the proxy’s storage layout, resulting in the following check to always return false:
IERC2771Context(address(this)).isTrustedForwarder(msg.sender)
Remediations to Consider
Consider implementing the following changed:
TWMultichainRegistry
to directly inherit from ERC2771Context
contract._setupForwarders
to the ERC2771Context
contract which adds the trusted forwarders to storage. _setupForwarders
from within TWMultichainRegistry::intialize
(similar to how _setupRole
is called).By doing so, the trusted forwarders will be correctly stored in the proxy’s storage layout.
The event PaymentsResetByAdmin
is defined in the corresponding interface for all three AirdropERC20/721/1155 contracts:
/// @notice Emitted when pending payments are cancelled, and processed count is reset.
event PaymentsResetByAdmin();
However, it is not being emitted int the resetRecipients
function.
Remediations to Consider
Emit the PaymentResetByAdmin
event at the end of the resetRecipients
function.
In AirdropERC20.sol, SafeERC20
is imported:
import "../openzeppelin-presets/token/ERC20/utils/SafeERC20.sol";
and referenced here:
using SafeERC20 for IERC20;
Remediations to Consider
remove the above lines as SafeERC20
is not used anywhere in code. Instead use:
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
TWMultichainRegistry’s purpose is to hold a global registry of deployments and thus intended to be deployed only once by ThirdWeb itself. In this manner, deploying the contract via the Clone pattern only adds complexity and gas costs, without any benefits.
Remediations to Consider
Deploy the contract without using the Clone pattern (and thus having a proxy in front). In this case all the logic for setting up the roles inside the initialize
function can be moved to the constructor.
AirdropERC721.sol, AirdropERC20.sol, and AirdropERC1155.sol each use a variant of AirdropContent
to store data in the airdropContent mapping to determine how to process airdrops. Some of this data is necessary, but in the case of both tokenAddress
and tokenOwner
, this data may be the same for an entire airdrop.
In the case where a collection owner want to airdrop 10k NFTs using AirdropERC721.sol’s addRecipients()
to store the data of each recipient, the tokenAddress
and tokenOwner
would be the same for each entry, causing many unnecessary storage writes.
Remediations to Consider
Instead of storing the tokenAddress
and tokenOwner
for each airdrop recipient, this data could be stored in a separate mapping, along with a start and end range for the indexes this data applies. Doing so would only require a single write for tokenAddress
and tokenOwner
per airdrop, substantially reducing the cost of calling addRecipients()
when airdropping multiple tokens of the same collection and owner.
Typically code run in the constructor is not stored in bytecode, and it is expected to be the same when code from a library is only used within the constructor.
However, recently it was found that there is an error causing code from libraries to be added to the bytecode when it is only used in the constructor, adding unnecessary bytecode and increasing deployment costs..
In the case of when TWProxy.sol is deployed, it results with a bytecode size of 668 bytes, but with all references to the Address.sol library removed from the constructor, the deployed bytecode can be brought down to 180 bytes.
Remediations to Consider
Remove references to libraries in constructors if that is the only place the functions are used, to save on deployment costs.
AirdropERC721.sol, AirdropERC20.sol, and AirdropERC1155.sol each use loops for setting recipients and airdropping tokens.
In cases of large airdrops, there are many math operations that run, each checking to make sure no underflow/overflows occur, these checks consume gas and can add up, and are unnecessary if there is no way to underflow or overflow.
Remediations to Consider
Wrap operations that will not overflow/underflow in an unchecked block in order to save gas for users, especially for what is considered to be safe is wrapping i++
in an unchecked block like: unchecked {i++;}
..
AirdropERC721.sol, AirdropERC20.sol, and AirdropERC1155.sol emit events when addRecipients()
, processPayments()
that includes the data of the AirdropContent
struct being added/processed for each recipient.
However, since this data is stored on chain, and can be queried for when needed, there is no need to emit this data.
Remediations to Consider
Instead of emitting the contents of an airdrop, emit the indexes of the content.
Macro makes no warranties, either express, implied, statutory, or otherwise, with respect to the services or deliverables provided in this report, and Macro specifically disclaims all implied warranties of merchantability, fitness for a particular purpose, noninfringement and those arising from a course of dealing, usage or trade with respect thereto, and all such warranties are hereby excluded to the fullest extent permitted by law.
Macro will not be liable for any lost profits, business, contracts, revenue, goodwill, production, anticipated savings, loss of data, or costs of procurement of substitute goods or services or for any claim or demand by any other party. In no event will Macro be liable for consequential, incidental, special, indirect, or exemplary damages arising out of this agreement or any work statement, however caused and (to the fullest extent permitted by law) under any theory of liability (including negligence), even if Macro has been advised of the possibility of such damages.
The scope of this report and review is limited to a review of only the code presented by the thirdweb team and only the source code Macro notes as being within the scope of Macro’s review within this report. This report does not include an audit of the deployment scripts used to deploy the Solidity contracts in the repository corresponding to this audit. Specifically, for the avoidance of doubt, this report does not constitute investment advice, is not intended to be relied upon as investment advice, is not an endorsement of this project or team, and it is not a guarantee as to the absolute security of the project. In this report you may through hypertext or other computer links, gain access to websites operated by persons other than Macro. Such hyperlinks are provided for your reference and convenience only, and are the exclusive responsibility of such websites’ owners. You agree that Macro is not responsible for the content or operation of such websites, and that Macro shall have no liability to your or any other person or entity for the use of third party websites. Macro assumes no responsibility for the use of third party software and shall have no liability whatsoever to any person or entity for the accuracy or completeness of any outcome generated by such software.