Skip to content

Commit

Permalink
Merge branch 'master' into interfaces/erc/4906
Browse files Browse the repository at this point in the history
  • Loading branch information
Amxx authored Feb 17, 2023
2 parents 5b12be1 + d5d9d4b commit 31119cb
Show file tree
Hide file tree
Showing 10 changed files with 77 additions and 57 deletions.
1 change: 1 addition & 0 deletions .github/workflows/checks.yml
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ jobs:
runs-on: ubuntu-latest
env:
FORCE_COLOR: 1
NODE_OPTIONS: --max_old_space_size=4096
GAS: true
steps:
- uses: actions/checkout@v3
Expand Down
25 changes: 25 additions & 0 deletions .github/workflows/release-cycle.yml
Original file line number Diff line number Diff line change
Expand Up @@ -142,6 +142,11 @@ jobs:
run: bash scripts/release/workflow/pack.sh
env:
PRERELEASE: ${{ needs.state.outputs.is_prerelease }}
- name: Upload tarball artifact
uses: actions/upload-artifact@v3
with:
name: ${{ github.ref_name }}
path: ${{ steps.pack.outputs.tarball }}
- name: Tag
run: npx changeset tag
- name: Publish
Expand All @@ -158,6 +163,26 @@ jobs:
PRERELEASE: ${{ needs.state.outputs.is_prerelease }}
with:
script: await require('./scripts/release/workflow/github-release.js')({ github, context })
outputs:
tarball_name: ${{ steps.pack.outputs.tarball_name }}

integrity_check:
needs: publish
name: Tarball Integrity Check
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Download tarball artifact
id: artifact
# Replace with actions/upload-artifact@v3 when
# https://github.com/actions/download-artifact/pull/194 gets released
uses: actions/download-artifact@e9ef242655d12993efdcda9058dee2db83a2cb9b
with:
name: ${{ github.ref_name }}
- name: Check integrity
run: bash scripts/release/workflow/integrity-check.sh
env:
TARBALL: ${{ steps.artifact.outputs.download-path }}/${{ needs.publish.outputs.tarball_name }}

merge:
needs: state
Expand Down
23 changes: 11 additions & 12 deletions contracts/token/ERC721/extensions/ERC721Wrapper.sol
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,6 @@
pragma solidity ^0.8.0;

import "../ERC721.sol";
import "../utils/ERC721Holder.sol";

/**
* @dev Extension of the ERC721 token contract to support token wrapping.
Expand All @@ -14,7 +13,7 @@ import "../utils/ERC721Holder.sol";
*
* _Available since v4.9.0_
*/
abstract contract ERC721Wrapper is ERC721, ERC721Holder {
abstract contract ERC721Wrapper is ERC721, IERC721Receiver {
IERC721 private immutable _underlying;

constructor(IERC721 underlyingToken) {
Expand All @@ -25,11 +24,15 @@ abstract contract ERC721Wrapper is ERC721, ERC721Holder {
* @dev Allow a user to deposit underlying tokens and mint the corresponding tokenIds.
*/
function depositFor(address account, uint256[] memory tokenIds) public virtual returns (bool) {
bytes memory data = abi.encodePacked(account);

uint256 length = tokenIds.length;
for (uint256 i = 0; i < length; ++i) {
underlying().safeTransferFrom(_msgSender(), address(this), tokenIds[i], data);
uint256 tokenId = tokenIds[i];

// This is an "unsafe" transfer that doesn't call any hook on the receiver. With underlying() being trusted
// (by design of this contract) and no other contracts expected to be called from there, we are safe.
// slither-disable-next-line reentrancy-no-eth
underlying().transferFrom(_msgSender(), address(this), tokenId);
_safeMint(account, tokenId);
}

return true;
Expand Down Expand Up @@ -64,16 +67,12 @@ abstract contract ERC721Wrapper is ERC721, ERC721Holder {
* for recovering in that scenario.
*/
function onERC721Received(
address operator,
address,
address from,
uint256 tokenId,
bytes memory data
) public override returns (bytes4) {
bytes memory
) public virtual override returns (bytes4) {
require(address(underlying()) == _msgSender(), "ERC721Wrapper: caller is not underlying");
if (data.length > 0) {
require(data.length == 20 && operator == address(this), "ERC721Wrapper: Invalid data format");
from = address(bytes20(data));
}
_safeMint(from, tokenId);
return IERC721Receiver.onERC721Received.selector;
}
Expand Down
2 changes: 2 additions & 0 deletions docs/modules/ROOT/nav.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -19,3 +19,5 @@
* xref:crosschain.adoc[Crosschain]
* xref:utilities.adoc[Utilities]
* xref:subgraphs::index.adoc[Subgraphs]
20 changes: 20 additions & 0 deletions scripts/release/workflow/integrity-check.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
#!/usr/bin/env bash

set -euo pipefail

CHECKSUMS="$RUNNER_TEMP/checksums.txt"

# Extract tarball content into a tmp directory
tar xf "$TARBALL" -C "$RUNNER_TEMP"

# Move to extracted directory
cd "$RUNNER_TEMP/package"

# Checksum all Solidity files
find . -type f -name "*.sol" | xargs shasum > "$CHECKSUMS"

# Back to directory with git contents
cd "$GITHUB_WORKSPACE/contracts"

# Check against tarball contents
shasum -c "$CHECKSUMS"
1 change: 1 addition & 0 deletions scripts/release/workflow/pack.sh
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ dist_tag() {

cd contracts
TARBALL="$(npm pack | tee /dev/stderr | tail -1)"
echo "tarball_name=$TARBALL" >> $GITHUB_OUTPUT
echo "tarball=$(pwd)/$TARBALL" >> $GITHUB_OUTPUT
echo "tag=$(dist_tag)" >> $GITHUB_OUTPUT
cd ..
2 changes: 1 addition & 1 deletion scripts/release/workflow/publish.sh
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,6 @@ delete_tag() {

if [ "$TAG" = tmp ]; then
delete_tag "$TAG"
elif ["$TAG" = latest ]; then
elif [ "$TAG" = latest ]; then
delete_tag next
fi
15 changes: 9 additions & 6 deletions test/helpers/governance.js
Original file line number Diff line number Diff line change
Expand Up @@ -115,19 +115,22 @@ class GovernorHelper {
: this.governor.castVote(...concatOpts([proposal.id, vote.support], opts));
}

waitForSnapshot(offset = 0) {
async waitForSnapshot(offset = 0) {
const proposal = this.currentProposal;
return this.governor.proposalSnapshot(proposal.id).then(timepoint => forward[this.mode](timepoint.addn(offset)));
const timepoint = await this.governor.proposalSnapshot(proposal.id);
return forward[this.mode](timepoint.addn(offset));
}

waitForDeadline(offset = 0) {
async waitForDeadline(offset = 0) {
const proposal = this.currentProposal;
return this.governor.proposalDeadline(proposal.id).then(timepoint => forward[this.mode](timepoint.addn(offset)));
const timepoint = await this.governor.proposalDeadline(proposal.id);
return forward[this.mode](timepoint.addn(offset));
}

waitForEta(offset = 0) {
async waitForEta(offset = 0) {
const proposal = this.currentProposal;
return this.governor.proposalEta(proposal.id).then(timestamp => forward.timestamp(timestamp.addn(offset)));
const timestamp = await this.governor.proposalEta(proposal.id);
return forward.timestamp(timestamp.addn(offset));
}

/**
Expand Down
11 changes: 6 additions & 5 deletions test/helpers/time.js
Original file line number Diff line number Diff line change
@@ -1,16 +1,17 @@
const { time } = require('@openzeppelin/test-helpers');
const ozHelpers = require('@openzeppelin/test-helpers');
const helpers = require('@nomicfoundation/hardhat-network-helpers');

module.exports = {
clock: {
blocknumber: () => web3.eth.getBlock('latest').then(block => block.number),
timestamp: () => web3.eth.getBlock('latest').then(block => block.timestamp),
blocknumber: () => helpers.time.latestBlock(),
timestamp: () => helpers.time.latest(),
},
clockFromReceipt: {
blocknumber: receipt => Promise.resolve(receipt.blockNumber),
timestamp: receipt => web3.eth.getBlock(receipt.blockNumber).then(block => block.timestamp),
},
forward: {
blocknumber: time.advanceBlockTo,
timestamp: time.increaseTo,
blocknumber: ozHelpers.time.advanceBlockTo,
timestamp: helpers.time.increaseTo,
},
};
34 changes: 1 addition & 33 deletions test/token/ERC721/extensions/ERC721Wrapper.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -242,39 +242,7 @@ contract('ERC721Wrapper', function (accounts) {
);
});

describe('when data length is > 0', function () {
it('reverts with arbitrary data', async function () {
await expectRevert(
this.underlying.methods['safeTransferFrom(address,address,uint256,bytes)'](
initialHolder,
this.token.address,
firstTokenId,
'0x0123',
{
from: initialHolder,
},
),
'ERC721Wrapper: Invalid data format',
);
});

it('reverts with correct data from an untrusted operator', async function () {
await expectRevert(
this.underlying.methods['safeTransferFrom(address,address,uint256,bytes)'](
initialHolder,
this.token.address,
firstTokenId,
anotherAccount,
{
from: initialHolder,
},
),
'ERC721Wrapper: Invalid data format',
);
});
});

it('mints a token to from if no data is specified', async function () {
it('mints a token to from', async function () {
const { tx } = await this.underlying.safeTransferFrom(initialHolder, this.token.address, firstTokenId, {
from: initialHolder,
});
Expand Down

0 comments on commit 31119cb

Please sign in to comment.