文件 1 的 5:ArtistProxy.sol
pragma solidity >=0.6.2;
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import "@openzeppelin/contracts/utils/cryptography/ECDSA.sol";
interface IBrainDrops {
function mint(address recipient, uint _projectId) external payable returns (uint256);
function updateProjectArtistName(uint256 _projectId, string memory _projectArtistName) external;
function updateProjectDescription(uint256 _projectId, string memory _projectDescription) external;
function updateProjectWebsite(uint256 _projectId, string memory _projectWebsite) external;
function updateProjectLicense(uint256 _projectId, string memory _projectLicense) external;
function updateProjectBaseURI(uint256 _projectId, string memory _projectBaseURI) external;
function updateProjectPricePerTokenInWei(uint256 _projectId, uint256 _pricePerTokenInWei) external;
function toggleProjectIsPaused(uint256 _projectId) external;
function setProvenanceHash(uint256 _projectId, string memory provenanceHash) external;
function balanceOf(address owner) external view returns (uint256 balance);
function ownerOf(uint256 tokenId) external view returns (address owner);
function isWhitelisted(address sender) external view returns (bool whitelisted);
function transferFrom(address from, address to, uint256 tokenId) external;
}
contract ArtistProxy is Ownable, ReentrancyGuard {
using ECDSA for bytes32;
constructor(address _braindropsAddress) {
braindrops = IBrainDrops(_braindropsAddress);
}
IBrainDrops public braindrops;
mapping(uint256 => mapping(address => bool)) public projectIdToGenesisDropAddressMinted;
mapping(uint256 => mapping(address => bool)) public projectIdToProjectAntiBotAddressMinted;
mapping(uint256 => address) public projectIdToArtistAddress;
mapping(uint256 => bool) public projectIdToProjectActivated;
mapping(uint256 => bool) public projectIdToProjectAntiBotActivated;
mapping(uint256 => bool) public projectIdToHolderActivated;
mapping(uint256 => bool) public projectIdToGenesisDropActivated;
mapping(uint256 => uint256) public projectIdToOlderProjectId;
mapping(uint256 => uint256) public projectIdToBurnableProjectId;
mapping(uint256 => address) public projectIdToBotPreventionAddress;
address public signingAddress;
modifier onlyArtist(uint256 _projectId) {
require(msg.sender == projectIdToArtistAddress[_projectId], "Only artist");
_;
}
modifier onlyWhitelisted() {
require(braindrops.isWhitelisted(msg.sender), "Only whitelisted");
_;
}
modifier onlyHolders(uint256 _projectId) {
require(braindrops.balanceOf(msg.sender) > 0, "Holders only");
_;
}
function setSigningAddress(address _signingAddress) public onlyWhitelisted {
signingAddress = _signingAddress;
}
function setArtist(uint projectId, address artistAddress) public onlyWhitelisted {
projectIdToArtistAddress[projectId] = artistAddress;
}
function updateProjectArtistName(uint256 _projectId, string memory _projectArtistName) onlyArtist(_projectId) public {
braindrops.updateProjectArtistName(_projectId, _projectArtistName);
}
function updateProjectDescription(uint256 _projectId, string memory _projectDescription) onlyArtist(_projectId) public {
braindrops.updateProjectDescription(_projectId, _projectDescription);
}
function updateProjectWebsite(uint256 _projectId, string memory _projectWebsite) onlyArtist(_projectId) public {
braindrops.updateProjectWebsite(_projectId, _projectWebsite);
}
function updateProjectLicense(uint256 _projectId, string memory _projectLicense) onlyArtist(_projectId) public {
braindrops.updateProjectLicense(_projectId, _projectLicense);
}
function updateProjectBaseURI(uint256 _projectId, string memory _projectBaseURI) onlyArtist(_projectId) public {
braindrops.updateProjectBaseURI(_projectId, _projectBaseURI);
}
function updateProjectPricePerTokenInWei(uint256 _projectId, uint256 _pricePerTokenInWei) onlyArtist(_projectId) public {
braindrops.updateProjectPricePerTokenInWei(_projectId, _pricePerTokenInWei);
}
function toggleProjectIsPaused(uint256 _projectId) public onlyArtist(_projectId) {
braindrops.toggleProjectIsPaused(_projectId);
}
function setProvenanceHash(uint256 _projectId, string memory provenanceHash) public onlyArtist(_projectId) {
braindrops.setProvenanceHash(_projectId, provenanceHash);
}
function toggleProjectIsActive(uint256 _projectId) public onlyArtist(_projectId) {
projectIdToProjectActivated[_projectId] = !projectIdToProjectActivated[_projectId];
}
function toggleProjectAntiBotActive(uint256 _projectId) public onlyArtist(_projectId) {
projectIdToProjectAntiBotActivated[_projectId] = !projectIdToProjectAntiBotActivated[_projectId];
}
function toggleProjectIsHolderActive(uint256 _projectId) public onlyArtist(_projectId) {
projectIdToHolderActivated[_projectId] = !projectIdToHolderActivated[_projectId];
}
function toggleProjectIsGenesisDropActive(uint256 _projectId) public onlyArtist(_projectId) {
projectIdToGenesisDropActivated[_projectId] = !projectIdToGenesisDropActivated[_projectId];
}
function setProjectIdToOlderProjectId(uint256 _projectId, uint256 _olderProjectId) public onlyArtist(_projectId) {
projectIdToOlderProjectId[_projectId] = _olderProjectId;
}
function setProjectIdToBurnableProjectId(uint256 _projectId, uint256 _olderProjectId) public onlyArtist(_projectId) {
projectIdToBurnableProjectId[_projectId] = _olderProjectId;
}
function _validateServerSignedMessage(bytes32 message, bytes calldata signature, uint _projectId) internal virtual {
require(projectIdToProjectAntiBotAddressMinted[_projectId][msg.sender] == false, "Cannot replay transaction");
bytes32 expectedMessage = keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", '20', msg.sender));
require(message == expectedMessage, "Malformed message");
address signer = message.recover(signature);
require(signer == signingAddress, "Invalid signature");
projectIdToProjectAntiBotAddressMinted[_projectId][msg.sender] = true;
}
function mintForServerSigned(address recipient, uint _projectId, bytes32 message, bytes calldata signature)
public
payable
nonReentrant
returns (uint256)
{
require(projectIdToProjectAntiBotActivated[_projectId], "Project must be active for anti-bot minting");
_validateServerSignedMessage(message, signature, _projectId);
return braindrops.mint{value: msg.value}(recipient, _projectId);
}
function mintForProjectBurnersOnly(address recipient, uint _projectId, uint _tokenId)
public
payable
nonReentrant
returns (uint256)
{
uint burnableProjectId = projectIdToBurnableProjectId[_projectId];
require(burnableProjectId > 0, "Project must be active for project-holder burn-mints");
uint projectId = (_tokenId - (_tokenId % 1000000)) / 1000000;
require(projectId == burnableProjectId, "token must belong to the project to burn");
braindrops.transferFrom(msg.sender, address(this), _tokenId);
return braindrops.mint{value: msg.value}(recipient, _projectId);
}
function mintForArtistsOnly(address recipient, uint _projectId)
public
payable
onlyArtist(_projectId)
returns (uint256)
{
return braindrops.mint{value: msg.value}(recipient, _projectId);
}
function mintForProjectSpecificHoldersOnly(address recipient, uint _projectId, uint _projectTokenId)
public
payable
nonReentrant
returns (uint256)
{
uint olderProjectId = projectIdToOlderProjectId[_projectId];
require(olderProjectId > 0, "Project must be active for project-holder specific mints");
uint _projectIdFromTokenId = (_projectTokenId - (_projectTokenId % 1000000)) / 1000000;
require(_projectIdFromTokenId == olderProjectId, "must pass in a token id from the correct project");
require(braindrops.ownerOf(_projectTokenId) == msg.sender, "sender must own token id passed in");
return braindrops.mint{value: msg.value}(recipient, _projectId);
}
function mintForGenesisDropHoldersOnly(address recipient, uint _projectId, uint _project1TokenId, uint _project2TokenId, uint _project3TokenId)
public
payable
nonReentrant
returns (uint256)
{
require(projectIdToGenesisDropActivated[_projectId], "Project must be active for genesis set holders");
require(projectIdToGenesisDropAddressMinted[_projectId][msg.sender] == false, "One mint per address");
uint _project1Id = (_project1TokenId - (_project1TokenId % 1000000)) / 1000000;
require(_project1Id == 1, "must pass in a token id from project 1");
uint _project2Id = (_project2TokenId - (_project2TokenId % 1000000)) / 1000000;
require(_project2Id == 2, "must pass in a token id from project 2");
uint _project3Id = (_project3TokenId - (_project3TokenId % 1000000)) / 1000000;
require(_project3Id == 3, "must pass in a token id from project 3");
require(braindrops.ownerOf(_project1TokenId) == msg.sender, "must own the selected token from project 1");
require(braindrops.ownerOf(_project2TokenId) == msg.sender, "must own the selected token from project 2");
require(braindrops.ownerOf(_project3TokenId) == msg.sender, "must own the selected token from project 3");
projectIdToGenesisDropAddressMinted[_projectId][msg.sender] = true;
return braindrops.mint{value: msg.value}(recipient, _projectId);
}
function mintForHoldersOnly(address recipient, uint _projectId)
public
payable
nonReentrant
onlyHolders(_projectId)
returns (uint256)
{
require(projectIdToHolderActivated[_projectId], "Project must be active for holders");
return braindrops.mint{value: msg.value}(recipient, _projectId);
}
function mint(address recipient, uint _projectId)
public
payable
nonReentrant
returns (uint256)
{
require(projectIdToProjectActivated[_projectId], "Project must be active");
return braindrops.mint{value: msg.value}(recipient, _projectId);
}
}
文件 2 的 5:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 3 的 5:ECDSA.sol
pragma solidity ^0.8.0;
library ECDSA {
enum RecoverError {
NoError,
InvalidSignature,
InvalidSignatureLength,
InvalidSignatureS,
InvalidSignatureV
}
function _throwError(RecoverError error) private pure {
if (error == RecoverError.NoError) {
return;
} else if (error == RecoverError.InvalidSignature) {
revert("ECDSA: invalid signature");
} else if (error == RecoverError.InvalidSignatureLength) {
revert("ECDSA: invalid signature length");
} else if (error == RecoverError.InvalidSignatureS) {
revert("ECDSA: invalid signature 's' value");
} else if (error == RecoverError.InvalidSignatureV) {
revert("ECDSA: invalid signature 'v' value");
}
}
function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) {
if (signature.length == 65) {
bytes32 r;
bytes32 s;
uint8 v;
assembly {
r := mload(add(signature, 0x20))
s := mload(add(signature, 0x40))
v := byte(0, mload(add(signature, 0x60)))
}
return tryRecover(hash, v, r, s);
} else if (signature.length == 64) {
bytes32 r;
bytes32 vs;
assembly {
r := mload(add(signature, 0x20))
vs := mload(add(signature, 0x40))
}
return tryRecover(hash, r, vs);
} else {
return (address(0), RecoverError.InvalidSignatureLength);
}
}
function recover(bytes32 hash, bytes memory signature) internal pure returns (address) {
(address recovered, RecoverError error) = tryRecover(hash, signature);
_throwError(error);
return recovered;
}
function tryRecover(
bytes32 hash,
bytes32 r,
bytes32 vs
) internal pure returns (address, RecoverError) {
bytes32 s;
uint8 v;
assembly {
s := and(vs, 0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff)
v := add(shr(255, vs), 27)
}
return tryRecover(hash, v, r, s);
}
function recover(
bytes32 hash,
bytes32 r,
bytes32 vs
) internal pure returns (address) {
(address recovered, RecoverError error) = tryRecover(hash, r, vs);
_throwError(error);
return recovered;
}
function tryRecover(
bytes32 hash,
uint8 v,
bytes32 r,
bytes32 s
) internal pure returns (address, RecoverError) {
if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
return (address(0), RecoverError.InvalidSignatureS);
}
if (v != 27 && v != 28) {
return (address(0), RecoverError.InvalidSignatureV);
}
address signer = ecrecover(hash, v, r, s);
if (signer == address(0)) {
return (address(0), RecoverError.InvalidSignature);
}
return (signer, RecoverError.NoError);
}
function recover(
bytes32 hash,
uint8 v,
bytes32 r,
bytes32 s
) internal pure returns (address) {
(address recovered, RecoverError error) = tryRecover(hash, v, r, s);
_throwError(error);
return recovered;
}
function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) {
return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", hash));
}
function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) {
return keccak256(abi.encodePacked("\x19\x01", domainSeparator, structHash));
}
}
文件 4 的 5:Ownable.sol
pragma solidity ^0.8.0;
import "../utils/Context.sol";
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_setOwner(_msgSender());
}
function owner() public view virtual returns (address) {
return _owner;
}
modifier onlyOwner() {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
_;
}
function renounceOwnership() public virtual onlyOwner {
_setOwner(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_setOwner(newOwner);
}
function _setOwner(address newOwner) private {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 5 的 5:ReentrancyGuard.sol
pragma solidity ^0.8.0;
abstract contract ReentrancyGuard {
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
constructor() {
_status = _NOT_ENTERED;
}
modifier nonReentrant() {
require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
_status = _ENTERED;
_;
_status = _NOT_ENTERED;
}
}
{
"compilationTarget": {
"contracts/ArtistProxy.sol": "ArtistProxy"
},
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_braindropsAddress","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"inputs":[],"name":"braindrops","outputs":[{"internalType":"contract IBrainDrops","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"_projectId","type":"uint256"}],"name":"mint","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"_projectId","type":"uint256"}],"name":"mintForArtistsOnly","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"uint256","name":"_project1TokenId","type":"uint256"},{"internalType":"uint256","name":"_project2TokenId","type":"uint256"},{"internalType":"uint256","name":"_project3TokenId","type":"uint256"}],"name":"mintForGenesisDropHoldersOnly","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"_projectId","type":"uint256"}],"name":"mintForHoldersOnly","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"mintForProjectBurnersOnly","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"uint256","name":"_projectTokenId","type":"uint256"}],"name":"mintForProjectSpecificHoldersOnly","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"bytes32","name":"message","type":"bytes32"},{"internalType":"bytes","name":"signature","type":"bytes"}],"name":"mintForServerSigned","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"projectIdToArtistAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"projectIdToBotPreventionAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"projectIdToBurnableProjectId","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"projectIdToGenesisDropActivated","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"","type":"address"}],"name":"projectIdToGenesisDropAddressMinted","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"projectIdToHolderActivated","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"projectIdToOlderProjectId","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"projectIdToProjectActivated","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"projectIdToProjectAntiBotActivated","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"","type":"address"}],"name":"projectIdToProjectAntiBotAddressMinted","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"projectId","type":"uint256"},{"internalType":"address","name":"artistAddress","type":"address"}],"name":"setArtist","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"uint256","name":"_olderProjectId","type":"uint256"}],"name":"setProjectIdToBurnableProjectId","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"uint256","name":"_olderProjectId","type":"uint256"}],"name":"setProjectIdToOlderProjectId","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"string","name":"provenanceHash","type":"string"}],"name":"setProvenanceHash","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_signingAddress","type":"address"}],"name":"setSigningAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"signingAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"}],"name":"toggleProjectAntiBotActive","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"}],"name":"toggleProjectIsActive","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"}],"name":"toggleProjectIsGenesisDropActive","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"}],"name":"toggleProjectIsHolderActive","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"}],"name":"toggleProjectIsPaused","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"string","name":"_projectArtistName","type":"string"}],"name":"updateProjectArtistName","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"string","name":"_projectBaseURI","type":"string"}],"name":"updateProjectBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"string","name":"_projectDescription","type":"string"}],"name":"updateProjectDescription","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"string","name":"_projectLicense","type":"string"}],"name":"updateProjectLicense","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"uint256","name":"_pricePerTokenInWei","type":"uint256"}],"name":"updateProjectPricePerTokenInWei","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_projectId","type":"uint256"},{"internalType":"string","name":"_projectWebsite","type":"string"}],"name":"updateProjectWebsite","outputs":[],"stateMutability":"nonpayable","type":"function"}]