# 20230522 - Uniswap
## createPair
``` solidity
function createPair(address tokenA, address tokenB) external returns (address pair) {
require(tokenA != tokenB, 'UniswapV2: IDENTICAL_ADDRESSES');
// 會先排序過
(address token0, address token1) = tokenA < tokenB ? (tokenA, tokenB) : (tokenB, tokenA);
require(token0 != address(0), 'UniswapV2: ZERO_ADDRESS');
require(getPair[token0][token1] == address(0), 'UniswapV2: PAIR_EXISTS'); // single check is sufficient
bytes memory bytecode = type(UniswapV2Pair).creationCode;
// 所以不管 createPair 傳入的token 順序為何,這邊 salt 都是一樣的
bytes32 salt = keccak256(abi.encodePacked(token0, token1));
// create2 創造的時候所需要的資訊跟 acconut 的 nonce 無關
// salt = oxff (constant) + deployer + keccak256(abi.encodePacked(token0, token1)) + init code hash
assembly {
// 因為 bytes 為非固定長度的資料
// 所以 solidity 在儲存資料時會預留前 32bytes 來表示bytes 的長度
// 所以這邊要從第 32 byte 開始讀取資料
// (這個行為針對動態的資料型別,像是字串也是這樣)
pair := create2(0, add(bytecode, 32), mload(bytecode), salt)
}
IUniswapV2Pair(pair).initialize(token0, token1);
getPair[token0][token1] = pair;
getPair[token1][token0] = pair;
allPairs.push(pair);
emit PairCreated(token0, token1, pair, allPairs.length);
}
```
### lock modifier
- 用來預防重送攻擊
``` solidity
modifier lock() {
require(unlocked == 1, 'UniswapV2: LOCKED');
unlocked = 0;
_;
unlocked = 1;
}
```
- 未來如果合約內有要 external call + 更改內部狀態的操作,要先更改內部狀態,再去做 external call (避免重送攻擊)
- 也可使用 [OpenZeppelin-ReentrancyGuard](https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/security/ReentrancyGuard.sol)
## UniswapV2Library
### pairFor
``` solidity
// Create2(value, offset, size, salt)
function pairFor(address factory, address tokenA, address tokenB) internal pure returns (address pair) {
(address token0, address token1) = sortTokens(tokenA, tokenB);
pair = address(uint(keccak256(abi.encodePacked(
hex'ff',
factory,
keccak256(abi.encodePacked(token0, token1)),
// 如果有在 local 改 uniswap contract code ,要注意這邊也需要更新,因為 init code 會不同
hex'96e8ac4277198ff8b6f785478aa9a39f403cb768dd02cbee326c3e7da348845f' // init code hash
))));
}
```