Ejercicio 2
Control de acceso RegisterAccess.sol
This commit is contained in:
Родитель
728f7ece0c
Коммит
809caca6e9
|
@ -0,0 +1,26 @@
|
|||
REMIX DEFAULT WORKSPACE
|
||||
|
||||
Remix default workspace is present when:
|
||||
i. Remix loads for the very first time
|
||||
ii. A new workspace is created
|
||||
iii. There are no files existing in the File Explorer
|
||||
|
||||
This workspace contains 3 directories:
|
||||
|
||||
1. 'contracts': Holds three contracts with different complexity level, denoted with number prefix in file name.
|
||||
2. 'scripts': Holds two scripts to deploy a contract. It is explained below.
|
||||
3. 'tests': Contains one Solidity test file for 'Ballot' contract & one JS test file for 'Storage' contract
|
||||
|
||||
SCRIPTS
|
||||
|
||||
The 'scripts' folder contains two example async/await scripts for deploying the 'Storage' contract.
|
||||
For the deployment of any other contract, 'contractName' and 'constructorArgs' should be updated (along with other code if required).
|
||||
|
||||
Also, there is a script containing some unit tests for Storage contract inside tests directory.
|
||||
|
||||
To run a script, right click on file name in the file explorer and click 'Run'. Remember, Solidity file must already be compiled.
|
||||
Output from script will appear in remix terminal.
|
||||
|
||||
Please note, 'require' statement is supported in a limited manner for Remix supported modules.
|
||||
For now, modules supported by Remix are ethers, web3, swarmgw, chai, remix and hardhat only for hardhat.ethers object/plugin.
|
||||
For unsupported modules, an error like this will be thrown: '<module_name> module require is not supported by Remix IDE will be shown.'
|
|
@ -0,0 +1,103 @@
|
|||
// SPDX-License-Identifier: MIT
|
||||
|
||||
pragma solidity 0.8.13;
|
||||
|
||||
// Array
|
||||
|
||||
|
||||
|
||||
contract RegisterAccess {
|
||||
|
||||
string[] private info;
|
||||
|
||||
address public owner;
|
||||
|
||||
mapping (address => bool) public whiteList;
|
||||
|
||||
|
||||
|
||||
constructor() {
|
||||
|
||||
owner = msg.sender;
|
||||
|
||||
whiteList[msg.sender] = true;
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
event InfoChange(string oldInfo, string newInfo);
|
||||
|
||||
|
||||
|
||||
modifier onlyOwner {
|
||||
|
||||
require(msg.sender == owner,"Only owner");
|
||||
|
||||
_;
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
modifier onlyWhitelist {
|
||||
|
||||
require(whiteList[msg.sender] == true, "Only whitelist");
|
||||
|
||||
_;
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
function getInfo(uint index) public view returns (string memory) {
|
||||
|
||||
return info[index];
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
function setInfo(uint index, string memory _info) public onlyWhitelist {
|
||||
|
||||
emit InfoChange (info[index], _info);
|
||||
|
||||
info[index] = _info;
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
function addInfo(string memory _info) public onlyWhitelist returns (uint index) {
|
||||
|
||||
info.push (_info);
|
||||
|
||||
index = info.length -1;
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
function listInfo() public view returns (string[] memory) {
|
||||
|
||||
return info;
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
function addMember (address _member) public onlyOwner {
|
||||
|
||||
whiteList[_member] = true;
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
function delMember (address _member) public onlyOwner {
|
||||
|
||||
whiteList[_member] = false;
|
||||
|
||||
}
|
||||
|
||||
}
|
Различия файлов скрыты, потому что одна или несколько строк слишком длинны
|
@ -0,0 +1,197 @@
|
|||
{
|
||||
"compiler": {
|
||||
"version": "0.8.13+commit.abaa5c0e"
|
||||
},
|
||||
"language": "Solidity",
|
||||
"output": {
|
||||
"abi": [
|
||||
{
|
||||
"inputs": [],
|
||||
"stateMutability": "nonpayable",
|
||||
"type": "constructor"
|
||||
},
|
||||
{
|
||||
"anonymous": false,
|
||||
"inputs": [
|
||||
{
|
||||
"indexed": false,
|
||||
"internalType": "string",
|
||||
"name": "oldInfo",
|
||||
"type": "string"
|
||||
},
|
||||
{
|
||||
"indexed": false,
|
||||
"internalType": "string",
|
||||
"name": "newInfo",
|
||||
"type": "string"
|
||||
}
|
||||
],
|
||||
"name": "InfoChange",
|
||||
"type": "event"
|
||||
},
|
||||
{
|
||||
"inputs": [
|
||||
{
|
||||
"internalType": "string",
|
||||
"name": "_info",
|
||||
"type": "string"
|
||||
}
|
||||
],
|
||||
"name": "addInfo",
|
||||
"outputs": [
|
||||
{
|
||||
"internalType": "uint256",
|
||||
"name": "index",
|
||||
"type": "uint256"
|
||||
}
|
||||
],
|
||||
"stateMutability": "nonpayable",
|
||||
"type": "function"
|
||||
},
|
||||
{
|
||||
"inputs": [
|
||||
{
|
||||
"internalType": "address",
|
||||
"name": "_member",
|
||||
"type": "address"
|
||||
}
|
||||
],
|
||||
"name": "addMember",
|
||||
"outputs": [],
|
||||
"stateMutability": "nonpayable",
|
||||
"type": "function"
|
||||
},
|
||||
{
|
||||
"inputs": [
|
||||
{
|
||||
"internalType": "address",
|
||||
"name": "_member",
|
||||
"type": "address"
|
||||
}
|
||||
],
|
||||
"name": "delMember",
|
||||
"outputs": [],
|
||||
"stateMutability": "nonpayable",
|
||||
"type": "function"
|
||||
},
|
||||
{
|
||||
"inputs": [
|
||||
{
|
||||
"internalType": "uint256",
|
||||
"name": "index",
|
||||
"type": "uint256"
|
||||
}
|
||||
],
|
||||
"name": "getInfo",
|
||||
"outputs": [
|
||||
{
|
||||
"internalType": "string",
|
||||
"name": "",
|
||||
"type": "string"
|
||||
}
|
||||
],
|
||||
"stateMutability": "view",
|
||||
"type": "function"
|
||||
},
|
||||
{
|
||||
"inputs": [],
|
||||
"name": "listInfo",
|
||||
"outputs": [
|
||||
{
|
||||
"internalType": "string[]",
|
||||
"name": "",
|
||||
"type": "string[]"
|
||||
}
|
||||
],
|
||||
"stateMutability": "view",
|
||||
"type": "function"
|
||||
},
|
||||
{
|
||||
"inputs": [],
|
||||
"name": "owner",
|
||||
"outputs": [
|
||||
{
|
||||
"internalType": "address",
|
||||
"name": "",
|
||||
"type": "address"
|
||||
}
|
||||
],
|
||||
"stateMutability": "view",
|
||||
"type": "function"
|
||||
},
|
||||
{
|
||||
"inputs": [
|
||||
{
|
||||
"internalType": "uint256",
|
||||
"name": "index",
|
||||
"type": "uint256"
|
||||
},
|
||||
{
|
||||
"internalType": "string",
|
||||
"name": "_info",
|
||||
"type": "string"
|
||||
}
|
||||
],
|
||||
"name": "setInfo",
|
||||
"outputs": [],
|
||||
"stateMutability": "nonpayable",
|
||||
"type": "function"
|
||||
},
|
||||
{
|
||||
"inputs": [
|
||||
{
|
||||
"internalType": "address",
|
||||
"name": "",
|
||||
"type": "address"
|
||||
}
|
||||
],
|
||||
"name": "whiteList",
|
||||
"outputs": [
|
||||
{
|
||||
"internalType": "bool",
|
||||
"name": "",
|
||||
"type": "bool"
|
||||
}
|
||||
],
|
||||
"stateMutability": "view",
|
||||
"type": "function"
|
||||
}
|
||||
],
|
||||
"devdoc": {
|
||||
"kind": "dev",
|
||||
"methods": {},
|
||||
"version": 1
|
||||
},
|
||||
"userdoc": {
|
||||
"kind": "user",
|
||||
"methods": {},
|
||||
"version": 1
|
||||
}
|
||||
},
|
||||
"settings": {
|
||||
"compilationTarget": {
|
||||
"RegisterAccess.sol": "RegisterAccess"
|
||||
},
|
||||
"evmVersion": "london",
|
||||
"libraries": {},
|
||||
"metadata": {
|
||||
"bytecodeHash": "ipfs"
|
||||
},
|
||||
"optimizer": {
|
||||
"enabled": false,
|
||||
"runs": 200
|
||||
},
|
||||
"remappings": []
|
||||
},
|
||||
"sources": {
|
||||
"RegisterAccess.sol": {
|
||||
"keccak256": "0xe0beb35f0b5e533f0e945db5718986e1cc8362938581d0c9958a3b63a6aafedc",
|
||||
"license": "MIT",
|
||||
"urls": [
|
||||
"bzz-raw://1ba84f3051eec7693297666421211e4aeb0923c1a72b53e4377c395c021cdacb",
|
||||
"dweb:/ipfs/QmaPLbWMhXgjAWaAXjTiAVhQACJLc8SeRaJZvYta9wFtW2"
|
||||
]
|
||||
}
|
||||
},
|
||||
"version": 1
|
||||
}
|
Различия файлов скрыты, потому что одна или несколько строк слишком длинны
|
@ -0,0 +1,83 @@
|
|||
{
|
||||
"id": "bb422ee4f05e281d5e9ffcc3067e4ce8",
|
||||
"_format": "hh-sol-build-info-1",
|
||||
"solcVersion": "0.8.13",
|
||||
"solcLongVersion": "0.8.13+commit.abaa5c0e",
|
||||
"input": {
|
||||
"language": "Solidity",
|
||||
"sources": {
|
||||
"RegisterAccess.sol": {
|
||||
"content": ""
|
||||
}
|
||||
},
|
||||
"settings": {
|
||||
"optimizer": {
|
||||
"enabled": false,
|
||||
"runs": 200
|
||||
},
|
||||
"outputSelection": {
|
||||
"*": {
|
||||
"": [
|
||||
"ast"
|
||||
],
|
||||
"*": [
|
||||
"abi",
|
||||
"metadata",
|
||||
"devdoc",
|
||||
"userdoc",
|
||||
"storageLayout",
|
||||
"evm.legacyAssembly",
|
||||
"evm.bytecode",
|
||||
"evm.deployedBytecode",
|
||||
"evm.methodIdentifiers",
|
||||
"evm.gasEstimates",
|
||||
"evm.assembly"
|
||||
]
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"output": {
|
||||
"errors": [
|
||||
{
|
||||
"component": "general",
|
||||
"errorCode": "1878",
|
||||
"formattedMessage": "Warning: SPDX license identifier not provided in source file. Before publishing, consider adding a comment containing \"SPDX-License-Identifier: <SPDX-License>\" to each source file. Use \"SPDX-License-Identifier: UNLICENSED\" for non-open-source code. Please see https://spdx.org for more information.\n--> RegisterAccess.sol\n\n",
|
||||
"message": "SPDX license identifier not provided in source file. Before publishing, consider adding a comment containing \"SPDX-License-Identifier: <SPDX-License>\" to each source file. Use \"SPDX-License-Identifier: UNLICENSED\" for non-open-source code. Please see https://spdx.org for more information.",
|
||||
"severity": "warning",
|
||||
"sourceLocation": {
|
||||
"end": -1,
|
||||
"file": "RegisterAccess.sol",
|
||||
"start": -1
|
||||
},
|
||||
"type": "Warning"
|
||||
},
|
||||
{
|
||||
"component": "general",
|
||||
"errorCode": "3420",
|
||||
"formattedMessage": "Warning: Source file does not specify required compiler version! Consider adding \"pragma solidity ^0.8.13;\"\n--> RegisterAccess.sol\n\n",
|
||||
"message": "Source file does not specify required compiler version! Consider adding \"pragma solidity ^0.8.13;\"",
|
||||
"severity": "warning",
|
||||
"sourceLocation": {
|
||||
"end": -1,
|
||||
"file": "RegisterAccess.sol",
|
||||
"start": -1
|
||||
},
|
||||
"type": "Warning"
|
||||
}
|
||||
],
|
||||
"sources": {
|
||||
"RegisterAccess.sol": {
|
||||
"ast": {
|
||||
"absolutePath": "RegisterAccess.sol",
|
||||
"exportedSymbols": {},
|
||||
"id": 1,
|
||||
"nodeType": "SourceUnit",
|
||||
"nodes": [],
|
||||
"src": "0:0:0"
|
||||
},
|
||||
"id": 0
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,29 @@
|
|||
// SPDX-License-Identifier: GPL-3.0
|
||||
|
||||
pragma solidity >=0.7.0 <0.9.0;
|
||||
|
||||
/**
|
||||
* @title Storage
|
||||
* @dev Store & retrieve value in a variable
|
||||
* @custom:dev-run-script ./scripts/deploy_with_ethers.ts
|
||||
*/
|
||||
contract Storage {
|
||||
|
||||
uint256 number;
|
||||
|
||||
/**
|
||||
* @dev Store value in variable
|
||||
* @param num value to store
|
||||
*/
|
||||
function store(uint256 num) public {
|
||||
number = num;
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Return value
|
||||
* @return value of 'number'
|
||||
*/
|
||||
function retrieve() public view returns (uint256){
|
||||
return number;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,54 @@
|
|||
// SPDX-License-Identifier: GPL-3.0
|
||||
|
||||
pragma solidity >=0.7.0 <0.9.0;
|
||||
|
||||
import "hardhat/console.sol";
|
||||
|
||||
/**
|
||||
* @title Owner
|
||||
* @dev Set & change owner
|
||||
*/
|
||||
contract Owner {
|
||||
|
||||
address private owner;
|
||||
|
||||
// event for EVM logging
|
||||
event OwnerSet(address indexed oldOwner, address indexed newOwner);
|
||||
|
||||
// modifier to check if caller is owner
|
||||
modifier isOwner() {
|
||||
// If the first argument of 'require' evaluates to 'false', execution terminates and all
|
||||
// changes to the state and to Ether balances are reverted.
|
||||
// This used to consume all gas in old EVM versions, but not anymore.
|
||||
// It is often a good idea to use 'require' to check if functions are called correctly.
|
||||
// As a second argument, you can also provide an explanation about what went wrong.
|
||||
require(msg.sender == owner, "Caller is not owner");
|
||||
_;
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Set contract deployer as owner
|
||||
*/
|
||||
constructor() {
|
||||
console.log("Owner contract deployed by:", msg.sender);
|
||||
owner = msg.sender; // 'msg.sender' is sender of current call, contract deployer for a constructor
|
||||
emit OwnerSet(address(0), owner);
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Change owner
|
||||
* @param newOwner address of new owner
|
||||
*/
|
||||
function changeOwner(address newOwner) public isOwner {
|
||||
emit OwnerSet(owner, newOwner);
|
||||
owner = newOwner;
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Return owner address
|
||||
* @return address of owner
|
||||
*/
|
||||
function getOwner() external view returns (address) {
|
||||
return owner;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,138 @@
|
|||
// SPDX-License-Identifier: GPL-3.0
|
||||
|
||||
pragma solidity >=0.7.0 <0.9.0;
|
||||
|
||||
/**
|
||||
* @title Ballot
|
||||
* @dev Implements voting process along with vote delegation
|
||||
*/
|
||||
contract Ballot {
|
||||
|
||||
struct Voter {
|
||||
uint weight; // weight is accumulated by delegation
|
||||
bool voted; // if true, that person already voted
|
||||
address delegate; // person delegated to
|
||||
uint vote; // index of the voted proposal
|
||||
}
|
||||
|
||||
struct Proposal {
|
||||
// If you can limit the length to a certain number of bytes,
|
||||
// always use one of bytes1 to bytes32 because they are much cheaper
|
||||
bytes32 name; // short name (up to 32 bytes)
|
||||
uint voteCount; // number of accumulated votes
|
||||
}
|
||||
|
||||
address public chairperson;
|
||||
|
||||
mapping(address => Voter) public voters;
|
||||
|
||||
Proposal[] public proposals;
|
||||
|
||||
/**
|
||||
* @dev Create a new ballot to choose one of 'proposalNames'.
|
||||
* @param proposalNames names of proposals
|
||||
*/
|
||||
constructor(bytes32[] memory proposalNames) {
|
||||
chairperson = msg.sender;
|
||||
voters[chairperson].weight = 1;
|
||||
|
||||
for (uint i = 0; i < proposalNames.length; i++) {
|
||||
// 'Proposal({...})' creates a temporary
|
||||
// Proposal object and 'proposals.push(...)'
|
||||
// appends it to the end of 'proposals'.
|
||||
proposals.push(Proposal({
|
||||
name: proposalNames[i],
|
||||
voteCount: 0
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Give 'voter' the right to vote on this ballot. May only be called by 'chairperson'.
|
||||
* @param voter address of voter
|
||||
*/
|
||||
function giveRightToVote(address voter) public {
|
||||
require(
|
||||
msg.sender == chairperson,
|
||||
"Only chairperson can give right to vote."
|
||||
);
|
||||
require(
|
||||
!voters[voter].voted,
|
||||
"The voter already voted."
|
||||
);
|
||||
require(voters[voter].weight == 0);
|
||||
voters[voter].weight = 1;
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Delegate your vote to the voter 'to'.
|
||||
* @param to address to which vote is delegated
|
||||
*/
|
||||
function delegate(address to) public {
|
||||
Voter storage sender = voters[msg.sender];
|
||||
require(!sender.voted, "You already voted.");
|
||||
require(to != msg.sender, "Self-delegation is disallowed.");
|
||||
|
||||
while (voters[to].delegate != address(0)) {
|
||||
to = voters[to].delegate;
|
||||
|
||||
// We found a loop in the delegation, not allowed.
|
||||
require(to != msg.sender, "Found loop in delegation.");
|
||||
}
|
||||
sender.voted = true;
|
||||
sender.delegate = to;
|
||||
Voter storage delegate_ = voters[to];
|
||||
if (delegate_.voted) {
|
||||
// If the delegate already voted,
|
||||
// directly add to the number of votes
|
||||
proposals[delegate_.vote].voteCount += sender.weight;
|
||||
} else {
|
||||
// If the delegate did not vote yet,
|
||||
// add to her weight.
|
||||
delegate_.weight += sender.weight;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Give your vote (including votes delegated to you) to proposal 'proposals[proposal].name'.
|
||||
* @param proposal index of proposal in the proposals array
|
||||
*/
|
||||
function vote(uint proposal) public {
|
||||
Voter storage sender = voters[msg.sender];
|
||||
require(sender.weight != 0, "Has no right to vote");
|
||||
require(!sender.voted, "Already voted.");
|
||||
sender.voted = true;
|
||||
sender.vote = proposal;
|
||||
|
||||
// If 'proposal' is out of the range of the array,
|
||||
// this will throw automatically and revert all
|
||||
// changes.
|
||||
proposals[proposal].voteCount += sender.weight;
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Computes the winning proposal taking all previous votes into account.
|
||||
* @return winningProposal_ index of winning proposal in the proposals array
|
||||
*/
|
||||
function winningProposal() public view
|
||||
returns (uint winningProposal_)
|
||||
{
|
||||
uint winningVoteCount = 0;
|
||||
for (uint p = 0; p < proposals.length; p++) {
|
||||
if (proposals[p].voteCount > winningVoteCount) {
|
||||
winningVoteCount = proposals[p].voteCount;
|
||||
winningProposal_ = p;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Calls winningProposal() function to get the index of the winner contained in the proposals array and then
|
||||
* @return winnerName_ the name of the winner
|
||||
*/
|
||||
function winnerName() public view
|
||||
returns (bytes32 winnerName_)
|
||||
{
|
||||
winnerName_ = proposals[winningProposal()].name;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,16 @@
|
|||
|
||||
// This script can be used to deploy the "Storage" contract using ethers.js library.
|
||||
// Please make sure to compile "./contracts/1_Storage.sol" file before running this script.
|
||||
// And use Right click -> "Run" from context menu of the file to run the script. Shortcut: Ctrl+Shift+S
|
||||
|
||||
import { deploy } from './ethers.ts'
|
||||
|
||||
(async () => {
|
||||
try {
|
||||
const result = await deploy('Storage', [])
|
||||
console.log(JSON.stringify(result, null, ' '))
|
||||
console.log(`address: ${result.address}`)
|
||||
} catch (e) {
|
||||
console.log(e.message)
|
||||
}
|
||||
})()
|
|
@ -0,0 +1,16 @@
|
|||
|
||||
// This script can be used to deploy the "Storage" contract using Web3 library.
|
||||
// Please make sure to compile "./contracts/1_Storage.sol" file before running this script.
|
||||
// And use Right click -> "Run" from context menu of the file to run the script. Shortcut: Ctrl+Shift+S
|
||||
|
||||
import { deploy } from './web3.ts'
|
||||
|
||||
(async () => {
|
||||
try {
|
||||
const result = await deploy('Storage', [])
|
||||
console.log(JSON.stringify(result, null, ' ')))
|
||||
console.log(`address: ${result.address}`)
|
||||
} catch (e) {
|
||||
console.log(e.message)
|
||||
}
|
||||
})()
|
|
@ -0,0 +1,25 @@
|
|||
|
||||
export const deploy = async (contractName: string, arguments: Array<any>, from?: string) => {
|
||||
|
||||
console.log(`deploying ${contractName}`)
|
||||
// Note that the script needs the ABI which is generated from the compilation artifact.
|
||||
// Make sure contract is compiled and artifacts are generated
|
||||
const artifactsPath = `browser/contracts/artifacts/${contractName}.json` // Change this for different path
|
||||
|
||||
const metadata = JSON.parse(await remix.call('fileManager', 'getFile', artifactsPath))
|
||||
// 'web3Provider' is a remix global variable object
|
||||
const signer = (new ethers.providers.Web3Provider(web3Provider)).getSigner()
|
||||
|
||||
let factory = new ethers.ContractFactory(metadata.abi, metadata.data.bytecode.object, signer);
|
||||
|
||||
let contract
|
||||
if (from) {
|
||||
contract = await factory.connect(from).deploy(...arguments);
|
||||
} else {
|
||||
contract = await factory.deploy(...arguments);
|
||||
}
|
||||
|
||||
// The contract is NOT deployed yet; we must wait until it is mined
|
||||
await contract.deployed()
|
||||
return contract
|
||||
}: Promise<any>
|
|
@ -0,0 +1,25 @@
|
|||
|
||||
export const deploy = async (contractName: string, arguments: Array<any>, from?: string, gas?: number) => {
|
||||
|
||||
console.log(`deploying ${contractName}`)
|
||||
// Note that the script needs the ABI which is generated from the compilation artifact.
|
||||
// Make sure contract is compiled and artifacts are generated
|
||||
const artifactsPath = `browser/contracts/artifacts/${contractName}.json` // Change this for different path
|
||||
|
||||
const metadata = JSON.parse(await remix.call('fileManager', 'getFile', artifactsPath))
|
||||
|
||||
const accounts = await web3.eth.getAccounts()
|
||||
|
||||
let contract = new web3.eth.Contract(metadata.abi)
|
||||
|
||||
contract = contract.deploy({
|
||||
data: metadata.data.bytecode.object,
|
||||
arguments
|
||||
})
|
||||
|
||||
const newContractInstance = await contract.send({
|
||||
from: from || accounts[0],
|
||||
gas: gas || 1500000
|
||||
})
|
||||
return newContractInstance.options
|
||||
}: Promise<any>
|
|
@ -0,0 +1,28 @@
|
|||
// SPDX-License-Identifier: GPL-3.0
|
||||
|
||||
pragma solidity >=0.7.0 <0.9.0;
|
||||
import "remix_tests.sol"; // this import is automatically injected by Remix.
|
||||
import "hardhat/console.sol";
|
||||
import "../contracts/3_Ballot.sol";
|
||||
|
||||
contract BallotTest {
|
||||
|
||||
bytes32[] proposalNames;
|
||||
|
||||
Ballot ballotToTest;
|
||||
function beforeAll () public {
|
||||
proposalNames.push(bytes32("candidate1"));
|
||||
ballotToTest = new Ballot(proposalNames);
|
||||
}
|
||||
|
||||
function checkWinningProposal () public {
|
||||
console.log("Running checkWinningProposal");
|
||||
ballotToTest.vote(0);
|
||||
Assert.equal(ballotToTest.winningProposal(), uint(0), "proposal at index 0 should be the winning proposal");
|
||||
Assert.equal(ballotToTest.winnerName(), bytes32("candidate1"), "candidate1 should be the winner name");
|
||||
}
|
||||
|
||||
function checkWinninProposalWithReturnValue () public view returns (bool) {
|
||||
return ballotToTest.winningProposal() == 0;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,22 @@
|
|||
// Right click on the script name and hit "Run" to execute
|
||||
const { expect } = require("chai");
|
||||
const { ethers } = require("hardhat");
|
||||
|
||||
describe("Storage", function () {
|
||||
it("test initial value", async function () {
|
||||
const Storage = await ethers.getContractFactory("Storage");
|
||||
const storage = await Storage.deploy();
|
||||
await storage.deployed();
|
||||
console.log('storage deployed at:'+ storage.address)
|
||||
expect((await storage.retrieve()).toNumber()).to.equal(0);
|
||||
});
|
||||
it("test updating and retrieving updated value", async function () {
|
||||
const Storage = await ethers.getContractFactory("Storage");
|
||||
const storage = await Storage.deploy();
|
||||
await storage.deployed();
|
||||
const storage2 = await ethers.getContractAt("Storage", storage.address);
|
||||
const setValue = await storage2.store(56);
|
||||
await setValue.wait();
|
||||
expect((await storage2.retrieve()).toNumber()).to.equal(56);
|
||||
});
|
||||
});
|
Загрузка…
Ссылка в новой задаче