Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Siyoung] Vote Contract 작성 #4

Open
wants to merge 5 commits into
base: develop
Choose a base branch
from
Open

[Siyoung] Vote Contract 작성 #4

wants to merge 5 commits into from

Conversation

sapcy
Copy link

@sapcy sapcy commented Feb 14, 2022

No description provided.

sapcy and others added 3 commits February 13, 2022 01:48
1. deployer -> owner 용어 변경
2. require 소유권자 검증이 반복되어 함수변경자로 적용 (onlyOwner)
3. startTime, endTime 적용
  - 현재 시간이 start, end time 사이에 있을 때만 vote 함수 실행 가능 => voteOpen(함수변경자)
  - 그 외 함수는 투표 시작시간 전에 배포해놓고 투표자를 세팅할 수도 있으므로 검증 제외
4. 임시 BadgemealNFT 컨트랙트를 통해 voters 매핑에 추가 전 NFT 홀더인지 검증
5. Vote Contract를 한번 배포해서 계속 쓰는 것이 아니라 투표 때마다 새로운 Vote Contract를 배포해서 사용 (투표 히스토리 목적)
6. 투표 종료 시간 체크는 백엔드에서 스케쥴러로 체크. 백엔드에서 winnerProposal 확인 후 BadgemealNFT 컨트랙트에 addWinnerProposals 함수로 메뉴 추가
- 그 외 함수는 투표 시작시간 전에 배포해놓고 투표자를 세팅할 수도 있으므로 시간 검증 제외
- 임시 BadgemealNFT 컨트랙트를 통해 voters 매핑에 추가 전 NFT 홀더인지 검증
- Vote Contract를 한번 배포해서 계속 쓰는 것이 아니라 투표 때마다 새로운 Vote Contract를 배포해서 사용 (투표 히스토리 관리, 투명성)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

투표 히스토리를 투명하게 관리하는 목적은 좋습니다. 다만 매번 새로운 컨트랙트를 배포하기 보다는 어짜피 컨트랙트 내에서 발생되는 트랜잭션으로 투명성을 확보할 수 있기 때문에 , 굳이 매번 새로 배포할 필요는 없을 것 같습니다!

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

투표마다 새로운 컨트랙트 배포에서 한번 배포 후 재사용 방식으로 변경하였습니다.

struct Proposal {
string name; // 메뉴 이름
string imageUrl; // 메뉴 이미지 url
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Proposal 에 imageUrl은 필요 없을 것 같습니다. 회의를 진행한 내용에 따르면 Proposal의 name을 가지고 기존 이미지에 텍스트만 교체하여 NFT를 발행하기로 했기 때문입니다. 따라서 채택된 메뉴가 DB에 저장될때 해당 메뉴에 부합하는 이미지를 자체적으로 매핑해서 관리하면 될 것 같습니다.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

imageUrl 제거하였습니다~

msg.sender == owner,
"msg sender is not a owner."
);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Owner를 체크하는 함수는 Ownable 컨트랙트를 상속받아서 사용하면 편하게 사용할 수 있습니다!
https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/access/Ownable.sol

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ownable 상속 방식으로 변경하였습니다.

function addNFTHolders(address[] memory _addresses) public onlyOwner {
for (uint i = 0; i < _addresses.length; i++) {
addNFTHolder(_addresses[i]);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

추측하기로는 parameter인 _addresses가 전체 뱃지밀 NFT 홀더의 address 리스트인 것 같은데요. 만약 홀더가 100명, 1000명 이상 되면 상당한 가스비를 소모할 것으로 예상됩니다. 최대한 실행이 많이 반복되는 코드는 줄이는 것이 좋을 것 같아요!

require(
BadgemealNFT(nftAddress).isHolder(_address),
concat(toString(_address), " is not NFT Holder.")
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

어떤 address가 인자로 들어왔는지 에러메세지에 함께 보여주는 부분이 좋은 것 같아요! 👍

// address -> string 변환 함수
function toString(address account) internal pure returns(string memory) {
return toString(abi.encodePacked(account));
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

유용한 util 함수네요👍

// string concat 함수
function concat(string memory a, string memory b) internal pure returns (string memory) {
return string(abi.encodePacked(a, b));
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

유용한 util 함수네요👍

uint startTime; // 투표 시작 시간

uint endTime; // 투표 마감 시간
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

startTime, endTime 을 세팅하는 아이디어 좋습니다 👍

owner = msg.sender;
startTime = _startTime;
endTime = _endTime;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

생성자에서 startTime, endTime 을 세팅하기보다 백엔드 스케줄러를 통해서 시작 전날 세팅하는 함수를 만드는 것도 좋을 것 같습니다!

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

투표 컨트랙트 배포해서 하나로 계속 쓰는 방식이면 수빈님 말대로 하는게 맞는 것 같습니다.

require(
now >= startTime && now < endTime,
"voting currently not open"
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

좋은 modifier 네요 👍

Copy link
Contributor

@Hyunki6040 Hyunki6040 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

투표의 startTime과 endTime 그리고 다양한 util를 구현해 활용하신 부분이 유용해요! 고생하셨습니다.

- desc: 투표자로 추가하려한 대상 address를 error message에 출력함으로써 어떤 address가 문제인지 메시지를 통해 확인 가능
```sol
function addVoter(address _address, address nftAddress) private {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

투표할 사람들을 미리 addVoter에 추가하고 투표한 사람들마다 처리하도록 같은데 맞나요?
매번 마스터 NFT가 민팅될때마다 addVoter를 호출해야해서 비효율적일 수 있을 것 같아요.
혹은 투표를 시간한 시점 이후에 마스터 NFT를 받은 사람은 투표자 명단에서 제외 될 수도 있고요.
투표할 때마다 NFT 소유자인지 아닌지는 vote function에서 확인하고 투표를 한 사람만 addVoter로 추가하면 좋을 것같아요

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

미리 voters에 추가하는 것은 아닙니다. 매번 투표자가 투표할 때마다 트랜잭션을 발생시키는 것보다 클라이언트단에서 투표 결과를 투표 종료전까지 DB로 저장한 다음 투표가 종료되면 한번의 트랜잭션으로 투표한 사람들의 address를 리스트로 넘겨 voters 매핑에 추가하고 투표결과를 vote함수로 반영하는 것으로 생각했습니다.
현기님 말씀대로 투표자가 투표할 때마다 검증하는 방식으로 한다면 마스터 NFT를 받으면서 생기는 투표자 유효여부에 따른 이슈를 줄일 수 있을 것 같네요.


struct Voter {
bool exist; // 투표자 존재 여부 (true,false)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

투표자 존재 여부의 용도가 뭔가요??

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

voters mapping에서 투표자 address가 매핑에 등록된 것인지 확인하는 과정에서 voters[address]가 null값이어도 리턴으로 Voter 구조체가 초기화된 값으로 나오길래 존재여부 변수(exist)를 추가하여 조건문에서 사용했습니다.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

exist변수 삭제 후 로직 변경하였습니다.

@Yunseongpyo
Copy link

고생하셨습니다👍

sapcy and others added 2 commits February 16, 2022 21:53
1. 투표 때마다 vote 컨트랙트 새로 배포 -> vote 컨트랙트 한번 배포 후 재사용
2. addVoters, addVoter 함수 제거
3. Ownable 컨트랙트 상속으로 컨트랙트 최적화
4. imageUrl 제거
5. vote함수에 조건 추가 (msg.sender가 NFT홀더인 경우)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants