Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
O
openzeppelin-contracts-upgradeable
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
俞永鹏
openzeppelin-contracts-upgradeable
Commits
fdf77a0d
Commit
fdf77a0d
authored
Dec 22, 2021
by
github-actions
Browse files
Options
Browse Files
Download
Plain Diff
Merge upstream master into patched/master
parents
a65b2bc6
10c8fcd3
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
42 additions
and
16 deletions
+42
-16
MinimalForwarder.sol
contracts/metatx/MinimalForwarder.sol
+9
-1
MinimalForwarder.test.js
test/metatx/MinimalForwarder.test.js
+33
-15
No files found.
contracts/metatx/MinimalForwarder.sol
View file @
fdf77a0d
...
@@ -50,9 +50,17 @@ contract MinimalForwarder is EIP712 {
...
@@ -50,9 +50,17 @@ contract MinimalForwarder is EIP712 {
(bool success, bytes memory returndata) = req.to.call{gas: req.gas, value: req.value}(
(bool success, bytes memory returndata) = req.to.call{gas: req.gas, value: req.value}(
abi.encodePacked(req.data, req.from)
abi.encodePacked(req.data, req.from)
);
);
// Validate that the relayer has sent enough gas for the call.
// Validate that the relayer has sent enough gas for the call.
// See https://ronan.eth.link/blog/ethereum-gas-dangers/
// See https://ronan.eth.link/blog/ethereum-gas-dangers/
assert(gasleft() > req.gas / 63);
if (gasleft() <= req.gas / 63) {
// We explicitly trigger invalid opcode to consume all gas and bubble-up the effects, since
// neither revert or assert consume all gas since Solidity 0.8.0
// https://docs.soliditylang.org/en/v0.8.0/control-structures.html#panic-via-assert-and-error-via-require
assembly {
invalid()
}
}
return (success, returndata);
return (success, returndata);
}
}
...
...
test/metatx/MinimalForwarder.test.js
View file @
fdf77a0d
...
@@ -6,6 +6,7 @@ const { expectRevert, constants } = require('@openzeppelin/test-helpers');
...
@@ -6,6 +6,7 @@ const { expectRevert, constants } = require('@openzeppelin/test-helpers');
const
{
expect
}
=
require
(
'chai'
);
const
{
expect
}
=
require
(
'chai'
);
const
MinimalForwarder
=
artifacts
.
require
(
'MinimalForwarder'
);
const
MinimalForwarder
=
artifacts
.
require
(
'MinimalForwarder'
);
const
CallReceiverMock
=
artifacts
.
require
(
'CallReceiverMock'
);
const
name
=
'MinimalForwarder'
;
const
name
=
'MinimalForwarder'
;
const
version
=
'0.0.1'
;
const
version
=
'0.0.1'
;
...
@@ -44,7 +45,7 @@ contract('MinimalForwarder', function (accounts) {
...
@@ -44,7 +45,7 @@ contract('MinimalForwarder', function (accounts) {
nonce
:
Number
(
await
this
.
forwarder
.
getNonce
(
this
.
sender
)),
nonce
:
Number
(
await
this
.
forwarder
.
getNonce
(
this
.
sender
)),
data
:
'0x'
,
data
:
'0x'
,
};
};
this
.
sign
=
ethSigUtil
.
signTypedMessage
(
this
.
sign
=
()
=>
ethSigUtil
.
signTypedMessage
(
this
.
wallet
.
getPrivateKey
(),
this
.
wallet
.
getPrivateKey
(),
{
{
data
:
{
data
:
{
...
@@ -65,7 +66,7 @@ contract('MinimalForwarder', function (accounts) {
...
@@ -65,7 +66,7 @@ contract('MinimalForwarder', function (accounts) {
});
});
it
(
'success'
,
async
function
()
{
it
(
'success'
,
async
function
()
{
expect
(
await
this
.
forwarder
.
verify
(
this
.
req
,
this
.
sign
)).
to
.
be
.
equal
(
true
);
expect
(
await
this
.
forwarder
.
verify
(
this
.
req
,
this
.
sign
()
)).
to
.
be
.
equal
(
true
);
});
});
afterEach
(
async
function
()
{
afterEach
(
async
function
()
{
...
@@ -76,27 +77,27 @@ contract('MinimalForwarder', function (accounts) {
...
@@ -76,27 +77,27 @@ contract('MinimalForwarder', function (accounts) {
context
(
'invalid signature'
,
function
()
{
context
(
'invalid signature'
,
function
()
{
it
(
'tampered from'
,
async
function
()
{
it
(
'tampered from'
,
async
function
()
{
expect
(
await
this
.
forwarder
.
verify
({
...
this
.
req
,
from
:
accounts
[
0
]
},
this
.
sign
))
expect
(
await
this
.
forwarder
.
verify
({
...
this
.
req
,
from
:
accounts
[
0
]
},
this
.
sign
()
))
.
to
.
be
.
equal
(
false
);
.
to
.
be
.
equal
(
false
);
});
});
it
(
'tampered to'
,
async
function
()
{
it
(
'tampered to'
,
async
function
()
{
expect
(
await
this
.
forwarder
.
verify
({
...
this
.
req
,
to
:
accounts
[
0
]
},
this
.
sign
))
expect
(
await
this
.
forwarder
.
verify
({
...
this
.
req
,
to
:
accounts
[
0
]
},
this
.
sign
()
))
.
to
.
be
.
equal
(
false
);
.
to
.
be
.
equal
(
false
);
});
});
it
(
'tampered value'
,
async
function
()
{
it
(
'tampered value'
,
async
function
()
{
expect
(
await
this
.
forwarder
.
verify
({
...
this
.
req
,
value
:
web3
.
utils
.
toWei
(
'1'
)
},
this
.
sign
))
expect
(
await
this
.
forwarder
.
verify
({
...
this
.
req
,
value
:
web3
.
utils
.
toWei
(
'1'
)
},
this
.
sign
()
))
.
to
.
be
.
equal
(
false
);
.
to
.
be
.
equal
(
false
);
});
});
it
(
'tampered nonce'
,
async
function
()
{
it
(
'tampered nonce'
,
async
function
()
{
expect
(
await
this
.
forwarder
.
verify
({
...
this
.
req
,
nonce
:
this
.
req
.
nonce
+
1
},
this
.
sign
))
expect
(
await
this
.
forwarder
.
verify
({
...
this
.
req
,
nonce
:
this
.
req
.
nonce
+
1
},
this
.
sign
()
))
.
to
.
be
.
equal
(
false
);
.
to
.
be
.
equal
(
false
);
});
});
it
(
'tampered data'
,
async
function
()
{
it
(
'tampered data'
,
async
function
()
{
expect
(
await
this
.
forwarder
.
verify
({
...
this
.
req
,
data
:
'0x1742'
},
this
.
sign
))
expect
(
await
this
.
forwarder
.
verify
({
...
this
.
req
,
data
:
'0x1742'
},
this
.
sign
()
))
.
to
.
be
.
equal
(
false
);
.
to
.
be
.
equal
(
false
);
});
});
it
(
'tampered signature'
,
async
function
()
{
it
(
'tampered signature'
,
async
function
()
{
const
tamperedsign
=
web3
.
utils
.
hexToBytes
(
this
.
sign
);
const
tamperedsign
=
web3
.
utils
.
hexToBytes
(
this
.
sign
()
);
tamperedsign
[
42
]
^=
0xff
;
tamperedsign
[
42
]
^=
0xff
;
expect
(
await
this
.
forwarder
.
verify
(
this
.
req
,
web3
.
utils
.
bytesToHex
(
tamperedsign
)))
expect
(
await
this
.
forwarder
.
verify
(
this
.
req
,
web3
.
utils
.
bytesToHex
(
tamperedsign
)))
.
to
.
be
.
equal
(
false
);
.
to
.
be
.
equal
(
false
);
...
@@ -112,7 +113,7 @@ contract('MinimalForwarder', function (accounts) {
...
@@ -112,7 +113,7 @@ contract('MinimalForwarder', function (accounts) {
});
});
it
(
'success'
,
async
function
()
{
it
(
'success'
,
async
function
()
{
await
this
.
forwarder
.
execute
(
this
.
req
,
this
.
sign
);
// expect to not revert
await
this
.
forwarder
.
execute
(
this
.
req
,
this
.
sign
()
);
// expect to not revert
});
});
afterEach
(
async
function
()
{
afterEach
(
async
function
()
{
...
@@ -124,36 +125,36 @@ contract('MinimalForwarder', function (accounts) {
...
@@ -124,36 +125,36 @@ contract('MinimalForwarder', function (accounts) {
context
(
'invalid signature'
,
function
()
{
context
(
'invalid signature'
,
function
()
{
it
(
'tampered from'
,
async
function
()
{
it
(
'tampered from'
,
async
function
()
{
await
expectRevert
(
await
expectRevert
(
this
.
forwarder
.
execute
({
...
this
.
req
,
from
:
accounts
[
0
]
},
this
.
sign
),
this
.
forwarder
.
execute
({
...
this
.
req
,
from
:
accounts
[
0
]
},
this
.
sign
()
),
'MinimalForwarder: signature does not match request'
,
'MinimalForwarder: signature does not match request'
,
);
);
});
});
it
(
'tampered to'
,
async
function
()
{
it
(
'tampered to'
,
async
function
()
{
await
expectRevert
(
await
expectRevert
(
this
.
forwarder
.
execute
({
...
this
.
req
,
to
:
accounts
[
0
]
},
this
.
sign
),
this
.
forwarder
.
execute
({
...
this
.
req
,
to
:
accounts
[
0
]
},
this
.
sign
()
),
'MinimalForwarder: signature does not match request'
,
'MinimalForwarder: signature does not match request'
,
);
);
});
});
it
(
'tampered value'
,
async
function
()
{
it
(
'tampered value'
,
async
function
()
{
await
expectRevert
(
await
expectRevert
(
this
.
forwarder
.
execute
({
...
this
.
req
,
value
:
web3
.
utils
.
toWei
(
'1'
)
},
this
.
sign
),
this
.
forwarder
.
execute
({
...
this
.
req
,
value
:
web3
.
utils
.
toWei
(
'1'
)
},
this
.
sign
()
),
'MinimalForwarder: signature does not match request'
,
'MinimalForwarder: signature does not match request'
,
);
);
});
});
it
(
'tampered nonce'
,
async
function
()
{
it
(
'tampered nonce'
,
async
function
()
{
await
expectRevert
(
await
expectRevert
(
this
.
forwarder
.
execute
({
...
this
.
req
,
nonce
:
this
.
req
.
nonce
+
1
},
this
.
sign
),
this
.
forwarder
.
execute
({
...
this
.
req
,
nonce
:
this
.
req
.
nonce
+
1
},
this
.
sign
()
),
'MinimalForwarder: signature does not match request'
,
'MinimalForwarder: signature does not match request'
,
);
);
});
});
it
(
'tampered data'
,
async
function
()
{
it
(
'tampered data'
,
async
function
()
{
await
expectRevert
(
await
expectRevert
(
this
.
forwarder
.
execute
({
...
this
.
req
,
data
:
'0x1742'
},
this
.
sign
),
this
.
forwarder
.
execute
({
...
this
.
req
,
data
:
'0x1742'
},
this
.
sign
()
),
'MinimalForwarder: signature does not match request'
,
'MinimalForwarder: signature does not match request'
,
);
);
});
});
it
(
'tampered signature'
,
async
function
()
{
it
(
'tampered signature'
,
async
function
()
{
const
tamperedsign
=
web3
.
utils
.
hexToBytes
(
this
.
sign
);
const
tamperedsign
=
web3
.
utils
.
hexToBytes
(
this
.
sign
()
);
tamperedsign
[
42
]
^=
0xff
;
tamperedsign
[
42
]
^=
0xff
;
await
expectRevert
(
await
expectRevert
(
this
.
forwarder
.
execute
(
this
.
req
,
web3
.
utils
.
bytesToHex
(
tamperedsign
)),
this
.
forwarder
.
execute
(
this
.
req
,
web3
.
utils
.
bytesToHex
(
tamperedsign
)),
...
@@ -161,6 +162,23 @@ contract('MinimalForwarder', function (accounts) {
...
@@ -161,6 +162,23 @@ contract('MinimalForwarder', function (accounts) {
);
);
});
});
});
});
it
(
'bubble out of gas'
,
async
function
()
{
const
receiver
=
await
CallReceiverMock
.
new
();
const
gasAvailable
=
100000
;
this
.
req
.
to
=
receiver
.
address
;
this
.
req
.
data
=
receiver
.
contract
.
methods
.
mockFunctionOutOfGas
().
encodeABI
();
this
.
req
.
gas
=
1000000
;
await
expectRevert
.
assertion
(
this
.
forwarder
.
execute
(
this
.
req
,
this
.
sign
(),
{
gas
:
gasAvailable
}),
);
const
{
transactions
}
=
await
web3
.
eth
.
getBlock
(
'latest'
);
const
{
gasUsed
}
=
await
web3
.
eth
.
getTransactionReceipt
(
transactions
[
0
]);
expect
(
gasUsed
).
to
.
be
.
equal
(
gasAvailable
);
});
});
});
});
});
});
});
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment