Ethereum Basics - Account/EVM

블록체인과 이더리움을 이해하기 위한 기초 개념인 계정과 EVM에 대해 알아봅니다.

Account란?

Account란 이더리움 네트워크에서 유일한 구분자로 이더리움 내에서 트랜잭션을 보낼 수 있는 Entity 입니다. Account는 EOA와 CA로 나눌 수 있으며 모든 Account는 Ethereum의 World State라는 Database에 State로 저장됩니다. Address와 World State에 저장되어 있는 Account State는 매핑되어 있어 Address를 이용하여 Account의 State를 확인할 수 있습니다.

World State는 노드마다 저장하고 있는 분산된 Ethereum Database로 가장 최신의 Account State(모든 계정의 현재 잔액, Storage State 등)을 저장하고 있습니다. 트랜잭션이 발생하면 Account의 data에 변경이 발생하므로 World State는 각 블록이 생성될 때 마다 업데이트 되어 최신화 됩니다.

  • EOA란?

Externally Owned Account의 약자로 비대칭 암호화 키 쌍으로 구성된 Account 입니다. 개인키를 이용하여 제어할 수 있으며 다른 EOA 혹은 Contract에 ETH를 전송하거나 상호작용하여 Method를 호출하는데 이용됩니다.

  • CA란?

Contract Account의 약자로 이더리움의 Account 중 하나인 Contract Account를 의미합니다. 보통 Contract Address의 의미로도 사용됩니다. 스마트 컨트랙트를 배포하면 자동으로 생성되며 트랜잭션 수신에 대한 응답으로만 트랜잭션을 실행할 수 있습니다.


Account의 구조

Account는 Address와 Mapping이 되어있으며 4개의 Field를 가지고 있습니다. 각 Field에 대한 설명은 다음과 같습니다.

  • Nonce : Ethereum에서 트랜잭션을 처리하기 위한 값으로 트랜잭션 Counter 역할을 합니다. 기본값은 0이며 트랜잭션을 실행할 때 마다 1씩 증가합니다.
  • Balance : Account가 소유하고 있는 ETH의 수량으로 단위는 Wei 입니다.
  • Storage hash : Contract의 상태 데이터를 저장한 머클 루트 입니다. CA에만 존재하는 값 입니다.
  • Code hash : EVM에서 실행될 Contract(EVM Code)의 Bytecode를 해시한 값 입니다. CA에만 존재합니다.

EOA의 경우, Contract가 아니기 때문에 Storage hash와 Code hash의 값이 빈 값으로 설정되어 있으나 CA에는 Storage hash와 Code hash의 값이 존재합니다. Account Storage는 계정의 상태 데이터를 해시하여 이를 머클 루트로 스마트 컨트랙트의 실행 결과에 따라 값이 변경됩니다. Code hash의 경우, 유저가 배포하는 컨트랙트를 컴파일한 Bytecode 값을 hash한 데이터가 저장됩니다.

EVM

EVM은 이더리움 가상 머신(Ethereum Virtual Machine)의 약자로 이더리움에서 스마트 컨트랙트를 실행하는데 사용되는 가상머신 입니다. 이더리움 노드마다 보유하고 있으며 EVM을 통해 모든 노드가 동일한 환경에서 계산을 수행하여 신뢰성을 보장합니다.

트랜잭션이 실행되면 EVM은 해당 트랜잭션에 대해 서명, gasPrice, gasLimit을 확인하여 유효성을 검사합니다. 이후 트랜잭션 실행에 필요한 모든 작업에 대한 Gas 비용을 계산하고, 해당 트랜잭션이 호출하는 스마트 컨트랙트를 불러옵니다. 불러온 스마트 컨트랙트의 코드를 실행한 후, 결과를 반환하여 이를 적용하여 상태(State)가 변경됩니다.

EVM의 구조는 다음과 같습니다.

  • Virtual ROM : EVM Code를 불러오는 가상의 ROM 입니다.
  • Program Counter : 실행 중인 명령어의 위치를 추적하기 위해 사용되는 Counter 입니다.
  • Gas Available : 스마트 컨트랙트의 작업이 실행될 떄마다 소비되는 Gas 비용을 추적하여 남은 Gas 양을 나타냅니다. 작업 중에는 항상 Gas Available의 값이 갱신되며 이 값이 0이 되면 작업이 중지됩니다. 트랜잭션의 gas 한도는 Raw Data에 작성된 gasLimit을 바탕으로 계산됩니다.
  • Stack : 트랜잭션 처리 중 계산 결과를 저장하는데 사용되며 32Bytes words를 최대 1024개 저장할 수 있습니다.
  • Memory : 동적 할당 메모리로 프로그램 실행 중 생성된 데이터를 저장하는데 사용됩니다.
  • Account Storage : Key-Value 형태로 저장되며 256bits 로 load/store 합니다. 해당 Account의 데이터를 불러와 트랜잭션을 실행하고 결과에 따라 상태를 변경하여 최신화 합니다. Account의 Storage 데이터를 EVM에 적용하여 사용합니다.

이더리움의 동작을 이해하셨나요?

Block과 Transaction, Account의 개념을 이해하셨다면 이제 이더리움이 동작하는 프로세스를 간단하게 정리해 볼까요?

  1. Account가 트랜잭션을 생성하고 서명하여 노드에게 전달합니다. (Account를 가진 유저라면 누구든 트랜잭션을 생성할 수 있습니다.)

  1. 노드는 이러한 트랜잭션을 검증한 뒤 트랜잭션을 Transaction pool에 저장한 후, 자신이 원하는 트랜잭션을 선택하여 트랜잭션을 실행합니다. (보통 수수료가 높은 트랜잭션을 선택합니다. 블록에 담기는 트랜잭션의 gas 크기가 제한되어 있기 때문에 높은 수수료를 주는 트랜잭션을 선택해야 많은 수수료를 받을 수 있기 때문입니다. 현재 이더리움에서 제한하는 블록 당 최대 가스량은 30,000,000 입니다.) 트랜잭션에 대한 자세한 정보는 이 곳(Transaction Part Link)을 클릭하여 확인할 수 있습니다.

  1. 노드는 실행된 트랜잭션과 결과(Receipt), 상태 변화(State)를 반영한 뒤 이러한 데이터를 이용해 새로운 블록을 생성합니다.

  1. 노드는 자신이 생성한 새로운 블록을 주변 노드들에게 전파하고 새로운 블록을 전파받은 노드들은 블록을 검증합니다. 검증이 완료되면 새로운 블록을 이전 블록에 연결하고 다음 블록 생성을 위해 위의 과정을 반복합니다. 검증하는 과정은 새로운 블록을 전파받은 노드가 전파받은 블록에 담긴 트랜잭션을 다시 실행하게 되며 이 결과로 얻은 데이터가 전파받은 블록의 데이터와 일치한다면 검증이 성공한 것 입니다.

Web3 Data API를 활용한 Account 정보 조회

아래 Nodit API들을 활용하여 Account 정보를 조회할 수 있습니다. API 상세 페이지에서 API를 호출해보고 응답에 포함된 계정 관련 데이터를 바로 확인해볼 수 있습니다.