Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
スマートコントラクトプログラミング / Smart Contract Programming
Search
Kenji Saito
PRO
October 18, 2025
Technology
0
4
スマートコントラクトプログラミング / Smart Contract Programming
慶應義塾大学大学院メディアデザイン研究科「サービスデザインプロジェクトB」2025秋 第3回にて使用したスライドです。
Kenji Saito
PRO
October 18, 2025
Tweet
Share
More Decks by Kenji Saito
See All by Kenji Saito
AI が研究する時代に、人はどう育つのか? — GAMER PAT にみる "シリアスゲームとしての知的訓練" / In an era where AI conducts research, how will humans develop? — "Intellectual Training as a Serious Game" Seen in GAMER PAT
ks91
PRO
0
36
FinTech 5-6 : The World of Apps
ks91
PRO
0
67
生成AI による論文執筆サポート・ワークショップ ─ サーベイ/リサーチクエスチョン編 / Workshop on AI-Assisted Paper Writing Support: Survey/Research Question Edition
ks91
PRO
0
66
ブロックチェーン概論とインストール大会 / Introduction to Blockchain and Installation Workshop
ks91
PRO
0
2
FinTech 3-4 : Internet Technology and Governance
ks91
PRO
0
71
民主主義と博愛(Humanitarianism) / Democracy and Humanitarianism
ks91
PRO
0
6
ブロックチェーン概論 / Introduction to Blockchain
ks91
PRO
0
12
ブロックチェーンと分散ファイナンス概論 / Introduction to Blockchain and Decentralized Finance
ks91
PRO
0
65
Proof of Authenticity of General IoT Information with Tamper-Evident Sensors and Blockchain
ks91
PRO
0
7
Other Decks in Technology
See All in Technology
AWS IoT 超入門 2025
hattori
0
340
Findy Team+ QAチーム これからのチャレンジ!
findy_eventslides
0
180
HR Force における DWH の併用事例 ~ サービス基盤としての BigQuery / 分析基盤としての Snowflake ~@Cross Data Platforms Meetup #2「BigQueryと愉快な仲間たち」
ryo_suzuki
0
220
AI時代こそ求められる設計力- AWSクラウドデザインパターン3選で信頼性と拡張性を高める-
kenichirokimura
3
320
BI ツールはもういらない?Amazon RedShift & MCP Server で試みる新しいデータ分析アプローチ
cdataj
0
160
Vibe Coding Year in Review. From Karpathy to Real-World Agents by Niels Rolland, CEO Paatch
vcoisne
0
140
GoでもGUIアプリを作りたい!
kworkdev
PRO
0
140
Introduction to Bill One Development Engineer
sansan33
PRO
0
300
プロダクトのコードから見るGoによるデザインパターンの実践 #go_night_talk
bengo4com
1
2.5k
AWS Top Engineer、浮いてませんか? / As an AWS Top Engineer, Are You Out of Place?
yuj1osm
2
210
AWSでAgentic AIを開発するための前提知識の整理
nasuvitz
2
160
フレームワークを意識させないワークショップづくり
keigosuda
0
180
Featured
See All Featured
Learning to Love Humans: Emotional Interface Design
aarron
274
41k
How to train your dragon (web standard)
notwaldorf
97
6.3k
Leading Effective Engineering Teams in the AI Era
addyosmani
5
430
Building a Scalable Design System with Sketch
lauravandoore
463
33k
Fantastic passwords and where to find them - at NoRuKo
philnash
52
3.4k
10 Git Anti Patterns You Should be Aware of
lemiorhan
PRO
657
61k
Responsive Adventures: Dirty Tricks From The Dark Corners of Front-End
smashingmag
252
21k
Let's Do A Bunch of Simple Stuff to Make Websites Faster
chriscoyier
508
140k
[RailsConf 2023 Opening Keynote] The Magic of Rails
eileencodes
31
9.7k
CSS Pre-Processors: Stylus, Less & Sass
bermonpainter
359
30k
Raft: Consensus for Rubyists
vanstee
140
7.1k
Unsuck your backbone
ammeep
671
58k
Transcript
generated by Stable Image Ultra B 2025 3 ( )
B 2025 — 3 — 2025-10-20 – p.1/66
https://speakerdeck.com/ks91/collections/service-design-project-b-2025-fall ( ) ( ) Discord Discord ( / /
) Zoom URL ( ) B 2025 — 3 — 2025-10-20 – p.2/66
( ) 1 10 6 • 2 10 13 •
3 10 20 • 4 10 27 2 5 11 10 1 6 11 24 2 7 12 1 8 12 8 ( ) Discord B 2025 — 3 — 2025-10-20 – p.3/66
1. Ethereum ( ) 2. ( ) 3. Ethereum (1)
: ERC20 (2) : (3) : B 2025 — 3 — 2025-10-20 – p.4/66
1. Ethereum ( ) macOS (Tahoe 26.0.1) + Homebrew Linux
(Ubuntu 24.04) (x86/AMD64 and ARM) Windows 11 + Windows Subsystem for Linux (Ubuntu ) Ethereum solidity brownie Lima on macOS B 2025 — 3 — 2025-10-20 – p.5/66
macOS : Homebrew (Apple macOS ) UNIX macOS Lima Homebrew
brew Apple brew ( ) brew Rosetta brew brew https://github.com/homebrew/install Path : https://zenn.dev/tet0h/articles/a92651d52bd82460aefb $ which brew /opt/homebrew/bin/brew B 2025 — 3 — 2025-10-20 – p.6/66
macOS : Rosetta 2 Apple x86/AMD64 ( ) $ softwareupdate
--install-rosetta lima Rosetta 2 lima B 2025 — 3 — 2025-10-20 – p.7/66
macOS : Lima macOS Linux ( ) https://github.com/lima-vm/lima QEMU (
) Rosetta 2 Apple x86/AMD64 brew (lima-additional-guestagents ) $ brew install lima Ubuntu 24.04 (x86/AMD64) “kmd.yaml” $ limactl start kmd.yaml B 2025 — 3 — 2025-10-20 – p.8/66
macOS : Lima Linux (Ubuntu) x86_64 $ limactl list Linux
(exit ) $ limactl shell kmd Linux limactl list STATUS Stopped limactl start kmd “.lima” diffdisk 100GB B 2025 — 3 — 2025-10-20 – p.9/66
Solidity Ethereum ( ) http://solidity.readthedocs.io/en/latest/installing-solidity.html Linux (on Windows) “sudo apt”,
macOS “brew” Linux ARM git clone dependencies solc 0.8.30 (10/4 ) $ solc --version brownie py-solc-x solc (dependencies) ( solc py-solc-x ) B 2025 — 3 — 2025-10-20 – p.10/66
Ganache Ethereum RPC npm ( ) macOS https://qiita.com/kyosuke5_20/items/c5f68fc9d89b84c0df09 Linux https://qiita.com/seibe/items/36cef7df85fe2cefa3ea
npm Ganache $ sudo npm install -g ganache B 2025 — 3 — 2025-10-20 – p.11/66
Python3 macOS $ brew install python3 Python 3.8 Homebrew $
brew uninstall --ignore-dependencies
[email protected]
$ brew install
[email protected]
/usr/local/opt/
[email protected]
/bin/python3 Linux ( python3.12.3 OK) $ sudo apt update $ sudo apt install python3-dev python3-venv python3-tk 16.10 python3(.8) apt (PPA ) python3 python3-dev python3-venv python3-tk B 2025 — 3 — 2025-10-20 – p.12/66
macOS : . . . macOS wget brew dyld: Library
not loaded: /usr/local/opt/gettext/lib/libintl.8.dylib $ brew uninstall --force gettext $ brew install gettext B 2025 — 3 — 2025-10-20 – p.13/66
Python (venv) macOS (Linux python3 ) OK Python $ python3
-m venv bbc2env $ source bbc2env/bin/activate (bbc2env) $ pip install -U pip bbc2env ( BBc-2 ) ( ) (bbc2env) $ deactivate B 2025 — 3 — 2025-10-20 – p.14/66
Brownie Python Ethereum https://eth-brownie.readthedocs.io Brownie $ pip install eth-brownie version
1.21.0 (10/4 ) $ pip install setuptools setuptools B 2025 — 3 — 2025-10-20 – p.15/66
$ brownie init ( ERC-20 ) $ brownie bake token
$ cd token B 2025 — 3 — 2025-10-20 – p.16/66
contracts (macOS Linux (x86/AMD64) ) $ brownie compile Linux (ARM)
(AMD64 ) $ python # >>> import solcx >>> solcx.compile_solc(’0.6.12’) # >>> quit() /tmp ∼/.solcx/solc-v0.6.12 ( ) $ pytest tests B 2025 — 3 — 2025-10-20 – p.17/66
Brownie $ brownie console Python >>> len(accounts) 10 >>> accounts[0].balance()
100000000000000000000 >>> quit() B 2025 — 3 — 2025-10-20 – p.18/66
2. ( ) Ethereum Sepolia B 2025 — 3 —
2025-10-20 – p.19/66
infura.io Brownie https://infura.io SIGN UP PROJECT ID infura Ethereum API
infura . . . infura.io PROJECT ID WEB3_INFURA_PROJECT_ID PROJECT ID (∼/.bash_profile ) $ export WEB3_INFURA_PROJECT_ID= PROJECT ID B 2025 — 3 — 2025-10-20 – p.20/66
Sepolia ETH brownie console Sepolia >>> network.disconnect() >>> network.connect(’sepolia’) >>>
len(accounts) 0 >>> accounts.add() <LocalAccount object ’ 16 ’> >>> accounts[0].private_key 16 >>> quit() Sepolia Testnet ETH Faucet ETH https://sepoliafaucet.com ( ) ( ) . . . Ethereum 0.001 ETH 0.1 Sepolia ETH B 2025 — 3 — 2025-10-20 – p.21/66
Sepolia ETH brownie console Sepolia ETH >>> network.disconnect() >>> network.connect(’sepolia’)
>>> len(accounts) 0 >>> accounts.add( 16 ) <LocalAccount object ’ 16 ’> >>> accounts[0].balace() ETH >>> quit() ETH >>> accounts[0].transfer(accounts[1], "0.01 ether") B 2025 — 3 — 2025-10-20 – p.22/66
3. Ethereum B 2025 — 3 — 2025-10-20 – p.23/66
Ethereum Vitalik Buterin, “Ethereum White Paper: A NEXT GENERATION SMART
CONTRACT & DECENTRALIZED APPLICATION PLATFORM” 12 (= ) = ⇒ → Dapps ( / / ) B 2025 — 3 — 2025-10-20 – p.24/66
( ) B 2025 — 3 — 2025-10-20 – p.25/66
B 2025 — 3 — 2025-10-20 – p.26/66
= run B 2025 — 3 — 2025-10-20 – p.27/66
Ether Ethereum EOA : Externally-Owned Account ( ) Ether EVM
EVM B 2025 — 3 — 2025-10-20 – p.28/66
EVM : Ethereum Virtual Machine ঢ়ଶ ֎෦ΞΫλʔ ΞΧϯτ &7. ίʔυ
&UIFSˠ(BT ىಈ ίϯτϥΫτ ϝοηʔδ͔ɺ৽͍ࣗ͠ಈΦϒδΣΫτ ΞΧϯτؒͰΓऔΓ͞ΕΔσʔλ&UIFS ετϨʔδ ঢ়ଶ` τϥϯβΫγϣϯ ηοτ σδλϧॺ໊ &7. ਓؒͱ͔ ࣗಈ ΦϒδΣΫτ Gas ( EVM = ( burn)) . . . B 2025 — 3 — 2025-10-20 – p.29/66
EVM : : ← : Solidity — JavaScript LLL —
Lisp Vyper — Python Fe — Vyper Rust ← NEW! Solidity Vyper Python B 2025 — 3 — 2025-10-20 – p.30/66
Solidity ( ) pragma solidityˆ0.7.0; contract IndivisibleAsset { /* */
string public _name; string public _symbol; uint256 public _quantity; address public _owner; constructor(string memory name, string memory symbol, uint256 quantity) public { _name = name; _symbol = symbol; _quantity = quantity; _owner = msg.sender; } function transfer(address to) public returns (bool) { require (_owner == msg.sender); _owner = to; return true; } } B 2025 — 3 — 2025-10-20 – p.31/66
( , ) (constructor) (deploy) = ( ) ( )
ETH Trusted Actor Model B 2025 — 3 — 2025-10-20 – p.32/66
Ethereum B 2025 — 3 — 2025-10-20 – p.33/66
(1) : ERC20 $ brownie bake token B 2025 —
3 — 2025-10-20 – p.34/66
pragma solidity ˆ0.6.0; /* 0.8.x */ contract Token { (
) : (EVM ) : constructor (...) public { /* */ : } function balanceOf(...) { /* ( ) */ : } : } constructor C (/* */ // ) B 2025 — 3 — 2025-10-20 – p.35/66
ERC-20 ERC (Ethereum Request for Comments) 20 https://github.com/ethereum/EIPs/blob/master/EIPS/eip-20.md (EIP: Ethereum
Improvement Proposal) ← EIP ( ) contract ERC20 { function totalSupply() constant returns (uint totalSupply); function balanceOf(address _owner) constant returns (uint balance); function transfer(address _to, uint _value) returns (bool success); function transferFrom(address _from, address _to, uint _value) returns (bool success); function approve(address _spender, uint _value) returns (bool success); function allowance(address _owner, address _spender) constant returns (uint remaining); event Transfer(address indexed _from, address indexed _to, uint _value); event Approval(address indexed _owner, address indexed _spender, uint _value); } name/ , symbol/ , decimals/ approve allowance ERC-777 ( ) B 2025 — 3 — 2025-10-20 – p.36/66
ERC20 ERC20 ( ) ⇒ ERC20 ( ERC20 ) ERC20
⇒ B 2025 — 3 — 2025-10-20 – p.37/66
(fungible) ERC-20 → ERC-223 (draft) or ERC-777 ( )( )
(non-fungible) ERC-721 ( ) ERC-1155 ( ) ERC-3525 (SLOT ) B 2025 — 3 — 2025-10-20 – p.38/66
Token ( ) string public symbol; string public name; uint256
public decimals; uint256 public totalSupply; mapping(address => uint256) balances; . . . name, symbol decimals : 2 100 1.00 mapping balances B 2025 — 3 — 2025-10-20 – p.39/66
Token ( ) event Transfer(address from, address to, uint256 value);
Token function emit Transfer() ( ) B 2025 — 3 — 2025-10-20 – p.40/66
Token ( ) constructor( string memory _symbol, string memory _name,
uint256 _decimals, uint256 _totalSupply ) public { symbol = _symbol; name = _name; decimals = _decimals; totalSupply = _totalSupply; balances[msg.sender] = _totalSupply; } msg.sender _totalSupply B 2025 — 3 — 2025-10-20 – p.41/66
Token balanceOf() function balanceOf(address _owner) public view returns (uint256) {
return balances[_owner]; } B 2025 — 3 — 2025-10-20 – p.42/66
Token transfer() function transfer(address _to, uint256 _value) public returns (bool)
{ balances[msg.sender] = balances[msg.sender].sub(_value); balances[_to] = balances[_to].add(_value); emit Transfer(msg.sender, _to, _value); return true; } ( ↑ ) OpenZeppelin https://openzeppelin.org B 2025 — 3 — 2025-10-20 – p.43/66
(2) : B 2025 — 3 — 2025-10-20 – p.44/66
GitHub git clone $ git clone https://github.com/ks91/sample-smart-contracts.git sample-smart-contracts contracts, scripts
tests token Token Solidity 0.5 0.6/0.7 (Token ) Solidity B 2025 — 3 — 2025-10-20 – p.45/66
(m2 ) 1 1 B 2025 — 3 — 2025-10-20
– p.46/66
IndivisibleAsset string public _name_; string public _symbol_; uint256 public _quantity_;
address public _owner_; _name_ ( ) _symbol_ _quantity_ (m2 ) _owner_ ‘_’ Solidity . . . B 2025 — 3 — 2025-10-20 – p.47/66
IndivisibleAsset event Transfer(address indexed from, address indexed to); from to
indexed B 2025 — 3 — 2025-10-20 – p.48/66
IndivisibleAsset constructor (string name, string symbol, uint256 quantity) public {
_name_ = name; _symbol_ = symbol; _quantity_ = quantity; _owner_ = msg.sender; } B 2025 — 3 — 2025-10-20 – p.49/66
IndivisibleAsset getOwner() function getOwner() public view returns (address) { return
(_owner_); } B 2025 — 3 — 2025-10-20 – p.50/66
IndivisibleAsset transfer() function transfer(address to) public returns (bool) { require(_owner_
== msg.sender); _owner_ = to; emit Transfer(msg.sender, to); return true; } require (function ) ( ) $ brownie compile B 2025 — 3 — 2025-10-20 – p.51/66
scripts/indivisible asset.py from brownie import * def main(): accounts[0].deploy(IndivisibleAsset, "5322
Endo, Fujisawa", "mˆ2", 300) SFC (300m2 ) ( ) B 2025 — 3 — 2025-10-20 – p.52/66
from brownie import * import pytest def test_owner_and_transfer(IndivisibleAsset): asset =
accounts[0].deploy(IndivisibleAsset, "5322 Endo", "mˆ2", 300) assert asset.getOwner() == accounts[0] asset.transfer(accounts[1], {’from’: accounts[0]}) assert asset.getOwner() == accounts[1] try: asset.transfer(accounts[0], {’from’: accounts[0]}) done = 1 except: done = 0 assert done == 0 accounts[0] accounts[1] B 2025 — 3 — 2025-10-20 – p.53/66
$ pytest tests/test_indivisible_asset.py B 2025 — 3 — 2025-10-20 –
p.54/66
(3) : B 2025 — 3 — 2025-10-20 – p.55/66
transfer settle, retrieve asset, retrieve token 3 B 2025 —
3 — 2025-10-20 – p.56/66
1. ( ) 2. ( ) 3. ( ) B
2025 — 3 — 2025-10-20 – p.57/66
OneTimeEscrow settle() function settle() public returns (bool) { require(_token_.balanceOf(address(this)) >=
_price_); /* this */ require(_asset_.getOwner() == address(this)); _token_.transfer(_seller_ , _price_); _asset_.transfer(_buyer_); emit Settled(); /* */ return true; } settle() transfer ( / ) $ brownie compile B 2025 — 3 — 2025-10-20 – p.58/66
(1) from brownie import * import pytest seller = accounts[0]
buyer = accounts[1] def test_deploy_and_settle(Token, IndivisibleAsset, OneTimeEscrow): asset = accounts[0].deploy(IndivisibleAsset, "5322 Endo", "mˆ2", 300) token = accounts[0].deploy(Token, "Test Token", "TEST", 18, "1000 ether") B 2025 — 3 — 2025-10-20 – p.59/66
(2) token.transfer(buyer, 300, {’from’: accounts[0]}) escrow = accounts[0].deploy(OneTimeEscrow, token, buyer,
asset, seller, 300) accounts[0]/seller accounts[1]/buyer 300 300 TX buyer seller 300 bake Token 300 ETH wei B 2025 — 3 — 2025-10-20 – p.60/66
(3) token.transfer(escrow, 300, {’from’: buyer}) asset.transfer(escrow, {’from’: seller}) assert token.balanceOf(seller)
== 999999999999999999700 assert token.balanceOf(buyer) == 0 assert token.balanceOf(escrow) == 300 assert asset.getOwner() == escrow buyer ( ) 300 seller ( ) B 2025 — 3 — 2025-10-20 – p.61/66
(4) escrow.settle({’from’: seller}) assert token.balanceOf(seller) == 1000000000000000000000 assert token.balanceOf(buyer) ==
0 assert token.balanceOf(escrow) == 0 assert asset.getOwner() == buyer settle() seller OK B 2025 — 3 — 2025-10-20 – p.62/66
$ pytest tests/test_one_time_escrow.py : settle() ⇒ Discord B 2025 —
3 — 2025-10-20 – p.63/66
B 2025 — 3 — 2025-10-20 – p.64/66
1. (1) (2) Ethereum 2025 10 25 ( ) 23:59
JST B 2025 — 3 — 2025-10-20 – p.65/66
B 2025 — 3 — 2025-10-20 – p.66/66