ETH Price: $2,089.81 (-0.31%)

Transaction Decoder

Block:
12959191 at Aug-04-2021 02:28:23 PM +UTC
Transaction Fee:
0.007743861 ETH $16.18
Gas Used:
164,763 Gas / 47 Gwei

Emitted Events:

131 MaticToken.Transfer( from=[Sender] 0x703227ad696e71d20acf075fe8a29266e4b4e811, to=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, value=2910000000000000000000 )
132 MaticToken.Approval( owner=[Sender] 0x703227ad696e71d20acf075fe8a29266e4b4e811, spender=[Receiver] AggregationRouterV3, value=115792089237316195423570985008687907853269984665640564029445584007913129639935 )
133 MaticToken.Approval( owner=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, spender=BPool, value=2910000000000000000000 )
134 BPool.0x8201aa3f00000000000000000000000000000000000000000000000000000000( 0x8201aa3f00000000000000000000000000000000000000000000000000000000, 0x000000000000000000000000c176761d388caf2f56cf03329d82e1e7c48ae09c, 0000000000000000000000000000000000000000000000000000000000000020, 00000000000000000000000000000000000000000000000000000000000000a4, 8201aa3f0000000000000000000000007d1afa7b718fb893db30a3abc0cfc608, aacfebb000000000000000000000000000000000000000000000009dc05cce28, c2b80000000000000000000000000000d2877702675e6ceb975b4a1dff9fb7ba, f4c91ea900000000000000000000000000000000000000000000000000000000, 00000001ffffffffffffffffffffffffffffffffffffffffffffffffffffffff, ffffffff00000000000000000000000000000000000000000000000000000000 )
135 BPool.LOG_SWAP( caller=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, tokenIn=MaticToken, tokenOut=WrappedLuna, tokenAmountIn=2910000000000000000000, tokenAmountOut=200621548478527897788 )
136 MaticToken.Transfer( from=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, to=BPool, value=2910000000000000000000 )
137 MaticToken.Approval( owner=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, spender=BPool, value=0 )
138 WrappedLuna.Transfer( from=BPool, to=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, value=200621548478527897788 )
139 WrappedLuna.Transfer( from=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, to=[Sender] 0x703227ad696e71d20acf075fe8a29266e4b4e811, value=200621548478527897788 )
140 AggregationRouterV3.Swapped( sender=[Sender] 0x703227ad696e71d20acf075fe8a29266e4b4e811, srcToken=MaticToken, dstToken=WrappedLuna, dstReceiver=[Sender] 0x703227ad696e71d20acf075fe8a29266e4b4e811, spentAmount=2910000000000000000000, returnAmount=200621548478527897788 )
141 ChiToken.Transfer( from=[Sender] 0x703227ad696e71d20acf075fe8a29266e4b4e811, to=0x0000000000000000000000000000000000000000, value=6 )
142 ChiToken.Approval( owner=[Sender] 0x703227ad696e71d20acf075fe8a29266e4b4e811, spender=[Receiver] AggregationRouterV3, value=115792089237316195423570985008687907853269984665640564039457584007913129639849 )

Account State Difference:

  Address   Before After State Difference Code
0x00000000...b0eF1fA1c
0x03670fd9...CFBD84546
0 Eth
Nonce: 0
0 Eth
Nonce: 0
From: 0 To: 0
0x1C34C1d2...CC92c8699
0x241cF029...7537bBF95
0 Eth
Nonce: 0
0 Eth
Nonce: 0
From: 0 To: 0
0x24c0c435...7ae17cE4e
0 Eth
Nonce: 0
0 Eth
Nonce: 0
From: 0 To: 0
0x703227AD...6E4B4e811
1.959740172433306436 Eth
Nonce: 202
1.951996311433306436 Eth
Nonce: 203
0.007743861
0x7D1AfA7B...8AaCfeBB0
(Miner: 0xb7e...707)
19.747214737587829544 Eth19.754958598587829544 Eth0.007743861
0xbB77c1ae...8E759e3e9
0 Eth
Nonce: 0
0 Eth
Nonce: 0
From: 0 To: 0
0xd2877702...AF4C91ea9
0xfc434f03...d8e792B06
0 Eth
Nonce: 0
0 Eth
Nonce: 0
From: 0 To: 0
0xfFfe5C5E...4d488C671
0 Eth
Nonce: 0
0 Eth
Nonce: 0
From: 0 To: 0

Execution Trace

AggregationRouterV3.discountedSwap( caller=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, desc=[{name:srcToken, type:address, order:1, indexed:false, value:0x7D1AfA7B718fb893dB30A3aBc0Cfc608AaCfeBB0, valueString:0x7D1AfA7B718fb893dB30A3aBc0Cfc608AaCfeBB0}, {name:dstToken, type:address, order:2, indexed:false, value:0xd2877702675e6cEb975b4A1dFf9fb7BAF4C91ea9, valueString:0xd2877702675e6cEb975b4A1dFf9fb7BAF4C91ea9}, {name:srcReceiver, type:address, order:3, indexed:false, value:0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, valueString:0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c}, {name:dstReceiver, type:address, order:4, indexed:false, value:0x703227AD696e71d20Acf075FE8a29266E4B4e811, valueString:0x703227AD696e71d20Acf075FE8a29266E4B4e811}, {name:amount, type:uint256, order:5, indexed:false, value:2910000000000000000000, valueString:2910000000000000000000}, {name:minReturnAmount, type:uint256, order:6, indexed:false, value:198615332993742618810, valueString:198615332993742618810}, {name:flags, type:uint256, order:7, indexed:false, value:12, valueString:12}, {name:permit, type:bytes, order:8, indexed:false, value:0x, valueString:0x}], data=0x000000000000000000000000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000000030000000000000000000000000000000000000000000000000000000000000060000000000000000000000000000000000000000000000000000000000000018000000000000000000000000000000000000000000000000000000000000002E000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000800000000000000000000000000000000000000000000000000000000000000064EB5625D90000000000000000000000007D1AFA7B718FB893DB30A3ABC0CFC608AACFEBB00000000000000000000000001C34C1D24BBA5CCDB173F203DBFF6CFCC92C869900000000000000000000000000000000000000000000009DC05CCE28C2B80000000000000000000000000000000000000000000000000000000000008000000000000000000000001C34C1D24BBA5CCDB173F203DBFF6CFCC92C869900000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000A48201AA3F0000000000000000000000007D1AFA7B718FB893DB30A3ABC0CFC608AACFEBB000000000000000000000000000000000000000000000009DC05CCE28C2B80000000000000000000000000000D2877702675E6CEB975B4A1DFF9FB7BAF4C91EA90000000000000000000000000000000000000000000000000000000000000001FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000001A4B3AF37C000000000000000000000000000000000000000000000000000000000000000808000000000000000000000000000000000000000000000000000000000000044000000000000000000000000D2877702675E6CEB975B4A1DFF9FB7BAF4C91EA9000000000000000000000000000000010000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000800000000000000000000000000000000000000000000000000000000000000064D1660F99000000000000000000000000D2877702675E6CEB975B4A1DFF9FB7BAF4C91EA9000000000000000000000000703227AD696E71D20ACF075FE8A29266E4B4E81100000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ) => ( returnAmount=200621548478527897788, gasLeft=59265, chiSpent=6 )
  • AggregationRouterV3.swap( caller=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, desc=[{name:srcToken, type:address, order:1, indexed:false, value:0x7D1AfA7B718fb893dB30A3aBc0Cfc608AaCfeBB0, valueString:0x7D1AfA7B718fb893dB30A3aBc0Cfc608AaCfeBB0}, {name:dstToken, type:address, order:2, indexed:false, value:0xd2877702675e6cEb975b4A1dFf9fb7BAF4C91ea9, valueString:0xd2877702675e6cEb975b4A1dFf9fb7BAF4C91ea9}, {name:srcReceiver, type:address, order:3, indexed:false, value:0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, valueString:0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c}, {name:dstReceiver, type:address, order:4, indexed:false, value:0x703227AD696e71d20Acf075FE8a29266E4B4e811, valueString:0x703227AD696e71d20Acf075FE8a29266E4B4e811}, {name:amount, type:uint256, order:5, indexed:false, value:2910000000000000000000, valueString:2910000000000000000000}, {name:minReturnAmount, type:uint256, order:6, indexed:false, value:198615332993742618810, valueString:198615332993742618810}, {name:flags, type:uint256, order:7, indexed:false, value:12, valueString:12}, {name:permit, type:bytes, order:8, indexed:false, value:0x, valueString:0x}], data=0x000000000000000000000000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000000030000000000000000000000000000000000000000000000000000000000000060000000000000000000000000000000000000000000000000000000000000018000000000000000000000000000000000000000000000000000000000000002E000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000800000000000000000000000000000000000000000000000000000000000000064EB5625D90000000000000000000000007D1AFA7B718FB893DB30A3ABC0CFC608AACFEBB00000000000000000000000001C34C1D24BBA5CCDB173F203DBFF6CFCC92C869900000000000000000000000000000000000000000000009DC05CCE28C2B80000000000000000000000000000000000000000000000000000000000008000000000000000000000001C34C1D24BBA5CCDB173F203DBFF6CFCC92C869900000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000A48201AA3F0000000000000000000000007D1AFA7B718FB893DB30A3ABC0CFC608AACFEBB000000000000000000000000000000000000000000000009DC05CCE28C2B80000000000000000000000000000D2877702675E6CEB975B4A1DFF9FB7BAF4C91EA90000000000000000000000000000000000000000000000000000000000000001FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000001A4B3AF37C000000000000000000000000000000000000000000000000000000000000000808000000000000000000000000000000000000000000000000000000000000044000000000000000000000000D2877702675E6CEB975B4A1DFF9FB7BAF4C91EA9000000000000000000000000000000010000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000800000000000000000000000000000000000000000000000000000000000000064D1660F99000000000000000000000000D2877702675E6CEB975B4A1DFF9FB7BAF4C91EA9000000000000000000000000703227AD696E71D20ACF075FE8A29266E4B4E81100000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ) => ( returnAmount=200621548478527897788, gasLeft=127625 )
    • MaticToken.transferFrom( from=0x703227AD696e71d20Acf075FE8a29266E4B4e811, to=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, value=2910000000000000000000 ) => ( True )
    • WrappedLuna.balanceOf( account=0x703227AD696e71d20Acf075FE8a29266E4B4e811 ) => ( 0 )
    • MEV Bot: 0xc17...09c.d9c45357( )
      • MEV Bot: 0xc17...09c.eb5625d9( )
        • MaticToken.approve( spender=0x1C34C1d24bBa5ccDB173F203DbFf6CFCC92c8699, value=2910000000000000000000 ) => ( True )
        • BPool.swapExactAmountIn( tokenIn=0x7D1AfA7B718fb893dB30A3aBc0Cfc608AaCfeBB0, tokenAmountIn=2910000000000000000000, tokenOut=0xd2877702675e6cEb975b4A1dFf9fb7BAF4C91ea9, minAmountOut=1, maxPrice=115792089237316195423570985008687907853269984665640564039457584007913129639935 ) => ( tokenAmountOut=200621548478527897788, spotPriceAfter=14828749007120913251 )
          • MaticToken.transferFrom( from=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, to=0x1C34C1d24bBa5ccDB173F203DbFf6CFCC92c8699, value=2910000000000000000000 ) => ( True )
          • WrappedLuna.transfer( recipient=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c, amount=200621548478527897788 ) => ( True )
          • MEV Bot: 0xc17...09c.b3af37c0( )
            • WrappedLuna.balanceOf( account=0xc176761d388CAf2F56Cf03329d82E1e7C48AE09c ) => ( 200621548478527897788 )
            • MEV Bot: 0xc17...09c.d1660f99( )
              • WrappedLuna.transfer( recipient=0x703227AD696e71d20Acf075FE8a29266E4B4e811, amount=200621548478527897788 ) => ( True )
              • WrappedLuna.balanceOf( account=0x703227AD696e71d20Acf075FE8a29266E4B4e811 ) => ( 200621548478527897788 )
              • MEV Bot: 0xc17...09c.1d97832e( )
              • ChiToken.freeFromUpTo( from=0x703227AD696e71d20Acf075FE8a29266E4B4e811, value=6 ) => ( 6 )
                • 0xfc434f0320219f15d072cf83c9640efd8e792b06.CALL( )
                  • ChiToken.SELFDESTRUCT( )
                  • 0x241cf029f165e017830aba0336c8dc97537bbf95.CALL( )
                    • ChiToken.SELFDESTRUCT( )
                    • 0xfffe5c5ecd124b5d91394fec1648e274d488c671.CALL( )
                      • ChiToken.SELFDESTRUCT( )
                      • 0x03670fd926edbab09b94ff9749f9c74cfbd84546.CALL( )
                        • ChiToken.SELFDESTRUCT( )
                        • 0x24c0c435fb8b84b3689cc14359c35af7ae17ce4e.CALL( )
                          • ChiToken.SELFDESTRUCT( )
                          • 0xbb77c1ae146e380008496fd1a1a0a1c8e759e3e9.CALL( )
                            • ChiToken.SELFDESTRUCT( )
                              File 1 of 5: AggregationRouterV3
                              /*
                                                                                         ,▄▓▓██▌   ,╓▄▄▓▓▓▓▓▓▓▓▄▄▄,,                          
                                                                                      ,▓██▓███▓▄▓███▓╬╬╬╬╬╬╬╬╬╬╬╬╬▓███▓▄,                     
                                                                                ▄█   ▓██╬╣███████╬▓▀╬╬▓▓▓████████████▓█████▄,                 
                                                                               ▓██▌ ▓██╬╣██████╬▓▌  ██████████████████████▌╙╙▀ⁿ
                                                                              ▐████████╬▓████▓▓█╨ ▄ ╟█████████▓▓╬╬╬╬╬▓▓█████▓▄
                                                                └▀▓▓▄╓        ╟█▓╣█████▓██████▀ ╓█▌ ███████▓▓▓▓▓╬╬╬╬╬╬╬╬╬╬╬╬▓██▓▄
                                                                   └▀████▓▄╥  ▐██╬╬██████████╙ Æ▀─ ▓███▀╚╠╬╩▀▀███████▓▓╬╬╬╬╬╬╬╬╬██▄
                                                                      └▀██▓▀▀█████▓╬▓██████▀     ▄█████▒╠"      └╙▓██████▓╬╬╬╬╬╬╬╬██▄
                                                                         └▀██▄,└╙▀▀████▌└╙    ^"▀╙╙╙"╙██      @▄    ╙▀███████╬╬╬╬╬╬╬██µ
                                                                            └▀██▓▄, ██▌       ╒       ╙█▓     ]▓█▓╔    ▀███████▓╬╬╬╬╬▓█▌
                                                                                ▀█████       ▓         ╟█▌    ]╠██▓░▒╓   ▀████████╬╬╬╬╣█▌
                                                                                ▐████      ╓█▀█▌      ,██▌    ╚Å███▓▒▒╠╓  ╙█████████╬╬╬╣█▌
                                                                                └████     ▓█░░▓█      ▀▀▀    φ▒╫████▒▒▒▒╠╓  █████████▓╬╬▓█µ
                                                                                 ╘███µ ▌▄█▓▄▓▀`     ,▀    ,╔╠░▓██████▌╠▒▒▒φ  ██████████╬╬██
                                                                                 ▐████µ╙▓▀`     ,▀╙,╔╔φφφ╠░▄▓███████▌░▓╙▒▒▒╠ └██╬███████╬▓█⌐
                                                                                 ╫██ ▓▌         ▌φ▒▒░▓██████████████▌▒░▓╚▒▒▒╠ ▓██╬▓██████╣█▌
                                                                                 ██▌           ▌╔▒▒▄████████████████▒▒▒░▌╠▒▒▒≥▐██▓╬╬███████▌
                                                                                 ██▌      ,╓φ╠▓«▒▒▓████▀  ▀█████████▌▒▒▒╟░▒▒▒▒▐███╬╬╣████▓█▌
                                                                                ▐██      ╠▒▄▓▓███▓████└     ▀████████▌▒▒░▌╚▒▒▒▐███▓╬╬████ ╙▌
                                                                                ███  )  ╠▒░░░▒░╬████▀        └████████░▒▒░╬∩▒▒▓████╬╬╣███
                                                                               ▓██    ╠╠▒▒▐█▀▀▌`░╫██           ███████▒▒▒▒░▒▒½█████╬╬╣███
                                                                              ███ ,█▄ ╠▒▒▒╫▌,▄▀,▒╫██           ╟██████▒▒▒░╣⌠▒▓█████╬╬╣██▌
                                                                             ╘██µ ██` ╠▒▒░██╬φ╠▄▓██`            ██████░░▌φ╠░▓█████▓╬╬▓██
                                                                              ╟██  .φ╠▒░▄█▀░░▄██▀└              █████▌▒╣φ▒░▓██████╬╬╣██
                                                                               ▀██▄▄▄╓▄███████▀                ▐█████░▓φ▒▄███████▓╬╣██
                                                                                 ╙▀▀▀██▀└                      ████▓▄▀φ▄▓████████╬▓█▀
                                                                                                              ▓███╬╩╔╣██████████▓██└
                                                                                                            ╓████▀▄▓████████▀████▀
                                                                                                          ,▓███████████████─]██╙
                                                                                                       ,▄▓██████████████▀└  ╙
                                                                                                  ,╓▄▓███████████████▀╙
                                                                                           `"▀▀▀████████▀▀▀▀`▄███▀▀└
                                                                                                            └└
                                                              
                                                  
                              
                                                  11\   11\                     11\             11\   11\            11\                                       11\       
                                                1111 |  \__|                    11 |            111\  11 |           11 |                                      11 |      
                                                \_11 |  11\ 1111111\   1111111\ 1111111\        1111\ 11 | 111111\ 111111\   11\  11\  11\  111111\   111111\  11 |  11\ 
                                                  11 |  11 |11  __11\ 11  _____|11  __11\       11 11\11 |11  __11\\_11  _|  11 | 11 | 11 |11  __11\ 11  __11\ 11 | 11  |
                                                  11 |  11 |11 |  11 |11 /      11 |  11 |      11 \1111 |11111111 | 11 |    11 | 11 | 11 |11 /  11 |11 |  \__|111111  / 
                                                  11 |  11 |11 |  11 |11 |      11 |  11 |      11 |\111 |11   ____| 11 |11\ 11 | 11 | 11 |11 |  11 |11 |      11  _11<  
                                                111111\ 11 |11 |  11 |\1111111\ 11 |  11 |      11 | \11 |\1111111\  \1111  |\11111\1111  |\111111  |11 |      11 | \11\ 
                                                \______|\__|\__|  \__| \_______|\__|  \__|      \__|  \__| \_______|  \____/  \_____\____/  \______/ \__|      \__|  \__|
                                                                                                                                                                         
                                                                                                                                                                         
                                                                                                                                                                         
                                                             111111\                                                               11\     11\                           
                                                            11  __11\                                                              11 |    \__|                          
                                                            11 /  11 | 111111\   111111\   111111\   111111\   111111\   111111\ 111111\   11\  111111\  1111111\        
                                                            11111111 |11  __11\ 11  __11\ 11  __11\ 11  __11\ 11  __11\  \____11\\_11  _|  11 |11  __11\ 11  __11\       
                                                            11  __11 |11 /  11 |11 /  11 |11 |  \__|11111111 |11 /  11 | 1111111 | 11 |    11 |11 /  11 |11 |  11 |      
                                                            11 |  11 |11 |  11 |11 |  11 |11 |      11   ____|11 |  11 |11  __11 | 11 |11\ 11 |11 |  11 |11 |  11 |      
                                                            11 |  11 |\1111111 |\1111111 |11 |      \1111111\ \1111111 |\1111111 | \1111  |11 |\111111  |11 |  11 |      
                                                            \__|  \__| \____11 | \____11 |\__|       \_______| \____11 | \_______|  \____/ \__| \______/ \__|  \__|      
                                                                      11\   11 |11\   11 |                    11\   11 |                                                 
                                                                      \111111  |\111111  |                    \111111  |                                                 
                                                                       \______/  \______/                      \______/                                                  
                                                                              1111111\                        11\                                                        
                                                                              11  __11\                       11 |                                                       
                                                                              11 |  11 | 111111\  11\   11\ 111111\    111111\   111111\                                 
                                                                              1111111  |11  __11\ 11 |  11 |\_11  _|  11  __11\ 11  __11\                                
                                                                              11  __11< 11 /  11 |11 |  11 |  11 |    11111111 |11 |  \__|                               
                                                                              11 |  11 |11 |  11 |11 |  11 |  11 |11\ 11   ____|11 |                                     
                                                                              11 |  11 |\111111  |\111111  |  \1111  |\1111111\ 11 |                                     
                                                                              \__|  \__| \______/  \______/    \____/  \_______|\__|                                     
                              */
                              
                              // File @openzeppelin/contracts/utils/Context.sol@v3.4.1
                              
                              // SPDX-License-Identifier: MIT
                              
                              pragma solidity >=0.6.0 <0.8.0;
                              
                              /*
                               * @dev Provides information about the current execution context, including the
                               * sender of the transaction and its data. While these are generally available
                               * via msg.sender and msg.data, they should not be accessed in such a direct
                               * manner, since when dealing with GSN meta-transactions the account sending and
                               * paying for execution may not be the actual sender (as far as an application
                               * is concerned).
                               *
                               * This contract is only required for intermediate, library-like contracts.
                               */
                              abstract contract Context {
                                  function _msgSender() internal view virtual returns (address payable) {
                                      return msg.sender;
                                  }
                              
                                  function _msgData() internal view virtual returns (bytes memory) {
                                      this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
                                      return msg.data;
                                  }
                              }
                              
                              
                              // File @openzeppelin/contracts/access/Ownable.sol@v3.4.1
                              
                              
                              
                              pragma solidity >=0.6.0 <0.8.0;
                              
                              /**
                               * @dev Contract module which provides a basic access control mechanism, where
                               * there is an account (an owner) that can be granted exclusive access to
                               * specific functions.
                               *
                               * By default, the owner account will be the one that deploys the contract. This
                               * can later be changed with {transferOwnership}.
                               *
                               * This module is used through inheritance. It will make available the modifier
                               * `onlyOwner`, which can be applied to your functions to restrict their use to
                               * the owner.
                               */
                              abstract contract Ownable is Context {
                                  address private _owner;
                              
                                  event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
                              
                                  /**
                                   * @dev Initializes the contract setting the deployer as the initial owner.
                                   */
                                  constructor () internal {
                                      address msgSender = _msgSender();
                                      _owner = msgSender;
                                      emit OwnershipTransferred(address(0), msgSender);
                                  }
                              
                                  /**
                                   * @dev Returns the address of the current owner.
                                   */
                                  function owner() public view virtual returns (address) {
                                      return _owner;
                                  }
                              
                                  /**
                                   * @dev Throws if called by any account other than the owner.
                                   */
                                  modifier onlyOwner() {
                                      require(owner() == _msgSender(), "Ownable: caller is not the owner");
                                      _;
                                  }
                              
                                  /**
                                   * @dev Leaves the contract without owner. It will not be possible to call
                                   * `onlyOwner` functions anymore. Can only be called by the current owner.
                                   *
                                   * NOTE: Renouncing ownership will leave the contract without an owner,
                                   * thereby removing any functionality that is only available to the owner.
                                   */
                                  function renounceOwnership() public virtual onlyOwner {
                                      emit OwnershipTransferred(_owner, address(0));
                                      _owner = address(0);
                                  }
                              
                                  /**
                                   * @dev Transfers ownership of the contract to a new account (`newOwner`).
                                   * Can only be called by the current owner.
                                   */
                                  function transferOwnership(address newOwner) public virtual onlyOwner {
                                      require(newOwner != address(0), "Ownable: new owner is the zero address");
                                      emit OwnershipTransferred(_owner, newOwner);
                                      _owner = newOwner;
                                  }
                              }
                              
                              
                              // File @openzeppelin/contracts/token/ERC20/IERC20.sol@v3.4.1
                              
                              
                              
                              pragma solidity >=0.6.0 <0.8.0;
                              
                              /**
                               * @dev Interface of the ERC20 standard as defined in the EIP.
                               */
                              interface IERC20 {
                                  /**
                                   * @dev Returns the amount of tokens in existence.
                                   */
                                  function totalSupply() external view returns (uint256);
                              
                                  /**
                                   * @dev Returns the amount of tokens owned by `account`.
                                   */
                                  function balanceOf(address account) external view returns (uint256);
                              
                                  /**
                                   * @dev Moves `amount` tokens from the caller's account to `recipient`.
                                   *
                                   * Returns a boolean value indicating whether the operation succeeded.
                                   *
                                   * Emits a {Transfer} event.
                                   */
                                  function transfer(address recipient, uint256 amount) external returns (bool);
                              
                                  /**
                                   * @dev Returns the remaining number of tokens that `spender` will be
                                   * allowed to spend on behalf of `owner` through {transferFrom}. This is
                                   * zero by default.
                                   *
                                   * This value changes when {approve} or {transferFrom} are called.
                                   */
                                  function allowance(address owner, address spender) external view returns (uint256);
                              
                                  /**
                                   * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                                   *
                                   * Returns a boolean value indicating whether the operation succeeded.
                                   *
                                   * IMPORTANT: Beware that changing an allowance with this method brings the risk
                                   * that someone may use both the old and the new allowance by unfortunate
                                   * transaction ordering. One possible solution to mitigate this race
                                   * condition is to first reduce the spender's allowance to 0 and set the
                                   * desired value afterwards:
                                   * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                                   *
                                   * Emits an {Approval} event.
                                   */
                                  function approve(address spender, uint256 amount) external returns (bool);
                              
                                  /**
                                   * @dev Moves `amount` tokens from `sender` to `recipient` using the
                                   * allowance mechanism. `amount` is then deducted from the caller's
                                   * allowance.
                                   *
                                   * Returns a boolean value indicating whether the operation succeeded.
                                   *
                                   * Emits a {Transfer} event.
                                   */
                                  function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
                              
                                  /**
                                   * @dev Emitted when `value` tokens are moved from one account (`from`) to
                                   * another (`to`).
                                   *
                                   * Note that `value` may be zero.
                                   */
                                  event Transfer(address indexed from, address indexed to, uint256 value);
                              
                                  /**
                                   * @dev Emitted when the allowance of a `spender` for an `owner` is set by
                                   * a call to {approve}. `value` is the new allowance.
                                   */
                                  event Approval(address indexed owner, address indexed spender, uint256 value);
                              }
                              
                              
                              // File @openzeppelin/contracts/math/SafeMath.sol@v3.4.1
                              
                              
                              
                              pragma solidity >=0.6.0 <0.8.0;
                              
                              /**
                               * @dev Wrappers over Solidity's arithmetic operations with added overflow
                               * checks.
                               *
                               * Arithmetic operations in Solidity wrap on overflow. This can easily result
                               * in bugs, because programmers usually assume that an overflow raises an
                               * error, which is the standard behavior in high level programming languages.
                               * `SafeMath` restores this intuition by reverting the transaction when an
                               * operation overflows.
                               *
                               * Using this library instead of the unchecked operations eliminates an entire
                               * class of bugs, so it's recommended to use it always.
                               */
                              library SafeMath {
                                  /**
                                   * @dev Returns the addition of two unsigned integers, with an overflow flag.
                                   *
                                   * _Available since v3.4._
                                   */
                                  function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                                      uint256 c = a + b;
                                      if (c < a) return (false, 0);
                                      return (true, c);
                                  }
                              
                                  /**
                                   * @dev Returns the substraction of two unsigned integers, with an overflow flag.
                                   *
                                   * _Available since v3.4._
                                   */
                                  function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                                      if (b > a) return (false, 0);
                                      return (true, a - b);
                                  }
                              
                                  /**
                                   * @dev Returns the multiplication of two unsigned integers, with an overflow flag.
                                   *
                                   * _Available since v3.4._
                                   */
                                  function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                                      // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                                      // benefit is lost if 'b' is also tested.
                                      // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                                      if (a == 0) return (true, 0);
                                      uint256 c = a * b;
                                      if (c / a != b) return (false, 0);
                                      return (true, c);
                                  }
                              
                                  /**
                                   * @dev Returns the division of two unsigned integers, with a division by zero flag.
                                   *
                                   * _Available since v3.4._
                                   */
                                  function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                                      if (b == 0) return (false, 0);
                                      return (true, a / b);
                                  }
                              
                                  /**
                                   * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag.
                                   *
                                   * _Available since v3.4._
                                   */
                                  function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                                      if (b == 0) return (false, 0);
                                      return (true, a % b);
                                  }
                              
                                  /**
                                   * @dev Returns the addition of two unsigned integers, reverting on
                                   * overflow.
                                   *
                                   * Counterpart to Solidity's `+` operator.
                                   *
                                   * Requirements:
                                   *
                                   * - Addition cannot overflow.
                                   */
                                  function add(uint256 a, uint256 b) internal pure returns (uint256) {
                                      uint256 c = a + b;
                                      require(c >= a, "SafeMath: addition overflow");
                                      return c;
                                  }
                              
                                  /**
                                   * @dev Returns the subtraction of two unsigned integers, reverting on
                                   * overflow (when the result is negative).
                                   *
                                   * Counterpart to Solidity's `-` operator.
                                   *
                                   * Requirements:
                                   *
                                   * - Subtraction cannot overflow.
                                   */
                                  function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                                      require(b <= a, "SafeMath: subtraction overflow");
                                      return a - b;
                                  }
                              
                                  /**
                                   * @dev Returns the multiplication of two unsigned integers, reverting on
                                   * overflow.
                                   *
                                   * Counterpart to Solidity's `*` operator.
                                   *
                                   * Requirements:
                                   *
                                   * - Multiplication cannot overflow.
                                   */
                                  function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                                      if (a == 0) return 0;
                                      uint256 c = a * b;
                                      require(c / a == b, "SafeMath: multiplication overflow");
                                      return c;
                                  }
                              
                                  /**
                                   * @dev Returns the integer division of two unsigned integers, reverting on
                                   * division by zero. The result is rounded towards zero.
                                   *
                                   * Counterpart to Solidity's `/` operator. Note: this function uses a
                                   * `revert` opcode (which leaves remaining gas untouched) while Solidity
                                   * uses an invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   *
                                   * - The divisor cannot be zero.
                                   */
                                  function div(uint256 a, uint256 b) internal pure returns (uint256) {
                                      require(b > 0, "SafeMath: division by zero");
                                      return a / b;
                                  }
                              
                                  /**
                                   * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
                                   * reverting when dividing by zero.
                                   *
                                   * Counterpart to Solidity's `%` operator. This function uses a `revert`
                                   * opcode (which leaves remaining gas untouched) while Solidity uses an
                                   * invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   *
                                   * - The divisor cannot be zero.
                                   */
                                  function mod(uint256 a, uint256 b) internal pure returns (uint256) {
                                      require(b > 0, "SafeMath: modulo by zero");
                                      return a % b;
                                  }
                              
                                  /**
                                   * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
                                   * overflow (when the result is negative).
                                   *
                                   * CAUTION: This function is deprecated because it requires allocating memory for the error
                                   * message unnecessarily. For custom revert reasons use {trySub}.
                                   *
                                   * Counterpart to Solidity's `-` operator.
                                   *
                                   * Requirements:
                                   *
                                   * - Subtraction cannot overflow.
                                   */
                                  function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                                      require(b <= a, errorMessage);
                                      return a - b;
                                  }
                              
                                  /**
                                   * @dev Returns the integer division of two unsigned integers, reverting with custom message on
                                   * division by zero. The result is rounded towards zero.
                                   *
                                   * CAUTION: This function is deprecated because it requires allocating memory for the error
                                   * message unnecessarily. For custom revert reasons use {tryDiv}.
                                   *
                                   * Counterpart to Solidity's `/` operator. Note: this function uses a
                                   * `revert` opcode (which leaves remaining gas untouched) while Solidity
                                   * uses an invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   *
                                   * - The divisor cannot be zero.
                                   */
                                  function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                                      require(b > 0, errorMessage);
                                      return a / b;
                                  }
                              
                                  /**
                                   * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
                                   * reverting with custom message when dividing by zero.
                                   *
                                   * CAUTION: This function is deprecated because it requires allocating memory for the error
                                   * message unnecessarily. For custom revert reasons use {tryMod}.
                                   *
                                   * Counterpart to Solidity's `%` operator. This function uses a `revert`
                                   * opcode (which leaves remaining gas untouched) while Solidity uses an
                                   * invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   *
                                   * - The divisor cannot be zero.
                                   */
                                  function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                                      require(b > 0, errorMessage);
                                      return a % b;
                                  }
                              }
                              
                              
                              // File @openzeppelin/contracts/utils/Address.sol@v3.4.1
                              
                              
                              
                              pragma solidity >=0.6.2 <0.8.0;
                              
                              /**
                               * @dev Collection of functions related to the address type
                               */
                              library Address {
                                  /**
                                   * @dev Returns true if `account` is a contract.
                                   *
                                   * [IMPORTANT]
                                   * ====
                                   * It is unsafe to assume that an address for which this function returns
                                   * false is an externally-owned account (EOA) and not a contract.
                                   *
                                   * Among others, `isContract` will return false for the following
                                   * types of addresses:
                                   *
                                   *  - an externally-owned account
                                   *  - a contract in construction
                                   *  - an address where a contract will be created
                                   *  - an address where a contract lived, but was destroyed
                                   * ====
                                   */
                                  function isContract(address account) internal view returns (bool) {
                                      // This method relies on extcodesize, which returns 0 for contracts in
                                      // construction, since the code is only stored at the end of the
                                      // constructor execution.
                              
                                      uint256 size;
                                      // solhint-disable-next-line no-inline-assembly
                                      assembly { size := extcodesize(account) }
                                      return size > 0;
                                  }
                              
                                  /**
                                   * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
                                   * `recipient`, forwarding all available gas and reverting on errors.
                                   *
                                   * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
                                   * of certain opcodes, possibly making contracts go over the 2300 gas limit
                                   * imposed by `transfer`, making them unable to receive funds via
                                   * `transfer`. {sendValue} removes this limitation.
                                   *
                                   * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
                                   *
                                   * IMPORTANT: because control is transferred to `recipient`, care must be
                                   * taken to not create reentrancy vulnerabilities. Consider using
                                   * {ReentrancyGuard} or the
                                   * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                                   */
                                  function sendValue(address payable recipient, uint256 amount) internal {
                                      require(address(this).balance >= amount, "Address: insufficient balance");
                              
                                      // solhint-disable-next-line avoid-low-level-calls, avoid-call-value
                                      (bool success, ) = recipient.call{ value: amount }("");
                                      require(success, "Address: unable to send value, recipient may have reverted");
                                  }
                              
                                  /**
                                   * @dev Performs a Solidity function call using a low level `call`. A
                                   * plain`call` is an unsafe replacement for a function call: use this
                                   * function instead.
                                   *
                                   * If `target` reverts with a revert reason, it is bubbled up by this
                                   * function (like regular Solidity function calls).
                                   *
                                   * Returns the raw returned data. To convert to the expected return value,
                                   * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
                                   *
                                   * Requirements:
                                   *
                                   * - `target` must be a contract.
                                   * - calling `target` with `data` must not revert.
                                   *
                                   * _Available since v3.1._
                                   */
                                  function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                                    return functionCall(target, data, "Address: low-level call failed");
                                  }
                              
                                  /**
                                   * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                                   * `errorMessage` as a fallback revert reason when `target` reverts.
                                   *
                                   * _Available since v3.1._
                                   */
                                  function functionCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) {
                                      return functionCallWithValue(target, data, 0, errorMessage);
                                  }
                              
                                  /**
                                   * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                                   * but also transferring `value` wei to `target`.
                                   *
                                   * Requirements:
                                   *
                                   * - the calling contract must have an ETH balance of at least `value`.
                                   * - the called Solidity function must be `payable`.
                                   *
                                   * _Available since v3.1._
                                   */
                                  function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                                      return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                                  }
                              
                                  /**
                                   * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                                   * with `errorMessage` as a fallback revert reason when `target` reverts.
                                   *
                                   * _Available since v3.1._
                                   */
                                  function functionCallWithValue(address target, bytes memory data, uint256 value, string memory errorMessage) internal returns (bytes memory) {
                                      require(address(this).balance >= value, "Address: insufficient balance for call");
                                      require(isContract(target), "Address: call to non-contract");
                              
                                      // solhint-disable-next-line avoid-low-level-calls
                                      (bool success, bytes memory returndata) = target.call{ value: value }(data);
                                      return _verifyCallResult(success, returndata, errorMessage);
                                  }
                              
                                  /**
                                   * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                                   * but performing a static call.
                                   *
                                   * _Available since v3.3._
                                   */
                                  function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                                      return functionStaticCall(target, data, "Address: low-level static call failed");
                                  }
                              
                                  /**
                                   * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                                   * but performing a static call.
                                   *
                                   * _Available since v3.3._
                                   */
                                  function functionStaticCall(address target, bytes memory data, string memory errorMessage) internal view returns (bytes memory) {
                                      require(isContract(target), "Address: static call to non-contract");
                              
                                      // solhint-disable-next-line avoid-low-level-calls
                                      (bool success, bytes memory returndata) = target.staticcall(data);
                                      return _verifyCallResult(success, returndata, errorMessage);
                                  }
                              
                                  /**
                                   * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                                   * but performing a delegate call.
                                   *
                                   * _Available since v3.4._
                                   */
                                  function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                                      return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                                  }
                              
                                  /**
                                   * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                                   * but performing a delegate call.
                                   *
                                   * _Available since v3.4._
                                   */
                                  function functionDelegateCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) {
                                      require(isContract(target), "Address: delegate call to non-contract");
                              
                                      // solhint-disable-next-line avoid-low-level-calls
                                      (bool success, bytes memory returndata) = target.delegatecall(data);
                                      return _verifyCallResult(success, returndata, errorMessage);
                                  }
                              
                                  function _verifyCallResult(bool success, bytes memory returndata, string memory errorMessage) private pure returns(bytes memory) {
                                      if (success) {
                                          return returndata;
                                      } else {
                                          // Look for revert reason and bubble it up if present
                                          if (returndata.length > 0) {
                                              // The easiest way to bubble the revert reason is using memory via assembly
                              
                                              // solhint-disable-next-line no-inline-assembly
                                              assembly {
                                                  let returndata_size := mload(returndata)
                                                  revert(add(32, returndata), returndata_size)
                                              }
                                          } else {
                                              revert(errorMessage);
                                          }
                                      }
                                  }
                              }
                              
                              
                              // File @openzeppelin/contracts/token/ERC20/SafeERC20.sol@v3.4.1
                              
                              
                              
                              pragma solidity >=0.6.0 <0.8.0;
                              
                              
                              
                              /**
                               * @title SafeERC20
                               * @dev Wrappers around ERC20 operations that throw on failure (when the token
                               * contract returns false). Tokens that return no value (and instead revert or
                               * throw on failure) are also supported, non-reverting calls are assumed to be
                               * successful.
                               * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
                               * which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
                               */
                              library SafeERC20 {
                                  using SafeMath for uint256;
                                  using Address for address;
                              
                                  function safeTransfer(IERC20 token, address to, uint256 value) internal {
                                      _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
                                  }
                              
                                  function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
                                      _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
                                  }
                              
                                  /**
                                   * @dev Deprecated. This function has issues similar to the ones found in
                                   * {IERC20-approve}, and its usage is discouraged.
                                   *
                                   * Whenever possible, use {safeIncreaseAllowance} and
                                   * {safeDecreaseAllowance} instead.
                                   */
                                  function safeApprove(IERC20 token, address spender, uint256 value) internal {
                                      // safeApprove should only be called when setting an initial allowance,
                                      // or when resetting it to zero. To increase and decrease it, use
                                      // 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
                                      // solhint-disable-next-line max-line-length
                                      require((value == 0) || (token.allowance(address(this), spender) == 0),
                                          "SafeERC20: approve from non-zero to non-zero allowance"
                                      );
                                      _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
                                  }
                              
                                  function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
                                      uint256 newAllowance = token.allowance(address(this), spender).add(value);
                                      _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
                                  }
                              
                                  function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
                                      uint256 newAllowance = token.allowance(address(this), spender).sub(value, "SafeERC20: decreased allowance below zero");
                                      _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
                                  }
                              
                                  /**
                                   * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
                                   * on the return value: the return value is optional (but if data is returned, it must not be false).
                                   * @param token The token targeted by the call.
                                   * @param data The call data (encoded using abi.encode or one of its variants).
                                   */
                                  function _callOptionalReturn(IERC20 token, bytes memory data) private {
                                      // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
                                      // we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that
                                      // the target address contains contract code and also asserts for success in the low-level call.
                              
                                      bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
                                      if (returndata.length > 0) { // Return data is optional
                                          // solhint-disable-next-line max-line-length
                                          require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
                                      }
                                  }
                              }
                              
                              
                              // File contracts/helpers/UniERC20.sol
                              
                              
                              
                              pragma solidity ^0.6.12;
                              
                              
                              library UniERC20 {
                                  using SafeMath for uint256;
                              
                                  IERC20 private constant _ETH_ADDRESS = IERC20(0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE);
                                  IERC20 private constant _ZERO_ADDRESS = IERC20(0);
                              
                                  function isETH(IERC20 token) internal pure returns (bool) {
                                      return (token == _ZERO_ADDRESS || token == _ETH_ADDRESS);
                                  }
                              
                                  function uniBalanceOf(IERC20 token, address account) internal view returns (uint256) {
                                      if (isETH(token)) {
                                          return account.balance;
                                      } else {
                                          return token.balanceOf(account);
                                      }
                                  }
                              
                                  function uniTransfer(IERC20 token, address payable to, uint256 amount) internal {
                                      if (amount > 0) {
                                          if (isETH(token)) {
                                              to.transfer(amount);
                                          } else {
                                              _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, amount));
                                          }
                                      }
                                  }
                              
                                  function uniApprove(IERC20 token, address to, uint256 amount) internal {
                                      require(!isETH(token), "Approve called on ETH");
                              
                                      // solhint-disable-next-line avoid-low-level-calls
                                      (bool success, bytes memory returndata) = address(token).call(abi.encodeWithSelector(token.approve.selector, to, amount));
                              
                                      if (!success || (returndata.length > 0 && !abi.decode(returndata, (bool)))) {
                                          _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, to, 0));
                                          _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, to, amount));
                                      }
                                  }
                              
                                  function _callOptionalReturn(IERC20 token, bytes memory data) private {
                                      // solhint-disable-next-line avoid-low-level-calls
                                      (bool success, bytes memory returndata) = address(token).call(data);
                                      require(success, "low-level call failed");
                              
                                      if (returndata.length > 0) { // Return data is optional
                                          require(abi.decode(returndata, (bool)), "ERC20 operation did not succeed");
                                      }
                                  }
                              }
                              
                              
                              // File contracts/interfaces/IChi.sol
                              
                              
                              
                              pragma solidity ^0.6.12;
                              
                              interface IChi is IERC20 {
                                  function mint(uint256 value) external;
                                  function free(uint256 value) external returns (uint256 freed);
                                  function freeFromUpTo(address from, uint256 value) external returns (uint256 freed);
                              }
                              
                              
                              // File contracts/interfaces/IGasDiscountExtension.sol
                              
                              
                              
                              pragma solidity ^0.6.12;
                              
                              interface IGasDiscountExtension {
                                  function calculateGas(uint256 gasUsed, uint256 flags, uint256 calldataLength) external view returns (IChi, uint256);
                              }
                              
                              
                              // File contracts/interfaces/IAggregationExecutor.sol
                              
                              
                              
                              pragma solidity ^0.6.12;
                              
                              interface IAggregationExecutor is IGasDiscountExtension {
                                  function callBytes(bytes calldata data) external payable;  // 0xd9c45357
                              }
                              
                              
                              // File contracts/helpers/RevertReasonParser.sol
                              
                              
                              
                              pragma solidity ^0.6.12;
                              
                              
                              library RevertReasonParser {
                                  function parse(bytes memory data, string memory prefix) internal pure returns (string memory) {
                                      // https://solidity.readthedocs.io/en/latest/control-structures.html#revert
                                      // We assume that revert reason is abi-encoded as Error(string)
                              
                                      // 68 = 4-byte selector 0x08c379a0 + 32 bytes offset + 32 bytes length
                                      if (data.length >= 68 && data[0] == "\x08" && data[1] == "\xc3" && data[2] == "\x79" && data[3] == "\xa0") {
                                          string memory reason;
                                          // solhint-disable no-inline-assembly
                                          assembly {
                                              // 68 = 32 bytes data length + 4-byte selector + 32 bytes offset
                                              reason := add(data, 68)
                                          }
                                          /*
                                              revert reason is padded up to 32 bytes with ABI encoder: Error(string)
                                              also sometimes there is extra 32 bytes of zeros padded in the end:
                                              https://github.com/ethereum/solidity/issues/10170
                                              because of that we can't check for equality and instead check
                                              that string length + extra 68 bytes is less than overall data length
                                          */
                                          require(data.length >= 68 + bytes(reason).length, "Invalid revert reason");
                                          return string(abi.encodePacked(prefix, "Error(", reason, ")"));
                                      }
                                      // 36 = 4-byte selector 0x4e487b71 + 32 bytes integer
                                      else if (data.length == 36 && data[0] == "\x4e" && data[1] == "\x48" && data[2] == "\x7b" && data[3] == "\x71") {
                                          uint256 code;
                                          // solhint-disable no-inline-assembly
                                          assembly {
                                              // 36 = 32 bytes data length + 4-byte selector
                                              code := mload(add(data, 36))
                                          }
                                          return string(abi.encodePacked(prefix, "Panic(", _toHex(code), ")"));
                                      }
                              
                                      return string(abi.encodePacked(prefix, "Unknown(", _toHex(data), ")"));
                                  }
                              
                                  function _toHex(uint256 value) private pure returns(string memory) {
                                      return _toHex(abi.encodePacked(value));
                                  }
                              
                                  function _toHex(bytes memory data) private pure returns(string memory) {
                                      bytes16 alphabet = 0x30313233343536373839616263646566;
                                      bytes memory str = new bytes(2 + data.length * 2);
                                      str[0] = "0";
                                      str[1] = "x";
                                      for (uint256 i = 0; i < data.length; i++) {
                                          str[2 * i + 2] = alphabet[uint8(data[i] >> 4)];
                                          str[2 * i + 3] = alphabet[uint8(data[i] & 0x0f)];
                                      }
                                      return string(str);
                                  }
                              }
                              
                              
                              // File contracts/interfaces/IERC20Permit.sol
                              
                              
                              
                              pragma solidity ^0.6.12;
                              
                              
                              interface IERC20Permit {
                                  function permit(address owner, address spender, uint256 amount, uint256 deadline, uint8 v, bytes32 r, bytes32 s) external;
                              }
                              
                              
                              // File contracts/helpers/Permitable.sol
                              
                              
                              
                              pragma solidity ^0.6.12;
                              
                              
                              
                              contract Permitable {
                                  event Error(
                                      string reason
                                  );
                              
                                  function _permit(IERC20 token, uint256 amount, bytes calldata permit) internal {
                                      if (permit.length == 32 * 7) {
                                          // solhint-disable-next-line avoid-low-level-calls
                                          (bool success, bytes memory result) = address(token).call(abi.encodePacked(IERC20Permit.permit.selector, permit));
                                          if (!success) {
                                              string memory reason = RevertReasonParser.parse(result, "Permit call failed: ");
                                              if (token.allowance(msg.sender, address(this)) < amount) {
                                                  revert(reason);
                                              } else {
                                                  emit Error(reason);
                                              }
                                          }
                                      }
                                  }
                              }
                              
                              
                              // File contracts/UnoswapRouter.sol
                              
                              
                              
                              pragma solidity ^0.6.12;
                              
                              contract UnoswapRouter is Permitable {
                                  uint256 private constant _TRANSFER_FROM_CALL_SELECTOR_32 = 0x23b872dd00000000000000000000000000000000000000000000000000000000;
                                  uint256 private constant _WETH_DEPOSIT_CALL_SELECTOR_32 = 0xd0e30db000000000000000000000000000000000000000000000000000000000;
                                  uint256 private constant _WETH_WITHDRAW_CALL_SELECTOR_32 = 0x2e1a7d4d00000000000000000000000000000000000000000000000000000000;
                                  uint256 private constant _ERC20_TRANSFER_CALL_SELECTOR_32 = 0xa9059cbb00000000000000000000000000000000000000000000000000000000;
                                  uint256 private constant _ADDRESS_MASK =   0x000000000000000000000000ffffffffffffffffffffffffffffffffffffffff;
                                  uint256 private constant _REVERSE_MASK =   0x8000000000000000000000000000000000000000000000000000000000000000;
                                  uint256 private constant _WETH_MASK =      0x4000000000000000000000000000000000000000000000000000000000000000;
                                  uint256 private constant _NUMERATOR_MASK = 0x0000000000000000ffffffff0000000000000000000000000000000000000000;
                                  uint256 private constant _WETH = 0x000000000000000000000000C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2;
                                  uint256 private constant _UNISWAP_PAIR_RESERVES_CALL_SELECTOR_32 = 0x0902f1ac00000000000000000000000000000000000000000000000000000000;
                                  uint256 private constant _UNISWAP_PAIR_SWAP_CALL_SELECTOR_32 = 0x022c0d9f00000000000000000000000000000000000000000000000000000000;
                                  uint256 private constant _DENOMINATOR = 1000000000;
                                  uint256 private constant _NUMERATOR_OFFSET = 160;
                              
                                  receive() external payable {
                                      // solhint-disable-next-line avoid-tx-origin
                                      require(msg.sender != tx.origin, "ETH deposit rejected");
                                  }
                              
                                  function unoswapWithPermit(
                                      IERC20 srcToken,
                                      uint256 amount,
                                      uint256 minReturn,
                                      bytes32[] calldata pools,
                                      bytes calldata permit
                                  ) external payable returns(uint256 returnAmount) {
                                      _permit(srcToken, amount, permit);
                                      return unoswap(srcToken, amount, minReturn, pools);
                                  }
                              
                                  function unoswap(
                                      IERC20 srcToken,
                                      uint256 amount,
                                      uint256 minReturn,
                                      bytes32[] calldata /* pools */
                                  ) public payable returns(uint256 returnAmount) {
                                      assembly {  // solhint-disable-line no-inline-assembly
                                          function reRevert() {
                                              returndatacopy(0, 0, returndatasize())
                                              revert(0, returndatasize())
                                          }
                              
                                          function revertWithReason(m, len) {
                                              mstore(0, 0x08c379a000000000000000000000000000000000000000000000000000000000)
                                              mstore(0x20, 0x0000002000000000000000000000000000000000000000000000000000000000)
                                              mstore(0x40, m)
                                              revert(0, len)
                                          }
                              
                                          function swap(emptyPtr, swapAmount, pair, reversed, numerator, dst) -> ret {
                                              mstore(emptyPtr, _UNISWAP_PAIR_RESERVES_CALL_SELECTOR_32)
                                              if iszero(staticcall(gas(), pair, emptyPtr, 0x4, emptyPtr, 0x40)) {
                                                  reRevert()
                                              }
                              
                                              let reserve0 := mload(emptyPtr)
                                              let reserve1 := mload(add(emptyPtr, 0x20))
                                              if reversed {
                                                  let tmp := reserve0
                                                  reserve0 := reserve1
                                                  reserve1 := tmp
                                              }
                                              ret := mul(swapAmount, numerator)
                                              ret := div(mul(ret, reserve1), add(ret, mul(reserve0, _DENOMINATOR)))
                              
                                              mstore(emptyPtr, _UNISWAP_PAIR_SWAP_CALL_SELECTOR_32)
                                              switch reversed
                                              case 0 {
                                                  mstore(add(emptyPtr, 0x04), 0)
                                                  mstore(add(emptyPtr, 0x24), ret)
                                              }
                                              default {
                                                  mstore(add(emptyPtr, 0x04), ret)
                                                  mstore(add(emptyPtr, 0x24), 0)
                                              }
                                              mstore(add(emptyPtr, 0x44), dst)
                                              mstore(add(emptyPtr, 0x64), 0x80)
                                              mstore(add(emptyPtr, 0x84), 0)
                                              if iszero(call(gas(), pair, 0, emptyPtr, 0xa4, 0, 0)) {
                                                  reRevert()
                                              }
                                          }
                              
                                          let emptyPtr := mload(0x40)
                                          mstore(0x40, add(emptyPtr, 0xc0))
                              
                                          let poolsOffset := add(calldataload(0x64), 0x4)
                                          let poolsEndOffset := calldataload(poolsOffset)
                                          poolsOffset := add(poolsOffset, 0x20)
                                          poolsEndOffset := add(poolsOffset, mul(0x20, poolsEndOffset))
                                          let rawPair := calldataload(poolsOffset)
                                          switch srcToken
                                          case 0 {
                                              if iszero(eq(amount, callvalue())) {
                                                  revertWithReason(0x00000011696e76616c6964206d73672e76616c75650000000000000000000000, 0x55)  // "invalid msg.value"
                                              }
                              
                                              mstore(emptyPtr, _WETH_DEPOSIT_CALL_SELECTOR_32)
                                              if iszero(call(gas(), _WETH, amount, emptyPtr, 0x4, 0, 0)) {
                                                  reRevert()
                                              }
                              
                                              mstore(emptyPtr, _ERC20_TRANSFER_CALL_SELECTOR_32)
                                              mstore(add(emptyPtr, 0x4), and(rawPair, _ADDRESS_MASK))
                                              mstore(add(emptyPtr, 0x24), amount)
                                              if iszero(call(gas(), _WETH, 0, emptyPtr, 0x44, 0, 0)) {
                                                  reRevert()
                                              }
                                          }
                                          default {
                                              if callvalue() {
                                                  revertWithReason(0x00000011696e76616c6964206d73672e76616c75650000000000000000000000, 0x55)  // "invalid msg.value"
                                              }
                              
                                              mstore(emptyPtr, _TRANSFER_FROM_CALL_SELECTOR_32)
                                              mstore(add(emptyPtr, 0x4), caller())
                                              mstore(add(emptyPtr, 0x24), and(rawPair, _ADDRESS_MASK))
                                              mstore(add(emptyPtr, 0x44), amount)
                                              if iszero(call(gas(), srcToken, 0, emptyPtr, 0x64, 0, 0)) {
                                                  reRevert()
                                              }
                                          }
                              
                                          returnAmount := amount
                              
                                          for {let i := add(poolsOffset, 0x20)} lt(i, poolsEndOffset) {i := add(i, 0x20)} {
                                              let nextRawPair := calldataload(i)
                              
                                              returnAmount := swap(
                                                  emptyPtr,
                                                  returnAmount,
                                                  and(rawPair, _ADDRESS_MASK),
                                                  and(rawPair, _REVERSE_MASK),
                                                  shr(_NUMERATOR_OFFSET, and(rawPair, _NUMERATOR_MASK)),
                                                  and(nextRawPair, _ADDRESS_MASK)
                                              )
                              
                                              rawPair := nextRawPair
                                          }
                              
                                          switch and(rawPair, _WETH_MASK)
                                          case 0 {
                                              returnAmount := swap(
                                                  emptyPtr,
                                                  returnAmount,
                                                  and(rawPair, _ADDRESS_MASK),
                                                  and(rawPair, _REVERSE_MASK),
                                                  shr(_NUMERATOR_OFFSET, and(rawPair, _NUMERATOR_MASK)),
                                                  caller()
                                              )
                                          }
                                          default {
                                              returnAmount := swap(
                                                  emptyPtr,
                                                  returnAmount,
                                                  and(rawPair, _ADDRESS_MASK),
                                                  and(rawPair, _REVERSE_MASK),
                                                  shr(_NUMERATOR_OFFSET, and(rawPair, _NUMERATOR_MASK)),
                                                  address()
                                              )
                              
                                              mstore(emptyPtr, _WETH_WITHDRAW_CALL_SELECTOR_32)
                                              mstore(add(emptyPtr, 0x04), returnAmount)
                                              if iszero(call(gas(), _WETH, 0, emptyPtr, 0x24, 0, 0)) {
                                                  reRevert()
                                              }
                              
                                              if iszero(call(gas(), caller(), returnAmount, 0, 0, 0, 0)) {
                                                  reRevert()
                                              }
                                          }
                              
                                          if lt(returnAmount, minReturn) {
                                              revertWithReason(0x000000164d696e2072657475726e206e6f742072656163686564000000000000, 0x5a)  // "Min return not reached"
                                          }
                                      }
                                  }
                              }
                              
                              
                              // File contracts/AggregationRouterV3.sol
                              
                              
                              
                              pragma solidity ^0.6.12;
                              pragma experimental ABIEncoderV2;
                              
                              
                              
                              
                              
                              
                              contract AggregationRouterV3 is Ownable, UnoswapRouter {
                                  using SafeMath for uint256;
                                  using SafeERC20 for IERC20;
                                  using UniERC20 for IERC20;
                              
                                  uint256 private constant _PARTIAL_FILL = 0x01;
                                  uint256 private constant _REQUIRES_EXTRA_ETH = 0x02;
                                  uint256 private constant _SHOULD_CLAIM = 0x04;
                                  uint256 private constant _BURN_FROM_MSG_SENDER = 0x08;
                                  uint256 private constant _BURN_FROM_TX_ORIGIN = 0x10;
                              
                                  struct SwapDescription {
                                      IERC20 srcToken;
                                      IERC20 dstToken;
                                      address srcReceiver;
                                      address dstReceiver;
                                      uint256 amount;
                                      uint256 minReturnAmount;
                                      uint256 flags;
                                      bytes permit;
                                  }
                              
                                  event Swapped(
                                      address sender,
                                      IERC20 srcToken,
                                      IERC20 dstToken,
                                      address dstReceiver,
                                      uint256 spentAmount,
                                      uint256 returnAmount
                                  );
                              
                                  function discountedSwap(
                                      IAggregationExecutor caller,
                                      SwapDescription calldata desc,
                                      bytes calldata data
                                  )
                                      external
                                      payable
                                      returns (uint256 returnAmount, uint256 gasLeft, uint256 chiSpent)
                                  {
                                      uint256 initialGas = gasleft();
                              
                                      address chiSource = address(0);
                                      if (desc.flags & _BURN_FROM_MSG_SENDER != 0) {
                                          chiSource = msg.sender;
                                      } else if (desc.flags & _BURN_FROM_TX_ORIGIN != 0) {
                                          chiSource = tx.origin; // solhint-disable-line avoid-tx-origin
                                      } else {
                                          revert("Incorrect CHI burn flags");
                                      }
                              
                                      // solhint-disable-next-line avoid-low-level-calls
                                      (bool success, bytes memory returnData) = address(this).delegatecall(abi.encodeWithSelector(this.swap.selector, caller, desc, data));
                                      if (success) {
                                          (returnAmount,) = abi.decode(returnData, (uint256, uint256));
                                      } else {
                                          if (msg.value > 0) {
                                              msg.sender.transfer(msg.value);
                                          }
                                          emit Error(RevertReasonParser.parse(returnData, "Swap failed: "));
                                      }
                              
                                      (IChi chi, uint256 amount) = caller.calculateGas(initialGas.sub(gasleft()), desc.flags, msg.data.length);
                                      if (amount > 0) {
                                          chiSpent = chi.freeFromUpTo(chiSource, amount);
                                      }
                                      gasLeft = gasleft();
                                  }
                              
                                  function swap(
                                      IAggregationExecutor caller,
                                      SwapDescription calldata desc,
                                      bytes calldata data
                                  )
                                      external
                                      payable
                                      returns (uint256 returnAmount, uint256 gasLeft)
                                  {
                                      require(desc.minReturnAmount > 0, "Min return should not be 0");
                                      require(data.length > 0, "data should be not zero");
                              
                                      uint256 flags = desc.flags;
                                      IERC20 srcToken = desc.srcToken;
                                      IERC20 dstToken = desc.dstToken;
                              
                                      if (flags & _REQUIRES_EXTRA_ETH != 0) {
                                          require(msg.value > (srcToken.isETH() ? desc.amount : 0), "Invalid msg.value");
                                      } else {
                                          require(msg.value == (srcToken.isETH() ? desc.amount : 0), "Invalid msg.value");
                                      }
                              
                                      if (flags & _SHOULD_CLAIM != 0) {
                                          require(!srcToken.isETH(), "Claim token is ETH");
                                          _permit(srcToken, desc.amount, desc.permit);
                                          srcToken.safeTransferFrom(msg.sender, desc.srcReceiver, desc.amount);
                                      }
                              
                                      address dstReceiver = (desc.dstReceiver == address(0)) ? msg.sender : desc.dstReceiver;
                                      uint256 initialSrcBalance = (flags & _PARTIAL_FILL != 0) ? srcToken.uniBalanceOf(msg.sender) : 0;
                                      uint256 initialDstBalance = dstToken.uniBalanceOf(dstReceiver);
                              
                                      {
                                          // solhint-disable-next-line avoid-low-level-calls
                                          (bool success, bytes memory result) = address(caller).call{value: msg.value}(abi.encodePacked(caller.callBytes.selector, data));
                                          if (!success) {
                                              revert(RevertReasonParser.parse(result, "callBytes failed: "));
                                          }
                                      }
                              
                                      uint256 spentAmount = desc.amount;
                                      returnAmount = dstToken.uniBalanceOf(dstReceiver).sub(initialDstBalance);
                              
                                      if (flags & _PARTIAL_FILL != 0) {
                                          spentAmount = initialSrcBalance.add(desc.amount).sub(srcToken.uniBalanceOf(msg.sender));
                                          require(returnAmount.mul(desc.amount) >= desc.minReturnAmount.mul(spentAmount), "Return amount is not enough");
                                      } else {
                                          require(returnAmount >= desc.minReturnAmount, "Return amount is not enough");
                                      }
                              
                                      emit Swapped(
                                          msg.sender,
                                          srcToken,
                                          dstToken,
                                          dstReceiver,
                                          spentAmount,
                                          returnAmount
                                      );
                              
                                      gasLeft = gasleft();
                                  }
                              
                                  function rescueFunds(IERC20 token, uint256 amount) external onlyOwner {
                                      token.uniTransfer(msg.sender, amount);
                                  }
                              
                                  function destroy() external onlyOwner {
                                      selfdestruct(msg.sender);
                                  }
                              }

                              File 2 of 5: MaticToken
                              pragma solidity 0.5.2;
                              
                              // File: openzeppelin-solidity/contracts/token/ERC20/IERC20.sol
                              
                              /**
                               * @title ERC20 interface
                               * @dev see https://github.com/ethereum/EIPs/issues/20
                               */
                              interface IERC20 {
                                  function transfer(address to, uint256 value) external returns (bool);
                              
                                  function approve(address spender, uint256 value) external returns (bool);
                              
                                  function transferFrom(address from, address to, uint256 value) external returns (bool);
                              
                                  function totalSupply() external view returns (uint256);
                              
                                  function balanceOf(address who) external view returns (uint256);
                              
                                  function allowance(address owner, address spender) external view returns (uint256);
                              
                                  event Transfer(address indexed from, address indexed to, uint256 value);
                              
                                  event Approval(address indexed owner, address indexed spender, uint256 value);
                              }
                              
                              // File: openzeppelin-solidity/contracts/math/SafeMath.sol
                              
                              /**
                               * @title SafeMath
                               * @dev Unsigned math operations with safety checks that revert on error
                               */
                              library SafeMath {
                                  /**
                                  * @dev Multiplies two unsigned integers, reverts on overflow.
                                  */
                                  function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                                      // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                                      // benefit is lost if 'b' is also tested.
                                      // See: https://github.com/OpenZeppelin/openzeppelin-solidity/pull/522
                                      if (a == 0) {
                                          return 0;
                                      }
                              
                                      uint256 c = a * b;
                                      require(c / a == b);
                              
                                      return c;
                                  }
                              
                                  /**
                                  * @dev Integer division of two unsigned integers truncating the quotient, reverts on division by zero.
                                  */
                                  function div(uint256 a, uint256 b) internal pure returns (uint256) {
                                      // Solidity only automatically asserts when dividing by 0
                                      require(b > 0);
                                      uint256 c = a / b;
                                      // assert(a == b * c + a % b); // There is no case in which this doesn't hold
                              
                                      return c;
                                  }
                              
                                  /**
                                  * @dev Subtracts two unsigned integers, reverts on overflow (i.e. if subtrahend is greater than minuend).
                                  */
                                  function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                                      require(b <= a);
                                      uint256 c = a - b;
                              
                                      return c;
                                  }
                              
                                  /**
                                  * @dev Adds two unsigned integers, reverts on overflow.
                                  */
                                  function add(uint256 a, uint256 b) internal pure returns (uint256) {
                                      uint256 c = a + b;
                                      require(c >= a);
                              
                                      return c;
                                  }
                              
                                  /**
                                  * @dev Divides two unsigned integers and returns the remainder (unsigned integer modulo),
                                  * reverts when dividing by zero.
                                  */
                                  function mod(uint256 a, uint256 b) internal pure returns (uint256) {
                                      require(b != 0);
                                      return a % b;
                                  }
                              }
                              
                              // File: openzeppelin-solidity/contracts/token/ERC20/ERC20.sol
                              
                              /**
                               * @title Standard ERC20 token
                               *
                               * @dev Implementation of the basic standard token.
                               * https://github.com/ethereum/EIPs/blob/master/EIPS/eip-20.md
                               * Originally based on code by FirstBlood:
                               * https://github.com/Firstbloodio/token/blob/master/smart_contract/FirstBloodToken.sol
                               *
                               * This implementation emits additional Approval events, allowing applications to reconstruct the allowance status for
                               * all accounts just by listening to said events. Note that this isn't required by the specification, and other
                               * compliant implementations may not do it.
                               */
                              contract ERC20 is IERC20 {
                                  using SafeMath for uint256;
                              
                                  mapping (address => uint256) private _balances;
                              
                                  mapping (address => mapping (address => uint256)) private _allowed;
                              
                                  uint256 private _totalSupply;
                              
                                  /**
                                  * @dev Total number of tokens in existence
                                  */
                                  function totalSupply() public view returns (uint256) {
                                      return _totalSupply;
                                  }
                              
                                  /**
                                  * @dev Gets the balance of the specified address.
                                  * @param owner The address to query the balance of.
                                  * @return An uint256 representing the amount owned by the passed address.
                                  */
                                  function balanceOf(address owner) public view returns (uint256) {
                                      return _balances[owner];
                                  }
                              
                                  /**
                                   * @dev Function to check the amount of tokens that an owner allowed to a spender.
                                   * @param owner address The address which owns the funds.
                                   * @param spender address The address which will spend the funds.
                                   * @return A uint256 specifying the amount of tokens still available for the spender.
                                   */
                                  function allowance(address owner, address spender) public view returns (uint256) {
                                      return _allowed[owner][spender];
                                  }
                              
                                  /**
                                  * @dev Transfer token for a specified address
                                  * @param to The address to transfer to.
                                  * @param value The amount to be transferred.
                                  */
                                  function transfer(address to, uint256 value) public returns (bool) {
                                      _transfer(msg.sender, to, value);
                                      return true;
                                  }
                              
                                  /**
                                   * @dev Approve the passed address to spend the specified amount of tokens on behalf of msg.sender.
                                   * Beware that changing an allowance with this method brings the risk that someone may use both the old
                                   * and the new allowance by unfortunate transaction ordering. One possible solution to mitigate this
                                   * race condition is to first reduce the spender's allowance to 0 and set the desired value afterwards:
                                   * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                                   * @param spender The address which will spend the funds.
                                   * @param value The amount of tokens to be spent.
                                   */
                                  function approve(address spender, uint256 value) public returns (bool) {
                                      require(spender != address(0));
                              
                                      _allowed[msg.sender][spender] = value;
                                      emit Approval(msg.sender, spender, value);
                                      return true;
                                  }
                              
                                  /**
                                   * @dev Transfer tokens from one address to another.
                                   * Note that while this function emits an Approval event, this is not required as per the specification,
                                   * and other compliant implementations may not emit the event.
                                   * @param from address The address which you want to send tokens from
                                   * @param to address The address which you want to transfer to
                                   * @param value uint256 the amount of tokens to be transferred
                                   */
                                  function transferFrom(address from, address to, uint256 value) public returns (bool) {
                                      _allowed[from][msg.sender] = _allowed[from][msg.sender].sub(value);
                                      _transfer(from, to, value);
                                      emit Approval(from, msg.sender, _allowed[from][msg.sender]);
                                      return true;
                                  }
                              
                                  /**
                                   * @dev Increase the amount of tokens that an owner allowed to a spender.
                                   * approve should be called when allowed_[_spender] == 0. To increment
                                   * allowed value is better to use this function to avoid 2 calls (and wait until
                                   * the first transaction is mined)
                                   * From MonolithDAO Token.sol
                                   * Emits an Approval event.
                                   * @param spender The address which will spend the funds.
                                   * @param addedValue The amount of tokens to increase the allowance by.
                                   */
                                  function increaseAllowance(address spender, uint256 addedValue) public returns (bool) {
                                      require(spender != address(0));
                              
                                      _allowed[msg.sender][spender] = _allowed[msg.sender][spender].add(addedValue);
                                      emit Approval(msg.sender, spender, _allowed[msg.sender][spender]);
                                      return true;
                                  }
                              
                                  /**
                                   * @dev Decrease the amount of tokens that an owner allowed to a spender.
                                   * approve should be called when allowed_[_spender] == 0. To decrement
                                   * allowed value is better to use this function to avoid 2 calls (and wait until
                                   * the first transaction is mined)
                                   * From MonolithDAO Token.sol
                                   * Emits an Approval event.
                                   * @param spender The address which will spend the funds.
                                   * @param subtractedValue The amount of tokens to decrease the allowance by.
                                   */
                                  function decreaseAllowance(address spender, uint256 subtractedValue) public returns (bool) {
                                      require(spender != address(0));
                              
                                      _allowed[msg.sender][spender] = _allowed[msg.sender][spender].sub(subtractedValue);
                                      emit Approval(msg.sender, spender, _allowed[msg.sender][spender]);
                                      return true;
                                  }
                              
                                  /**
                                  * @dev Transfer token for a specified addresses
                                  * @param from The address to transfer from.
                                  * @param to The address to transfer to.
                                  * @param value The amount to be transferred.
                                  */
                                  function _transfer(address from, address to, uint256 value) internal {
                                      require(to != address(0));
                              
                                      _balances[from] = _balances[from].sub(value);
                                      _balances[to] = _balances[to].add(value);
                                      emit Transfer(from, to, value);
                                  }
                              
                                  /**
                                   * @dev Internal function that mints an amount of the token and assigns it to
                                   * an account. This encapsulates the modification of balances such that the
                                   * proper events are emitted.
                                   * @param account The account that will receive the created tokens.
                                   * @param value The amount that will be created.
                                   */
                                  function _mint(address account, uint256 value) internal {
                                      require(account != address(0));
                              
                                      _totalSupply = _totalSupply.add(value);
                                      _balances[account] = _balances[account].add(value);
                                      emit Transfer(address(0), account, value);
                                  }
                              
                                  /**
                                   * @dev Internal function that burns an amount of the token of a given
                                   * account.
                                   * @param account The account whose tokens will be burnt.
                                   * @param value The amount that will be burnt.
                                   */
                                  function _burn(address account, uint256 value) internal {
                                      require(account != address(0));
                              
                                      _totalSupply = _totalSupply.sub(value);
                                      _balances[account] = _balances[account].sub(value);
                                      emit Transfer(account, address(0), value);
                                  }
                              
                                  /**
                                   * @dev Internal function that burns an amount of the token of a given
                                   * account, deducting from the sender's allowance for said account. Uses the
                                   * internal burn function.
                                   * Emits an Approval event (reflecting the reduced allowance).
                                   * @param account The account whose tokens will be burnt.
                                   * @param value The amount that will be burnt.
                                   */
                                  function _burnFrom(address account, uint256 value) internal {
                                      _allowed[account][msg.sender] = _allowed[account][msg.sender].sub(value);
                                      _burn(account, value);
                                      emit Approval(account, msg.sender, _allowed[account][msg.sender]);
                                  }
                              }
                              
                              // File: openzeppelin-solidity/contracts/access/Roles.sol
                              
                              /**
                               * @title Roles
                               * @dev Library for managing addresses assigned to a Role.
                               */
                              library Roles {
                                  struct Role {
                                      mapping (address => bool) bearer;
                                  }
                              
                                  /**
                                   * @dev give an account access to this role
                                   */
                                  function add(Role storage role, address account) internal {
                                      require(account != address(0));
                                      require(!has(role, account));
                              
                                      role.bearer[account] = true;
                                  }
                              
                                  /**
                                   * @dev remove an account's access to this role
                                   */
                                  function remove(Role storage role, address account) internal {
                                      require(account != address(0));
                                      require(has(role, account));
                              
                                      role.bearer[account] = false;
                                  }
                              
                                  /**
                                   * @dev check if an account has this role
                                   * @return bool
                                   */
                                  function has(Role storage role, address account) internal view returns (bool) {
                                      require(account != address(0));
                                      return role.bearer[account];
                                  }
                              }
                              
                              // File: openzeppelin-solidity/contracts/access/roles/PauserRole.sol
                              
                              contract PauserRole {
                                  using Roles for Roles.Role;
                              
                                  event PauserAdded(address indexed account);
                                  event PauserRemoved(address indexed account);
                              
                                  Roles.Role private _pausers;
                              
                                  constructor () internal {
                                      _addPauser(msg.sender);
                                  }
                              
                                  modifier onlyPauser() {
                                      require(isPauser(msg.sender));
                                      _;
                                  }
                              
                                  function isPauser(address account) public view returns (bool) {
                                      return _pausers.has(account);
                                  }
                              
                                  function addPauser(address account) public onlyPauser {
                                      _addPauser(account);
                                  }
                              
                                  function renouncePauser() public {
                                      _removePauser(msg.sender);
                                  }
                              
                                  function _addPauser(address account) internal {
                                      _pausers.add(account);
                                      emit PauserAdded(account);
                                  }
                              
                                  function _removePauser(address account) internal {
                                      _pausers.remove(account);
                                      emit PauserRemoved(account);
                                  }
                              }
                              
                              // File: openzeppelin-solidity/contracts/lifecycle/Pausable.sol
                              
                              /**
                               * @title Pausable
                               * @dev Base contract which allows children to implement an emergency stop mechanism.
                               */
                              contract Pausable is PauserRole {
                                  event Paused(address account);
                                  event Unpaused(address account);
                              
                                  bool private _paused;
                              
                                  constructor () internal {
                                      _paused = false;
                                  }
                              
                                  /**
                                   * @return true if the contract is paused, false otherwise.
                                   */
                                  function paused() public view returns (bool) {
                                      return _paused;
                                  }
                              
                                  /**
                                   * @dev Modifier to make a function callable only when the contract is not paused.
                                   */
                                  modifier whenNotPaused() {
                                      require(!_paused);
                                      _;
                                  }
                              
                                  /**
                                   * @dev Modifier to make a function callable only when the contract is paused.
                                   */
                                  modifier whenPaused() {
                                      require(_paused);
                                      _;
                                  }
                              
                                  /**
                                   * @dev called by the owner to pause, triggers stopped state
                                   */
                                  function pause() public onlyPauser whenNotPaused {
                                      _paused = true;
                                      emit Paused(msg.sender);
                                  }
                              
                                  /**
                                   * @dev called by the owner to unpause, returns to normal state
                                   */
                                  function unpause() public onlyPauser whenPaused {
                                      _paused = false;
                                      emit Unpaused(msg.sender);
                                  }
                              }
                              
                              // File: openzeppelin-solidity/contracts/token/ERC20/ERC20Pausable.sol
                              
                              /**
                               * @title Pausable token
                               * @dev ERC20 modified with pausable transfers.
                               **/
                              contract ERC20Pausable is ERC20, Pausable {
                                  function transfer(address to, uint256 value) public whenNotPaused returns (bool) {
                                      return super.transfer(to, value);
                                  }
                              
                                  function transferFrom(address from, address to, uint256 value) public whenNotPaused returns (bool) {
                                      return super.transferFrom(from, to, value);
                                  }
                              
                                  function approve(address spender, uint256 value) public whenNotPaused returns (bool) {
                                      return super.approve(spender, value);
                                  }
                              
                                  function increaseAllowance(address spender, uint addedValue) public whenNotPaused returns (bool success) {
                                      return super.increaseAllowance(spender, addedValue);
                                  }
                              
                                  function decreaseAllowance(address spender, uint subtractedValue) public whenNotPaused returns (bool success) {
                                      return super.decreaseAllowance(spender, subtractedValue);
                                  }
                              }
                              
                              // File: openzeppelin-solidity/contracts/token/ERC20/ERC20Detailed.sol
                              
                              /**
                               * @title ERC20Detailed token
                               * @dev The decimals are only for visualization purposes.
                               * All the operations are done using the smallest and indivisible token unit,
                               * just as on Ethereum all the operations are done in wei.
                               */
                              contract ERC20Detailed is IERC20 {
                                  string private _name;
                                  string private _symbol;
                                  uint8 private _decimals;
                              
                                  constructor (string memory name, string memory symbol, uint8 decimals) public {
                                      _name = name;
                                      _symbol = symbol;
                                      _decimals = decimals;
                                  }
                              
                                  /**
                                   * @return the name of the token.
                                   */
                                  function name() public view returns (string memory) {
                                      return _name;
                                  }
                              
                                  /**
                                   * @return the symbol of the token.
                                   */
                                  function symbol() public view returns (string memory) {
                                      return _symbol;
                                  }
                              
                                  /**
                                   * @return the number of decimals of the token.
                                   */
                                  function decimals() public view returns (uint8) {
                                      return _decimals;
                                  }
                              }
                              
                              // File: contracts/MaticToken.sol
                              
                              contract MaticToken is ERC20Pausable, ERC20Detailed {
                                  constructor (string memory name, string memory symbol, uint8 decimals, uint256 totalSupply)
                                  public
                                  ERC20Detailed (name, symbol, decimals) {
                                      _mint(msg.sender, totalSupply);
                                  }
                              }

                              File 3 of 5: BPool
                              {"BColor.sol":{"content":"// This program is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n\n// This program is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n// GNU General Public License for more details.\n\n// You should have received a copy of the GNU General Public License\n// along with this program.  If not, see \u003chttp://www.gnu.org/licenses/\u003e.\n\npragma solidity 0.5.12;\n\ncontract BColor {\n    function getColor()\n        external view\n        returns (bytes32);\n}\n\ncontract BBronze is BColor {\n    function getColor()\n        external view\n        returns (bytes32) {\n            return bytes32(\"BRONZE\");\n        }\n}\n"},"BConst.sol":{"content":"// This program is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n\n// This program is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n// GNU General Public License for more details.\n\n// You should have received a copy of the GNU General Public License\n// along with this program.  If not, see \u003chttp://www.gnu.org/licenses/\u003e.\n\npragma solidity 0.5.12;\n\nimport \"./BColor.sol\";\n\ncontract BConst is BBronze {\n    uint public constant BONE              = 10**18;\n\n    uint public constant MIN_BOUND_TOKENS  = 2;\n    uint public constant MAX_BOUND_TOKENS  = 8;\n\n    uint public constant MIN_FEE           = BONE / 10**6;\n    uint public constant MAX_FEE           = BONE / 10;\n    uint public constant EXIT_FEE          = 0;\n\n    uint public constant MIN_WEIGHT        = BONE;\n    uint public constant MAX_WEIGHT        = BONE * 50;\n    uint public constant MAX_TOTAL_WEIGHT  = BONE * 50;\n    uint public constant MIN_BALANCE       = BONE / 10**12;\n\n    uint public constant INIT_POOL_SUPPLY  = BONE * 100;\n\n    uint public constant MIN_BPOW_BASE     = 1 wei;\n    uint public constant MAX_BPOW_BASE     = (2 * BONE) - 1 wei;\n    uint public constant BPOW_PRECISION    = BONE / 10**10;\n\n    uint public constant MAX_IN_RATIO      = BONE / 2;\n    uint public constant MAX_OUT_RATIO     = (BONE / 3) + 1 wei;\n}\n"},"BMath.sol":{"content":"// This program is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n\n// This program is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n// GNU General Public License for more details.\n\n// You should have received a copy of the GNU General Public License\n// along with this program.  If not, see \u003chttp://www.gnu.org/licenses/\u003e.\n\npragma solidity 0.5.12;\n\nimport \"./BNum.sol\";\n\ncontract BMath is BBronze, BConst, BNum {\n    /**********************************************************************************************\n    // calcSpotPrice                                                                             //\n    // sP = spotPrice                                                                            //\n    // bI = tokenBalanceIn                ( bI / wI )         1                                  //\n    // bO = tokenBalanceOut         sP =  -----------  *  ----------                             //\n    // wI = tokenWeightIn                 ( bO / wO )     ( 1 - sF )                             //\n    // wO = tokenWeightOut                                                                       //\n    // sF = swapFee                                                                              //\n    **********************************************************************************************/\n    function calcSpotPrice(\n        uint tokenBalanceIn,\n        uint tokenWeightIn,\n        uint tokenBalanceOut,\n        uint tokenWeightOut,\n        uint swapFee\n    )\n        public pure\n        returns (uint spotPrice)\n    {\n        uint numer = bdiv(tokenBalanceIn, tokenWeightIn);\n        uint denom = bdiv(tokenBalanceOut, tokenWeightOut);\n        uint ratio = bdiv(numer, denom);\n        uint scale = bdiv(BONE, bsub(BONE, swapFee));\n        return  (spotPrice = bmul(ratio, scale));\n    }\n\n    /**********************************************************************************************\n    // calcOutGivenIn                                                                            //\n    // aO = tokenAmountOut                                                                       //\n    // bO = tokenBalanceOut                                                                      //\n    // bI = tokenBalanceIn              /      /            bI             \\    (wI / wO) \\      //\n    // aI = tokenAmountIn    aO = bO * |  1 - | --------------------------  | ^            |     //\n    // wI = tokenWeightIn               \\      \\ ( bI + ( aI * ( 1 - sF )) /              /      //\n    // wO = tokenWeightOut                                                                       //\n    // sF = swapFee                                                                              //\n    **********************************************************************************************/\n    function calcOutGivenIn(\n        uint tokenBalanceIn,\n        uint tokenWeightIn,\n        uint tokenBalanceOut,\n        uint tokenWeightOut,\n        uint tokenAmountIn,\n        uint swapFee\n    )\n        public pure\n        returns (uint tokenAmountOut)\n    {\n        uint weightRatio = bdiv(tokenWeightIn, tokenWeightOut);\n        uint adjustedIn = bsub(BONE, swapFee);\n        adjustedIn = bmul(tokenAmountIn, adjustedIn);\n        uint y = bdiv(tokenBalanceIn, badd(tokenBalanceIn, adjustedIn));\n        uint foo = bpow(y, weightRatio);\n        uint bar = bsub(BONE, foo);\n        tokenAmountOut = bmul(tokenBalanceOut, bar);\n        return tokenAmountOut;\n    }\n\n    /**********************************************************************************************\n    // calcInGivenOut                                                                            //\n    // aI = tokenAmountIn                                                                        //\n    // bO = tokenBalanceOut               /  /     bO      \\    (wO / wI)      \\                 //\n    // bI = tokenBalanceIn          bI * |  | ------------  | ^            - 1  |                //\n    // aO = tokenAmountOut    aI =        \\  \\ ( bO - aO ) /                   /                 //\n    // wI = tokenWeightIn           --------------------------------------------                 //\n    // wO = tokenWeightOut                          ( 1 - sF )                                   //\n    // sF = swapFee                                                                              //\n    **********************************************************************************************/\n    function calcInGivenOut(\n        uint tokenBalanceIn,\n        uint tokenWeightIn,\n        uint tokenBalanceOut,\n        uint tokenWeightOut,\n        uint tokenAmountOut,\n        uint swapFee\n    )\n        public pure\n        returns (uint tokenAmountIn)\n    {\n        uint weightRatio = bdiv(tokenWeightOut, tokenWeightIn);\n        uint diff = bsub(tokenBalanceOut, tokenAmountOut);\n        uint y = bdiv(tokenBalanceOut, diff);\n        uint foo = bpow(y, weightRatio);\n        foo = bsub(foo, BONE);\n        tokenAmountIn = bsub(BONE, swapFee);\n        tokenAmountIn = bdiv(bmul(tokenBalanceIn, foo), tokenAmountIn);\n        return tokenAmountIn;\n    }\n\n    /**********************************************************************************************\n    // calcPoolOutGivenSingleIn                                                                  //\n    // pAo = poolAmountOut         /                                              \\              //\n    // tAi = tokenAmountIn        ///      /     //    wI \\      \\\\       \\     wI \\             //\n    // wI = tokenWeightIn        //| tAi *| 1 - || 1 - --  | * sF || + tBi \\    --  \\            //\n    // tW = totalWeight     pAo=||  \\      \\     \\\\    tW /      //         | ^ tW   | * pS - pS //\n    // tBi = tokenBalanceIn      \\\\  ------------------------------------- /        /            //\n    // pS = poolSupply            \\\\                    tBi               /        /             //\n    // sF = swapFee                \\                                              /              //\n    **********************************************************************************************/\n    function calcPoolOutGivenSingleIn(\n        uint tokenBalanceIn,\n        uint tokenWeightIn,\n        uint poolSupply,\n        uint totalWeight,\n        uint tokenAmountIn,\n        uint swapFee\n    )\n        public pure\n        returns (uint poolAmountOut)\n    {\n        // Charge the trading fee for the proportion of tokenAi\n        ///  which is implicitly traded to the other pool tokens.\n        // That proportion is (1- weightTokenIn)\n        // tokenAiAfterFee = tAi * (1 - (1-weightTi) * poolFee);\n        uint normalizedWeight = bdiv(tokenWeightIn, totalWeight);\n        uint zaz = bmul(bsub(BONE, normalizedWeight), swapFee); \n        uint tokenAmountInAfterFee = bmul(tokenAmountIn, bsub(BONE, zaz));\n\n        uint newTokenBalanceIn = badd(tokenBalanceIn, tokenAmountInAfterFee);\n        uint tokenInRatio = bdiv(newTokenBalanceIn, tokenBalanceIn);\n\n        // uint newPoolSupply = (ratioTi ^ weightTi) * poolSupply;\n        uint poolRatio = bpow(tokenInRatio, normalizedWeight);\n        uint newPoolSupply = bmul(poolRatio, poolSupply);\n        poolAmountOut = bsub(newPoolSupply, poolSupply);\n        return poolAmountOut;\n    }\n\n    /**********************************************************************************************\n    // calcSingleInGivenPoolOut                                                                  //\n    // tAi = tokenAmountIn              //(pS + pAo)\\     /    1    \\\\                           //\n    // pS = poolSupply                 || ---------  | ^ | --------- || * bI - bI                //\n    // pAo = poolAmountOut              \\\\    pS    /     \\(wI / tW)//                           //\n    // bI = balanceIn          tAi =  --------------------------------------------               //\n    // wI = weightIn                              /      wI  \\                                   //\n    // tW = totalWeight                          |  1 - ----  |  * sF                            //\n    // sF = swapFee                               \\      tW  /                                   //\n    **********************************************************************************************/\n    function calcSingleInGivenPoolOut(\n        uint tokenBalanceIn,\n        uint tokenWeightIn,\n        uint poolSupply,\n        uint totalWeight,\n        uint poolAmountOut,\n        uint swapFee\n    )\n        public pure\n        returns (uint tokenAmountIn)\n    {\n        uint normalizedWeight = bdiv(tokenWeightIn, totalWeight);\n        uint newPoolSupply = badd(poolSupply, poolAmountOut);\n        uint poolRatio = bdiv(newPoolSupply, poolSupply);\n      \n        //uint newBalTi = poolRatio^(1/weightTi) * balTi;\n        uint boo = bdiv(BONE, normalizedWeight); \n        uint tokenInRatio = bpow(poolRatio, boo);\n        uint newTokenBalanceIn = bmul(tokenInRatio, tokenBalanceIn);\n        uint tokenAmountInAfterFee = bsub(newTokenBalanceIn, tokenBalanceIn);\n        // Do reverse order of fees charged in joinswap_ExternAmountIn, this way \n        //     ``` pAo == joinswap_ExternAmountIn(Ti, joinswap_PoolAmountOut(pAo, Ti)) ```\n        //uint tAi = tAiAfterFee / (1 - (1-weightTi) * swapFee) ;\n        uint zar = bmul(bsub(BONE, normalizedWeight), swapFee);\n        tokenAmountIn = bdiv(tokenAmountInAfterFee, bsub(BONE, zar));\n        return tokenAmountIn;\n    }\n\n    /**********************************************************************************************\n    // calcSingleOutGivenPoolIn                                                                  //\n    // tAo = tokenAmountOut            /      /                                             \\\\   //\n    // bO = tokenBalanceOut           /      // pS - (pAi * (1 - eF)) \\     /    1    \\      \\\\  //\n    // pAi = poolAmountIn            | bO - || ----------------------- | ^ | --------- | * b0 || //\n    // ps = poolSupply                \\      \\\\          pS           /     \\(wO / tW)/      //  //\n    // wI = tokenWeightIn      tAo =   \\      \\                                             //   //\n    // tW = totalWeight                    /     /      wO \\       \\                             //\n    // sF = swapFee                    *  | 1 - |  1 - ---- | * sF  |                            //\n    // eF = exitFee                        \\     \\      tW /       /                             //\n    **********************************************************************************************/\n    function calcSingleOutGivenPoolIn(\n        uint tokenBalanceOut,\n        uint tokenWeightOut,\n        uint poolSupply,\n        uint totalWeight,\n        uint poolAmountIn,\n        uint swapFee\n    )\n        public pure\n        returns (uint tokenAmountOut)\n    {\n        uint normalizedWeight = bdiv(tokenWeightOut, totalWeight);\n        // charge exit fee on the pool token side\n        // pAiAfterExitFee = pAi*(1-exitFee)\n        uint poolAmountInAfterExitFee = bmul(poolAmountIn, bsub(BONE, EXIT_FEE));\n        uint newPoolSupply = bsub(poolSupply, poolAmountInAfterExitFee);\n        uint poolRatio = bdiv(newPoolSupply, poolSupply);\n     \n        // newBalTo = poolRatio^(1/weightTo) * balTo;\n        uint tokenOutRatio = bpow(poolRatio, bdiv(BONE, normalizedWeight));\n        uint newTokenBalanceOut = bmul(tokenOutRatio, tokenBalanceOut);\n\n        uint tokenAmountOutBeforeSwapFee = bsub(tokenBalanceOut, newTokenBalanceOut);\n\n        // charge swap fee on the output token side \n        //uint tAo = tAoBeforeSwapFee * (1 - (1-weightTo) * swapFee)\n        uint zaz = bmul(bsub(BONE, normalizedWeight), swapFee); \n        tokenAmountOut = bmul(tokenAmountOutBeforeSwapFee, bsub(BONE, zaz));\n        return tokenAmountOut;\n    }\n\n    /**********************************************************************************************\n    // calcPoolInGivenSingleOut                                                                  //\n    // pAi = poolAmountIn               // /               tAo             \\\\     / wO \\     \\   //\n    // bO = tokenBalanceOut            // | bO - -------------------------- |\\   | ---- |     \\  //\n    // tAo = tokenAmountOut      pS - ||   \\     1 - ((1 - (tO / tW)) * sF)/  | ^ \\ tW /  * pS | //\n    // ps = poolSupply                 \\\\ -----------------------------------/                /  //\n    // wO = tokenWeightOut  pAi =       \\\\               bO                 /                /   //\n    // tW = totalWeight           -------------------------------------------------------------  //\n    // sF = swapFee                                        ( 1 - eF )                            //\n    // eF = exitFee                                                                              //\n    **********************************************************************************************/\n    function calcPoolInGivenSingleOut(\n        uint tokenBalanceOut,\n        uint tokenWeightOut,\n        uint poolSupply,\n        uint totalWeight,\n        uint tokenAmountOut,\n        uint swapFee\n    )\n        public pure\n        returns (uint poolAmountIn)\n    {\n\n        // charge swap fee on the output token side \n        uint normalizedWeight = bdiv(tokenWeightOut, totalWeight);\n        //uint tAoBeforeSwapFee = tAo / (1 - (1-weightTo) * swapFee) ;\n        uint zoo = bsub(BONE, normalizedWeight);\n        uint zar = bmul(zoo, swapFee); \n        uint tokenAmountOutBeforeSwapFee = bdiv(tokenAmountOut, bsub(BONE, zar));\n\n        uint newTokenBalanceOut = bsub(tokenBalanceOut, tokenAmountOutBeforeSwapFee);\n        uint tokenOutRatio = bdiv(newTokenBalanceOut, tokenBalanceOut);\n\n        //uint newPoolSupply = (ratioTo ^ weightTo) * poolSupply;\n        uint poolRatio = bpow(tokenOutRatio, normalizedWeight);\n        uint newPoolSupply = bmul(poolRatio, poolSupply);\n        uint poolAmountInAfterExitFee = bsub(poolSupply, newPoolSupply);\n\n        // charge exit fee on the pool token side\n        // pAi = pAiAfterExitFee/(1-exitFee)\n        poolAmountIn = bdiv(poolAmountInAfterExitFee, bsub(BONE, EXIT_FEE));\n        return poolAmountIn;\n    }\n\n\n}\n"},"BNum.sol":{"content":"// This program is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n\n// This program is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n// GNU General Public License for more details.\n\n// You should have received a copy of the GNU General Public License\n// along with this program.  If not, see \u003chttp://www.gnu.org/licenses/\u003e.\n\npragma solidity 0.5.12;\n\nimport \"./BConst.sol\";\n\ncontract BNum is BConst {\n\n    function btoi(uint a)\n        internal pure \n        returns (uint)\n    {\n        return a / BONE;\n    }\n\n    function bfloor(uint a)\n        internal pure\n        returns (uint)\n    {\n        return btoi(a) * BONE;\n    }\n\n    function badd(uint a, uint b)\n        internal pure\n        returns (uint)\n    {\n        uint c = a + b;\n        require(c \u003e= a, \"ERR_ADD_OVERFLOW\");\n        return c;\n    }\n\n    function bsub(uint a, uint b)\n        internal pure\n        returns (uint)\n    {\n        (uint c, bool flag) = bsubSign(a, b);\n        require(!flag, \"ERR_SUB_UNDERFLOW\");\n        return c;\n    }\n\n    function bsubSign(uint a, uint b)\n        internal pure\n        returns (uint, bool)\n    {\n        if (a \u003e= b) {\n            return (a - b, false);\n        } else {\n            return (b - a, true);\n        }\n    }\n\n    function bmul(uint a, uint b)\n        internal pure\n        returns (uint)\n    {\n        uint c0 = a * b;\n        require(a == 0 || c0 / a == b, \"ERR_MUL_OVERFLOW\");\n        uint c1 = c0 + (BONE / 2);\n        require(c1 \u003e= c0, \"ERR_MUL_OVERFLOW\");\n        uint c2 = c1 / BONE;\n        return c2;\n    }\n\n    function bdiv(uint a, uint b)\n        internal pure\n        returns (uint)\n    {\n        require(b != 0, \"ERR_DIV_ZERO\");\n        uint c0 = a * BONE;\n        require(a == 0 || c0 / a == BONE, \"ERR_DIV_INTERNAL\"); // bmul overflow\n        uint c1 = c0 + (b / 2);\n        require(c1 \u003e= c0, \"ERR_DIV_INTERNAL\"); //  badd require\n        uint c2 = c1 / b;\n        return c2;\n    }\n\n    // DSMath.wpow\n    function bpowi(uint a, uint n)\n        internal pure\n        returns (uint)\n    {\n        uint z = n % 2 != 0 ? a : BONE;\n\n        for (n /= 2; n != 0; n /= 2) {\n            a = bmul(a, a);\n\n            if (n % 2 != 0) {\n                z = bmul(z, a);\n            }\n        }\n        return z;\n    }\n\n    // Compute b^(e.w) by splitting it into (b^e)*(b^0.w).\n    // Use `bpowi` for `b^e` and `bpowK` for k iterations\n    // of approximation of b^0.w\n    function bpow(uint base, uint exp)\n        internal pure\n        returns (uint)\n    {\n        require(base \u003e= MIN_BPOW_BASE, \"ERR_BPOW_BASE_TOO_LOW\");\n        require(base \u003c= MAX_BPOW_BASE, \"ERR_BPOW_BASE_TOO_HIGH\");\n\n        uint whole  = bfloor(exp);   \n        uint remain = bsub(exp, whole);\n\n        uint wholePow = bpowi(base, btoi(whole));\n\n        if (remain == 0) {\n            return wholePow;\n        }\n\n        uint partialResult = bpowApprox(base, remain, BPOW_PRECISION);\n        return bmul(wholePow, partialResult);\n    }\n\n    function bpowApprox(uint base, uint exp, uint precision)\n        internal pure\n        returns (uint)\n    {\n        // term 0:\n        uint a     = exp;\n        (uint x, bool xneg)  = bsubSign(base, BONE);\n        uint term = BONE;\n        uint sum   = term;\n        bool negative = false;\n\n\n        // term(k) = numer / denom \n        //         = (product(a - i - 1, i=1--\u003ek) * x^k) / (k!)\n        // each iteration, multiply previous term by (a-(k-1)) * x / k\n        // continue until term is less than precision\n        for (uint i = 1; term \u003e= precision; i++) {\n            uint bigK = i * BONE;\n            (uint c, bool cneg) = bsubSign(a, bsub(bigK, BONE));\n            term = bmul(term, bmul(c, x));\n            term = bdiv(term, bigK);\n            if (term == 0) break;\n\n            if (xneg) negative = !negative;\n            if (cneg) negative = !negative;\n            if (negative) {\n                sum = bsub(sum, term);\n            } else {\n                sum = badd(sum, term);\n            }\n        }\n\n        return sum;\n    }\n\n}\n"},"BPool.sol":{"content":"// This program is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n\n// This program is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n// GNU General Public License for more details.\n\n// You should have received a copy of the GNU General Public License\n// along with this program.  If not, see \u003chttp://www.gnu.org/licenses/\u003e.\n\npragma solidity 0.5.12;\n\nimport \"./BToken.sol\";\nimport \"./BMath.sol\";\n\ncontract BPool is BBronze, BToken, BMath {\n\n    struct Record {\n        bool bound;   // is token bound to pool\n        uint index;   // private\n        uint denorm;  // denormalized weight\n        uint balance;\n    }\n\n    event LOG_SWAP(\n        address indexed caller,\n        address indexed tokenIn,\n        address indexed tokenOut,\n        uint256         tokenAmountIn,\n        uint256         tokenAmountOut\n    );\n\n    event LOG_JOIN(\n        address indexed caller,\n        address indexed tokenIn,\n        uint256         tokenAmountIn\n    );\n\n    event LOG_EXIT(\n        address indexed caller,\n        address indexed tokenOut,\n        uint256         tokenAmountOut\n    );\n\n    event LOG_CALL(\n        bytes4  indexed sig,\n        address indexed caller,\n        bytes           data\n    ) anonymous;\n\n    modifier _logs_() {\n        emit LOG_CALL(msg.sig, msg.sender, msg.data);\n        _;\n    }\n\n    modifier _lock_() {\n        require(!_mutex, \"ERR_REENTRY\");\n        _mutex = true;\n        _;\n        _mutex = false;\n    }\n\n    modifier _viewlock_() {\n        require(!_mutex, \"ERR_REENTRY\");\n        _;\n    }\n\n    bool private _mutex;\n\n    address private _factory;    // BFactory address to push token exitFee to\n    address private _controller; // has CONTROL role\n    bool private _publicSwap; // true if PUBLIC can call SWAP functions\n\n    // `setSwapFee` and `finalize` require CONTROL\n    // `finalize` sets `PUBLIC can SWAP`, `PUBLIC can JOIN`\n    uint private _swapFee;\n    bool private _finalized;\n\n    address[] private _tokens;\n    mapping(address=\u003eRecord) private  _records;\n    uint private _totalWeight;\n\n    constructor() public {\n        _controller = msg.sender;\n        _factory = msg.sender;\n        _swapFee = MIN_FEE;\n        _publicSwap = false;\n        _finalized = false;\n    }\n\n    function isPublicSwap()\n        external view\n        returns (bool)\n    {\n        return _publicSwap;\n    }\n\n    function isFinalized()\n        external view\n        returns (bool)\n    {\n        return _finalized;\n    }\n\n    function isBound(address t)\n        external view\n        returns (bool)\n    {\n        return _records[t].bound;\n    }\n\n    function getNumTokens()\n        external view\n        returns (uint) \n    {\n        return _tokens.length;\n    }\n\n    function getCurrentTokens()\n        external view _viewlock_\n        returns (address[] memory tokens)\n    {\n        return _tokens;\n    }\n\n    function getFinalTokens()\n        external view\n        _viewlock_\n        returns (address[] memory tokens)\n    {\n        require(_finalized, \"ERR_NOT_FINALIZED\");\n        return _tokens;\n    }\n\n    function getDenormalizedWeight(address token)\n        external view\n        _viewlock_\n        returns (uint)\n    {\n\n        require(_records[token].bound, \"ERR_NOT_BOUND\");\n        return _records[token].denorm;\n    }\n\n    function getTotalDenormalizedWeight()\n        external view\n        _viewlock_\n        returns (uint)\n    {\n        return _totalWeight;\n    }\n\n    function getNormalizedWeight(address token)\n        external view\n        _viewlock_\n        returns (uint)\n    {\n\n        require(_records[token].bound, \"ERR_NOT_BOUND\");\n        uint denorm = _records[token].denorm;\n        return bdiv(denorm, _totalWeight);\n    }\n\n    function getBalance(address token)\n        external view\n        _viewlock_\n        returns (uint)\n    {\n\n        require(_records[token].bound, \"ERR_NOT_BOUND\");\n        return _records[token].balance;\n    }\n\n    function getSwapFee()\n        external view\n        _viewlock_\n        returns (uint)\n    {\n        return _swapFee;\n    }\n\n    function getController()\n        external view\n        _viewlock_\n        returns (address)\n    {\n        return _controller;\n    }\n\n    function setSwapFee(uint swapFee)\n        external\n        _logs_\n        _lock_\n    { \n        require(!_finalized, \"ERR_IS_FINALIZED\");\n        require(msg.sender == _controller, \"ERR_NOT_CONTROLLER\");\n        require(swapFee \u003e= MIN_FEE, \"ERR_MIN_FEE\");\n        require(swapFee \u003c= MAX_FEE, \"ERR_MAX_FEE\");\n        _swapFee = swapFee;\n    }\n\n    function setController(address manager)\n        external\n        _logs_\n        _lock_\n    {\n        require(msg.sender == _controller, \"ERR_NOT_CONTROLLER\");\n        _controller = manager;\n    }\n\n    function setPublicSwap(bool public_)\n        external\n        _logs_\n        _lock_\n    {\n        require(!_finalized, \"ERR_IS_FINALIZED\");\n        require(msg.sender == _controller, \"ERR_NOT_CONTROLLER\");\n        _publicSwap = public_;\n    }\n\n    function finalize()\n        external\n        _logs_\n        _lock_\n    {\n        require(msg.sender == _controller, \"ERR_NOT_CONTROLLER\");\n        require(!_finalized, \"ERR_IS_FINALIZED\");\n        require(_tokens.length \u003e= MIN_BOUND_TOKENS, \"ERR_MIN_TOKENS\");\n\n        _finalized = true;\n        _publicSwap = true;\n\n        _mintPoolShare(INIT_POOL_SUPPLY);\n        _pushPoolShare(msg.sender, INIT_POOL_SUPPLY);\n    }\n\n\n    function bind(address token, uint balance, uint denorm)\n        external\n        _logs_\n        // _lock_  Bind does not lock because it jumps to `rebind`, which does\n    {\n        require(msg.sender == _controller, \"ERR_NOT_CONTROLLER\");\n        require(!_records[token].bound, \"ERR_IS_BOUND\");\n        require(!_finalized, \"ERR_IS_FINALIZED\");\n\n        require(_tokens.length \u003c MAX_BOUND_TOKENS, \"ERR_MAX_TOKENS\");\n\n        _records[token] = Record({\n            bound: true,\n            index: _tokens.length,\n            denorm: 0,    // balance and denorm will be validated\n            balance: 0   // and set by `rebind`\n        });\n        _tokens.push(token);\n        rebind(token, balance, denorm);\n    }\n\n    function rebind(address token, uint balance, uint denorm)\n        public\n        _logs_\n        _lock_\n    {\n\n        require(msg.sender == _controller, \"ERR_NOT_CONTROLLER\");\n        require(_records[token].bound, \"ERR_NOT_BOUND\");\n        require(!_finalized, \"ERR_IS_FINALIZED\");\n\n        require(denorm \u003e= MIN_WEIGHT, \"ERR_MIN_WEIGHT\");\n        require(denorm \u003c= MAX_WEIGHT, \"ERR_MAX_WEIGHT\");\n        require(balance \u003e= MIN_BALANCE, \"ERR_MIN_BALANCE\");\n\n        // Adjust the denorm and totalWeight\n        uint oldWeight = _records[token].denorm;\n        if (denorm \u003e oldWeight) {\n            _totalWeight = badd(_totalWeight, bsub(denorm, oldWeight));\n            require(_totalWeight \u003c= MAX_TOTAL_WEIGHT, \"ERR_MAX_TOTAL_WEIGHT\");\n        } else if (denorm \u003c oldWeight) {\n            _totalWeight = bsub(_totalWeight, bsub(oldWeight, denorm));\n        }        \n        _records[token].denorm = denorm;\n\n        // Adjust the balance record and actual token balance\n        uint oldBalance = _records[token].balance;\n        _records[token].balance = balance;\n        if (balance \u003e oldBalance) {\n            _pullUnderlying(token, msg.sender, bsub(balance, oldBalance));\n        } else if (balance \u003c oldBalance) {\n            // In this case liquidity is being withdrawn, so charge EXIT_FEE\n            uint tokenBalanceWithdrawn = bsub(oldBalance, balance);\n            uint tokenExitFee = bmul(tokenBalanceWithdrawn, EXIT_FEE);\n            _pushUnderlying(token, msg.sender, bsub(tokenBalanceWithdrawn, tokenExitFee));\n            _pushUnderlying(token, _factory, tokenExitFee);\n        }\n    }\n\n    function unbind(address token)\n        external\n        _logs_\n        _lock_\n    {\n\n        require(msg.sender == _controller, \"ERR_NOT_CONTROLLER\");\n        require(_records[token].bound, \"ERR_NOT_BOUND\");\n        require(!_finalized, \"ERR_IS_FINALIZED\");\n\n        uint tokenBalance = _records[token].balance;\n        uint tokenExitFee = bmul(tokenBalance, EXIT_FEE);\n\n        _totalWeight = bsub(_totalWeight, _records[token].denorm);\n\n        // Swap the token-to-unbind with the last token,\n        // then delete the last token\n        uint index = _records[token].index;\n        uint last = _tokens.length - 1;\n        _tokens[index] = _tokens[last];\n        _records[_tokens[index]].index = index;\n        _tokens.pop();\n        _records[token] = Record({\n            bound: false,\n            index: 0,\n            denorm: 0,\n            balance: 0\n        });\n\n        _pushUnderlying(token, msg.sender, bsub(tokenBalance, tokenExitFee));\n        _pushUnderlying(token, _factory, tokenExitFee);\n    }\n\n    // Absorb any tokens that have been sent to this contract into the pool\n    function gulp(address token)\n        external\n        _logs_\n        _lock_\n    {\n        require(_records[token].bound, \"ERR_NOT_BOUND\");\n        _records[token].balance = IERC20(token).balanceOf(address(this));\n    }\n\n    function getSpotPrice(address tokenIn, address tokenOut)\n        external view\n        _viewlock_\n        returns (uint spotPrice)\n    {\n        require(_records[tokenIn].bound, \"ERR_NOT_BOUND\");\n        require(_records[tokenOut].bound, \"ERR_NOT_BOUND\");\n        Record storage inRecord = _records[tokenIn];\n        Record storage outRecord = _records[tokenOut];\n        return calcSpotPrice(inRecord.balance, inRecord.denorm, outRecord.balance, outRecord.denorm, _swapFee);\n    }\n\n    function getSpotPriceSansFee(address tokenIn, address tokenOut)\n        external view\n        _viewlock_\n        returns (uint spotPrice)\n    {\n        require(_records[tokenIn].bound, \"ERR_NOT_BOUND\");\n        require(_records[tokenOut].bound, \"ERR_NOT_BOUND\");\n        Record storage inRecord = _records[tokenIn];\n        Record storage outRecord = _records[tokenOut];\n        return calcSpotPrice(inRecord.balance, inRecord.denorm, outRecord.balance, outRecord.denorm, 0);\n    }\n\n    function joinPool(uint poolAmountOut, uint[] calldata maxAmountsIn)\n        external\n        _logs_\n        _lock_\n    {\n        require(_finalized, \"ERR_NOT_FINALIZED\");\n\n        uint poolTotal = totalSupply();\n        uint ratio = bdiv(poolAmountOut, poolTotal);\n        require(ratio != 0, \"ERR_MATH_APPROX\");\n\n        for (uint i = 0; i \u003c _tokens.length; i++) {\n            address t = _tokens[i];\n            uint bal = _records[t].balance;\n            uint tokenAmountIn = bmul(ratio, bal);\n            require(tokenAmountIn != 0, \"ERR_MATH_APPROX\");\n            require(tokenAmountIn \u003c= maxAmountsIn[i], \"ERR_LIMIT_IN\");\n            _records[t].balance = badd(_records[t].balance, tokenAmountIn);\n            emit LOG_JOIN(msg.sender, t, tokenAmountIn);\n            _pullUnderlying(t, msg.sender, tokenAmountIn);\n        }\n        _mintPoolShare(poolAmountOut);\n        _pushPoolShare(msg.sender, poolAmountOut);\n    }\n\n    function exitPool(uint poolAmountIn, uint[] calldata minAmountsOut)\n        external\n        _logs_\n        _lock_\n    {\n        require(_finalized, \"ERR_NOT_FINALIZED\");\n\n        uint poolTotal = totalSupply();\n        uint exitFee = bmul(poolAmountIn, EXIT_FEE);\n        uint pAiAfterExitFee = bsub(poolAmountIn, exitFee);\n        uint ratio = bdiv(pAiAfterExitFee, poolTotal);\n        require(ratio != 0, \"ERR_MATH_APPROX\");\n\n        _pullPoolShare(msg.sender, poolAmountIn);\n        _pushPoolShare(_factory, exitFee);\n        _burnPoolShare(pAiAfterExitFee);\n\n        for (uint i = 0; i \u003c _tokens.length; i++) {\n            address t = _tokens[i];\n            uint bal = _records[t].balance;\n            uint tokenAmountOut = bmul(ratio, bal);\n            require(tokenAmountOut != 0, \"ERR_MATH_APPROX\");\n            require(tokenAmountOut \u003e= minAmountsOut[i], \"ERR_LIMIT_OUT\");\n            _records[t].balance = bsub(_records[t].balance, tokenAmountOut);\n            emit LOG_EXIT(msg.sender, t, tokenAmountOut);\n            _pushUnderlying(t, msg.sender, tokenAmountOut);\n        }\n\n    }\n\n\n    function swapExactAmountIn(\n        address tokenIn,\n        uint tokenAmountIn,\n        address tokenOut,\n        uint minAmountOut,\n        uint maxPrice\n    )\n        external\n        _logs_\n        _lock_\n        returns (uint tokenAmountOut, uint spotPriceAfter)\n    {\n\n        require(_records[tokenIn].bound, \"ERR_NOT_BOUND\");\n        require(_records[tokenOut].bound, \"ERR_NOT_BOUND\");\n        require(_publicSwap, \"ERR_SWAP_NOT_PUBLIC\");\n\n        Record storage inRecord = _records[address(tokenIn)];\n        Record storage outRecord = _records[address(tokenOut)];\n\n        require(tokenAmountIn \u003c= bmul(inRecord.balance, MAX_IN_RATIO), \"ERR_MAX_IN_RATIO\");\n\n        uint spotPriceBefore = calcSpotPrice(\n                                    inRecord.balance,\n                                    inRecord.denorm,\n                                    outRecord.balance,\n                                    outRecord.denorm,\n                                    _swapFee\n                                );\n        require(spotPriceBefore \u003c= maxPrice, \"ERR_BAD_LIMIT_PRICE\");\n\n        tokenAmountOut = calcOutGivenIn(\n                            inRecord.balance,\n                            inRecord.denorm,\n                            outRecord.balance,\n                            outRecord.denorm,\n                            tokenAmountIn,\n                            _swapFee\n                        );\n        require(tokenAmountOut \u003e= minAmountOut, \"ERR_LIMIT_OUT\");\n\n        inRecord.balance = badd(inRecord.balance, tokenAmountIn);\n        outRecord.balance = bsub(outRecord.balance, tokenAmountOut);\n\n        spotPriceAfter = calcSpotPrice(\n                                inRecord.balance,\n                                inRecord.denorm,\n                                outRecord.balance,\n                                outRecord.denorm,\n                                _swapFee\n                            );\n        require(spotPriceAfter \u003e= spotPriceBefore, \"ERR_MATH_APPROX\");     \n        require(spotPriceAfter \u003c= maxPrice, \"ERR_LIMIT_PRICE\");\n        require(spotPriceBefore \u003c= bdiv(tokenAmountIn, tokenAmountOut), \"ERR_MATH_APPROX\");\n\n        emit LOG_SWAP(msg.sender, tokenIn, tokenOut, tokenAmountIn, tokenAmountOut);\n\n        _pullUnderlying(tokenIn, msg.sender, tokenAmountIn);\n        _pushUnderlying(tokenOut, msg.sender, tokenAmountOut);\n\n        return (tokenAmountOut, spotPriceAfter);\n    }\n\n    function swapExactAmountOut(\n        address tokenIn,\n        uint maxAmountIn,\n        address tokenOut,\n        uint tokenAmountOut,\n        uint maxPrice\n    )\n        external\n        _logs_\n        _lock_ \n        returns (uint tokenAmountIn, uint spotPriceAfter)\n    {\n        require(_records[tokenIn].bound, \"ERR_NOT_BOUND\");\n        require(_records[tokenOut].bound, \"ERR_NOT_BOUND\");\n        require(_publicSwap, \"ERR_SWAP_NOT_PUBLIC\");\n\n        Record storage inRecord = _records[address(tokenIn)];\n        Record storage outRecord = _records[address(tokenOut)];\n\n        require(tokenAmountOut \u003c= bmul(outRecord.balance, MAX_OUT_RATIO), \"ERR_MAX_OUT_RATIO\");\n\n        uint spotPriceBefore = calcSpotPrice(\n                                    inRecord.balance,\n                                    inRecord.denorm,\n                                    outRecord.balance,\n                                    outRecord.denorm,\n                                    _swapFee\n                                );\n        require(spotPriceBefore \u003c= maxPrice, \"ERR_BAD_LIMIT_PRICE\");\n\n        tokenAmountIn = calcInGivenOut(\n                            inRecord.balance,\n                            inRecord.denorm,\n                            outRecord.balance,\n                            outRecord.denorm,\n                            tokenAmountOut,\n                            _swapFee\n                        );\n        require(tokenAmountIn \u003c= maxAmountIn, \"ERR_LIMIT_IN\");\n\n        inRecord.balance = badd(inRecord.balance, tokenAmountIn);\n        outRecord.balance = bsub(outRecord.balance, tokenAmountOut);\n\n        spotPriceAfter = calcSpotPrice(\n                                inRecord.balance,\n                                inRecord.denorm,\n                                outRecord.balance,\n                                outRecord.denorm,\n                                _swapFee\n                            );\n        require(spotPriceAfter \u003e= spotPriceBefore, \"ERR_MATH_APPROX\");\n        require(spotPriceAfter \u003c= maxPrice, \"ERR_LIMIT_PRICE\");\n        require(spotPriceBefore \u003c= bdiv(tokenAmountIn, tokenAmountOut), \"ERR_MATH_APPROX\");\n\n        emit LOG_SWAP(msg.sender, tokenIn, tokenOut, tokenAmountIn, tokenAmountOut);\n\n        _pullUnderlying(tokenIn, msg.sender, tokenAmountIn);\n        _pushUnderlying(tokenOut, msg.sender, tokenAmountOut);\n\n        return (tokenAmountIn, spotPriceAfter);\n    }\n\n\n    function joinswapExternAmountIn(address tokenIn, uint tokenAmountIn, uint minPoolAmountOut)\n        external\n        _logs_\n        _lock_\n        returns (uint poolAmountOut)\n\n    {        \n        require(_finalized, \"ERR_NOT_FINALIZED\");\n        require(_records[tokenIn].bound, \"ERR_NOT_BOUND\");\n        require(tokenAmountIn \u003c= bmul(_records[tokenIn].balance, MAX_IN_RATIO), \"ERR_MAX_IN_RATIO\");\n\n        Record storage inRecord = _records[tokenIn];\n\n        poolAmountOut = calcPoolOutGivenSingleIn(\n                            inRecord.balance,\n                            inRecord.denorm,\n                            _totalSupply,\n                            _totalWeight,\n                            tokenAmountIn,\n                            _swapFee\n                        );\n\n        require(poolAmountOut \u003e= minPoolAmountOut, \"ERR_LIMIT_OUT\");\n\n        inRecord.balance = badd(inRecord.balance, tokenAmountIn);\n\n        emit LOG_JOIN(msg.sender, tokenIn, tokenAmountIn);\n\n        _mintPoolShare(poolAmountOut);\n        _pushPoolShare(msg.sender, poolAmountOut);\n        _pullUnderlying(tokenIn, msg.sender, tokenAmountIn);\n\n        return poolAmountOut;\n    }\n\n    function joinswapPoolAmountOut(address tokenIn, uint poolAmountOut, uint maxAmountIn)\n        external\n        _logs_\n        _lock_\n        returns (uint tokenAmountIn)\n    {\n        require(_finalized, \"ERR_NOT_FINALIZED\");\n        require(_records[tokenIn].bound, \"ERR_NOT_BOUND\");\n\n        Record storage inRecord = _records[tokenIn];\n\n        tokenAmountIn = calcSingleInGivenPoolOut(\n                            inRecord.balance,\n                            inRecord.denorm,\n                            _totalSupply,\n                            _totalWeight,\n                            poolAmountOut,\n                            _swapFee\n                        );\n\n        require(tokenAmountIn != 0, \"ERR_MATH_APPROX\");\n        require(tokenAmountIn \u003c= maxAmountIn, \"ERR_LIMIT_IN\");\n        \n        require(tokenAmountIn \u003c= bmul(_records[tokenIn].balance, MAX_IN_RATIO), \"ERR_MAX_IN_RATIO\");\n\n        inRecord.balance = badd(inRecord.balance, tokenAmountIn);\n\n        emit LOG_JOIN(msg.sender, tokenIn, tokenAmountIn);\n\n        _mintPoolShare(poolAmountOut);\n        _pushPoolShare(msg.sender, poolAmountOut);\n        _pullUnderlying(tokenIn, msg.sender, tokenAmountIn);\n\n        return tokenAmountIn;\n    }\n\n    function exitswapPoolAmountIn(address tokenOut, uint poolAmountIn, uint minAmountOut)\n        external\n        _logs_\n        _lock_\n        returns (uint tokenAmountOut)\n    {\n        require(_finalized, \"ERR_NOT_FINALIZED\");\n        require(_records[tokenOut].bound, \"ERR_NOT_BOUND\");\n\n        Record storage outRecord = _records[tokenOut];\n\n        tokenAmountOut = calcSingleOutGivenPoolIn(\n                            outRecord.balance,\n                            outRecord.denorm,\n                            _totalSupply,\n                            _totalWeight,\n                            poolAmountIn,\n                            _swapFee\n                        );\n\n        require(tokenAmountOut \u003e= minAmountOut, \"ERR_LIMIT_OUT\");\n        \n        require(tokenAmountOut \u003c= bmul(_records[tokenOut].balance, MAX_OUT_RATIO), \"ERR_MAX_OUT_RATIO\");\n\n        outRecord.balance = bsub(outRecord.balance, tokenAmountOut);\n\n        uint exitFee = bmul(poolAmountIn, EXIT_FEE);\n\n        emit LOG_EXIT(msg.sender, tokenOut, tokenAmountOut);\n\n        _pullPoolShare(msg.sender, poolAmountIn);\n        _burnPoolShare(bsub(poolAmountIn, exitFee));\n        _pushPoolShare(_factory, exitFee);\n        _pushUnderlying(tokenOut, msg.sender, tokenAmountOut);\n\n        return tokenAmountOut;\n    }\n\n    function exitswapExternAmountOut(address tokenOut, uint tokenAmountOut, uint maxPoolAmountIn)\n        external\n        _logs_\n        _lock_\n        returns (uint poolAmountIn)\n    {\n        require(_finalized, \"ERR_NOT_FINALIZED\");\n        require(_records[tokenOut].bound, \"ERR_NOT_BOUND\");\n        require(tokenAmountOut \u003c= bmul(_records[tokenOut].balance, MAX_OUT_RATIO), \"ERR_MAX_OUT_RATIO\");\n\n        Record storage outRecord = _records[tokenOut];\n\n        poolAmountIn = calcPoolInGivenSingleOut(\n                            outRecord.balance,\n                            outRecord.denorm,\n                            _totalSupply,\n                            _totalWeight,\n                            tokenAmountOut,\n                            _swapFee\n                        );\n\n        require(poolAmountIn != 0, \"ERR_MATH_APPROX\");\n        require(poolAmountIn \u003c= maxPoolAmountIn, \"ERR_LIMIT_IN\");\n\n        outRecord.balance = bsub(outRecord.balance, tokenAmountOut);\n\n        uint exitFee = bmul(poolAmountIn, EXIT_FEE);\n\n        emit LOG_EXIT(msg.sender, tokenOut, tokenAmountOut);\n\n        _pullPoolShare(msg.sender, poolAmountIn);\n        _burnPoolShare(bsub(poolAmountIn, exitFee));\n        _pushPoolShare(_factory, exitFee);\n        _pushUnderlying(tokenOut, msg.sender, tokenAmountOut);        \n\n        return poolAmountIn;\n    }\n\n\n    // ==\n    // \u0027Underlying\u0027 token-manipulation functions make external calls but are NOT locked\n    // You must `_lock_` or otherwise ensure reentry-safety\n\n    function _pullUnderlying(address erc20, address from, uint amount)\n        internal\n    {\n        bool xfer = IERC20(erc20).transferFrom(from, address(this), amount);\n        require(xfer, \"ERR_ERC20_FALSE\");\n    }\n\n    function _pushUnderlying(address erc20, address to, uint amount)\n        internal\n    {\n        bool xfer = IERC20(erc20).transfer(to, amount);\n        require(xfer, \"ERR_ERC20_FALSE\");\n    }\n\n    function _pullPoolShare(address from, uint amount)\n        internal\n    {\n        _pull(from, amount);\n    }\n\n    function _pushPoolShare(address to, uint amount)\n        internal\n    {\n        _push(to, amount);\n    }\n\n    function _mintPoolShare(uint amount)\n        internal\n    {\n        _mint(amount);\n    }\n\n    function _burnPoolShare(uint amount)\n        internal\n    {\n        _burn(amount);\n    }\n\n}\n"},"BToken.sol":{"content":"// This program is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n\n// This program is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n// GNU General Public License for more details.\n\n// You should have received a copy of the GNU General Public License\n// along with this program.  If not, see \u003chttp://www.gnu.org/licenses/\u003e.\n\npragma solidity 0.5.12;\n\nimport \"./BNum.sol\";\n\n// Highly opinionated token implementation\n\ninterface IERC20 {\n    event Approval(address indexed src, address indexed dst, uint amt);\n    event Transfer(address indexed src, address indexed dst, uint amt);\n\n    function totalSupply() external view returns (uint);\n    function balanceOf(address whom) external view returns (uint);\n    function allowance(address src, address dst) external view returns (uint);\n\n    function approve(address dst, uint amt) external returns (bool);\n    function transfer(address dst, uint amt) external returns (bool);\n    function transferFrom(\n        address src, address dst, uint amt\n    ) external returns (bool);\n}\n\ncontract BTokenBase is BNum {\n\n    mapping(address =\u003e uint)                   internal _balance;\n    mapping(address =\u003e mapping(address=\u003euint)) internal _allowance;\n    uint internal _totalSupply;\n\n    event Approval(address indexed src, address indexed dst, uint amt);\n    event Transfer(address indexed src, address indexed dst, uint amt);\n\n    function _mint(uint amt) internal {\n        _balance[address(this)] = badd(_balance[address(this)], amt);\n        _totalSupply = badd(_totalSupply, amt);\n        emit Transfer(address(0), address(this), amt);\n    }\n\n    function _burn(uint amt) internal {\n        require(_balance[address(this)] \u003e= amt, \"ERR_INSUFFICIENT_BAL\");\n        _balance[address(this)] = bsub(_balance[address(this)], amt);\n        _totalSupply = bsub(_totalSupply, amt);\n        emit Transfer(address(this), address(0), amt);\n    }\n\n    function _move(address src, address dst, uint amt) internal {\n        require(_balance[src] \u003e= amt, \"ERR_INSUFFICIENT_BAL\");\n        _balance[src] = bsub(_balance[src], amt);\n        _balance[dst] = badd(_balance[dst], amt);\n        emit Transfer(src, dst, amt);\n    }\n\n    function _push(address to, uint amt) internal {\n        _move(address(this), to, amt);\n    }\n\n    function _pull(address from, uint amt) internal {\n        _move(from, address(this), amt);\n    }\n}\n\ncontract BToken is BTokenBase, IERC20 {\n\n    string  private _name     = \"Balancer Pool Token\";\n    string  private _symbol   = \"BPT\";\n    uint8   private _decimals = 18;\n\n    function name() public view returns (string memory) {\n        return _name;\n    }\n\n    function symbol() public view returns (string memory) {\n        return _symbol;\n    }\n\n    function decimals() public view returns(uint8) {\n        return _decimals;\n    }\n\n    function allowance(address src, address dst) external view returns (uint) {\n        return _allowance[src][dst];\n    }\n\n    function balanceOf(address whom) external view returns (uint) {\n        return _balance[whom];\n    }\n\n    function totalSupply() public view returns (uint) {\n        return _totalSupply;\n    }\n\n    function approve(address dst, uint amt) external returns (bool) {\n        _allowance[msg.sender][dst] = amt;\n        emit Approval(msg.sender, dst, amt);\n        return true;\n    }\n\n    function increaseApproval(address dst, uint amt) external returns (bool) {\n        _allowance[msg.sender][dst] = badd(_allowance[msg.sender][dst], amt);\n        emit Approval(msg.sender, dst, _allowance[msg.sender][dst]);\n        return true;\n    }\n\n    function decreaseApproval(address dst, uint amt) external returns (bool) {\n        uint oldValue = _allowance[msg.sender][dst];\n        if (amt \u003e oldValue) {\n            _allowance[msg.sender][dst] = 0;\n        } else {\n            _allowance[msg.sender][dst] = bsub(oldValue, amt);\n        }\n        emit Approval(msg.sender, dst, _allowance[msg.sender][dst]);\n        return true;\n    }\n\n    function transfer(address dst, uint amt) external returns (bool) {\n        _move(msg.sender, dst, amt);\n        return true;\n    }\n\n    function transferFrom(address src, address dst, uint amt) external returns (bool) {\n        require(msg.sender == src || amt \u003c= _allowance[src][msg.sender], \"ERR_BTOKEN_BAD_CALLER\");\n        _move(src, dst, amt);\n        if (msg.sender != src \u0026\u0026 _allowance[src][msg.sender] != uint256(-1)) {\n            _allowance[src][msg.sender] = bsub(_allowance[src][msg.sender], amt);\n            emit Approval(msg.sender, dst, _allowance[src][msg.sender]);\n        }\n        return true;\n    }\n}\n"}}

                              File 4 of 5: WrappedLuna
                              // SPDX-License-Identifier: MIT
                              pragma solidity >=0.6.0;
                              import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
                              import "@openzeppelin/contracts/access/Ownable.sol";
                              contract WrappedToken is ERC20, Ownable {
                                  event Burn(address indexed _sender, bytes32 indexed _to, uint256 amount);
                                  constructor(string memory name, string memory symbol)
                                      public
                                      ERC20(name, symbol)
                                  {}
                                  function burn(uint256 amount, bytes32 to) public {
                                      _burn(_msgSender(), amount);
                                      emit Burn(_msgSender(), to, amount);
                                  }
                                  function mint(address account, uint256 amount) public onlyOwner {
                                      _mint(account, amount);
                                  }
                              }
                              contract WrappedLuna is WrappedToken {
                                  constructor() public WrappedToken("Wrapped LUNA Token", "LUNA") {}
                              }
                              contract WrappedUST is WrappedToken {
                                  constructor() public WrappedToken("Wrapped UST Token", "UST") {}
                              }
                              contract WrappedKRT is WrappedToken {
                                  constructor() public WrappedToken("Wrapped KRT Token", "KRT") {}
                              }
                              contract WrappedSDT is WrappedToken {
                                  constructor() public WrappedToken("Wrapped SDT Token", "SDT") {}
                              }
                              contract WrappedMNT is WrappedToken {
                                  constructor() public WrappedToken("Wrapped MNT Token", "MNT") {}
                              }
                              contract WrappedMIR is WrappedToken {
                                  constructor() public WrappedToken("Wrapped MIR Token", "MIR") {}
                              }
                              contract WrappedmAAPL is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror AAPL Token", "mAAPL") {}
                              }
                              contract WrappedmGOOGL is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror GOOGL Token", "mGOOGL") {}
                              }
                              contract WrappedmTSLA is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror TSLA Token", "mTSLA") {}
                              }
                              contract WrappedmNFLX is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror NFLX Token", "mNFLX") {}
                              }
                              contract WrappedmQQQ is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror QQQ Token", "mQQQ") {}
                              }
                              contract WrappedmTWTR is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror TWTR Token", "mTWTR") {}
                              }
                              contract WrappedmMSFT is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror MSFT Token", "mMSFT") {}
                              }
                              contract WrappedmAMZN is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror AMZN Token", "mAMZN") {}
                              }
                              contract WrappedmBABA is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror BABA Token", "mBABA") {}
                              }
                              contract WrappedmIAU is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror IAU Token", "mIAU") {}
                              }
                              contract WrappedmSLV is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror SLV Token", "mSLV") {}
                              }
                              contract WrappedmUSO is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror USO Token", "mUSO") {}
                              }
                              contract WrappedmVIXY is WrappedToken {
                                  constructor() public WrappedToken("Wrapped Mirror VIXY Token", "mVIXY") {}
                              }
                              // SPDX-License-Identifier: MIT
                              pragma solidity ^0.6.0;
                              /*
                               * @dev Provides information about the current execution context, including the
                               * sender of the transaction and its data. While these are generally available
                               * via msg.sender and msg.data, they should not be accessed in such a direct
                               * manner, since when dealing with GSN meta-transactions the account sending and
                               * paying for execution may not be the actual sender (as far as an application
                               * is concerned).
                               *
                               * This contract is only required for intermediate, library-like contracts.
                               */
                              abstract contract Context {
                                  function _msgSender() internal view virtual returns (address payable) {
                                      return msg.sender;
                                  }
                                  function _msgData() internal view virtual returns (bytes memory) {
                                      this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
                                      return msg.data;
                                  }
                              }
                              // SPDX-License-Identifier: MIT
                              pragma solidity ^0.6.0;
                              import "../GSN/Context.sol";
                              /**
                               * @dev Contract module which provides a basic access control mechanism, where
                               * there is an account (an owner) that can be granted exclusive access to
                               * specific functions.
                               *
                               * By default, the owner account will be the one that deploys the contract. This
                               * can later be changed with {transferOwnership}.
                               *
                               * This module is used through inheritance. It will make available the modifier
                               * `onlyOwner`, which can be applied to your functions to restrict their use to
                               * the owner.
                               */
                              contract Ownable is Context {
                                  address private _owner;
                                  event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
                                  /**
                                   * @dev Initializes the contract setting the deployer as the initial owner.
                                   */
                                  constructor () internal {
                                      address msgSender = _msgSender();
                                      _owner = msgSender;
                                      emit OwnershipTransferred(address(0), msgSender);
                                  }
                                  /**
                                   * @dev Returns the address of the current owner.
                                   */
                                  function owner() public view returns (address) {
                                      return _owner;
                                  }
                                  /**
                                   * @dev Throws if called by any account other than the owner.
                                   */
                                  modifier onlyOwner() {
                                      require(_owner == _msgSender(), "Ownable: caller is not the owner");
                                      _;
                                  }
                                  /**
                                   * @dev Leaves the contract without owner. It will not be possible to call
                                   * `onlyOwner` functions anymore. Can only be called by the current owner.
                                   *
                                   * NOTE: Renouncing ownership will leave the contract without an owner,
                                   * thereby removing any functionality that is only available to the owner.
                                   */
                                  function renounceOwnership() public virtual onlyOwner {
                                      emit OwnershipTransferred(_owner, address(0));
                                      _owner = address(0);
                                  }
                                  /**
                                   * @dev Transfers ownership of the contract to a new account (`newOwner`).
                                   * Can only be called by the current owner.
                                   */
                                  function transferOwnership(address newOwner) public virtual onlyOwner {
                                      require(newOwner != address(0), "Ownable: new owner is the zero address");
                                      emit OwnershipTransferred(_owner, newOwner);
                                      _owner = newOwner;
                                  }
                              }
                              // SPDX-License-Identifier: MIT
                              pragma solidity ^0.6.0;
                              /**
                               * @dev Wrappers over Solidity's arithmetic operations with added overflow
                               * checks.
                               *
                               * Arithmetic operations in Solidity wrap on overflow. This can easily result
                               * in bugs, because programmers usually assume that an overflow raises an
                               * error, which is the standard behavior in high level programming languages.
                               * `SafeMath` restores this intuition by reverting the transaction when an
                               * operation overflows.
                               *
                               * Using this library instead of the unchecked operations eliminates an entire
                               * class of bugs, so it's recommended to use it always.
                               */
                              library SafeMath {
                                  /**
                                   * @dev Returns the addition of two unsigned integers, reverting on
                                   * overflow.
                                   *
                                   * Counterpart to Solidity's `+` operator.
                                   *
                                   * Requirements:
                                   *
                                   * - Addition cannot overflow.
                                   */
                                  function add(uint256 a, uint256 b) internal pure returns (uint256) {
                                      uint256 c = a + b;
                                      require(c >= a, "SafeMath: addition overflow");
                                      return c;
                                  }
                                  /**
                                   * @dev Returns the subtraction of two unsigned integers, reverting on
                                   * overflow (when the result is negative).
                                   *
                                   * Counterpart to Solidity's `-` operator.
                                   *
                                   * Requirements:
                                   *
                                   * - Subtraction cannot overflow.
                                   */
                                  function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                                      return sub(a, b, "SafeMath: subtraction overflow");
                                  }
                                  /**
                                   * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
                                   * overflow (when the result is negative).
                                   *
                                   * Counterpart to Solidity's `-` operator.
                                   *
                                   * Requirements:
                                   *
                                   * - Subtraction cannot overflow.
                                   */
                                  function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                                      require(b <= a, errorMessage);
                                      uint256 c = a - b;
                                      return c;
                                  }
                                  /**
                                   * @dev Returns the multiplication of two unsigned integers, reverting on
                                   * overflow.
                                   *
                                   * Counterpart to Solidity's `*` operator.
                                   *
                                   * Requirements:
                                   *
                                   * - Multiplication cannot overflow.
                                   */
                                  function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                                      // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                                      // benefit is lost if 'b' is also tested.
                                      // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                                      if (a == 0) {
                                          return 0;
                                      }
                                      uint256 c = a * b;
                                      require(c / a == b, "SafeMath: multiplication overflow");
                                      return c;
                                  }
                                  /**
                                   * @dev Returns the integer division of two unsigned integers. Reverts on
                                   * division by zero. The result is rounded towards zero.
                                   *
                                   * Counterpart to Solidity's `/` operator. Note: this function uses a
                                   * `revert` opcode (which leaves remaining gas untouched) while Solidity
                                   * uses an invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   *
                                   * - The divisor cannot be zero.
                                   */
                                  function div(uint256 a, uint256 b) internal pure returns (uint256) {
                                      return div(a, b, "SafeMath: division by zero");
                                  }
                                  /**
                                   * @dev Returns the integer division of two unsigned integers. Reverts with custom message on
                                   * division by zero. The result is rounded towards zero.
                                   *
                                   * Counterpart to Solidity's `/` operator. Note: this function uses a
                                   * `revert` opcode (which leaves remaining gas untouched) while Solidity
                                   * uses an invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   *
                                   * - The divisor cannot be zero.
                                   */
                                  function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                                      require(b > 0, errorMessage);
                                      uint256 c = a / b;
                                      // assert(a == b * c + a % b); // There is no case in which this doesn't hold
                                      return c;
                                  }
                                  /**
                                   * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
                                   * Reverts when dividing by zero.
                                   *
                                   * Counterpart to Solidity's `%` operator. This function uses a `revert`
                                   * opcode (which leaves remaining gas untouched) while Solidity uses an
                                   * invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   *
                                   * - The divisor cannot be zero.
                                   */
                                  function mod(uint256 a, uint256 b) internal pure returns (uint256) {
                                      return mod(a, b, "SafeMath: modulo by zero");
                                  }
                                  /**
                                   * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
                                   * Reverts with custom message when dividing by zero.
                                   *
                                   * Counterpart to Solidity's `%` operator. This function uses a `revert`
                                   * opcode (which leaves remaining gas untouched) while Solidity uses an
                                   * invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   *
                                   * - The divisor cannot be zero.
                                   */
                                  function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                                      require(b != 0, errorMessage);
                                      return a % b;
                                  }
                              }
                              // SPDX-License-Identifier: MIT
                              pragma solidity ^0.6.0;
                              import "../../GSN/Context.sol";
                              import "./IERC20.sol";
                              import "../../math/SafeMath.sol";
                              import "../../utils/Address.sol";
                              /**
                               * @dev Implementation of the {IERC20} interface.
                               *
                               * This implementation is agnostic to the way tokens are created. This means
                               * that a supply mechanism has to be added in a derived contract using {_mint}.
                               * For a generic mechanism see {ERC20PresetMinterPauser}.
                               *
                               * TIP: For a detailed writeup see our guide
                               * https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226[How
                               * to implement supply mechanisms].
                               *
                               * We have followed general OpenZeppelin guidelines: functions revert instead
                               * of returning `false` on failure. This behavior is nonetheless conventional
                               * and does not conflict with the expectations of ERC20 applications.
                               *
                               * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
                               * This allows applications to reconstruct the allowance for all accounts just
                               * by listening to said events. Other implementations of the EIP may not emit
                               * these events, as it isn't required by the specification.
                               *
                               * Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
                               * functions have been added to mitigate the well-known issues around setting
                               * allowances. See {IERC20-approve}.
                               */
                              contract ERC20 is Context, IERC20 {
                                  using SafeMath for uint256;
                                  using Address for address;
                                  mapping (address => uint256) private _balances;
                                  mapping (address => mapping (address => uint256)) private _allowances;
                                  uint256 private _totalSupply;
                                  string private _name;
                                  string private _symbol;
                                  uint8 private _decimals;
                                  /**
                                   * @dev Sets the values for {name} and {symbol}, initializes {decimals} with
                                   * a default value of 18.
                                   *
                                   * To select a different value for {decimals}, use {_setupDecimals}.
                                   *
                                   * All three of these values are immutable: they can only be set once during
                                   * construction.
                                   */
                                  constructor (string memory name, string memory symbol) public {
                                      _name = name;
                                      _symbol = symbol;
                                      _decimals = 18;
                                  }
                                  /**
                                   * @dev Returns the name of the token.
                                   */
                                  function name() public view returns (string memory) {
                                      return _name;
                                  }
                                  /**
                                   * @dev Returns the symbol of the token, usually a shorter version of the
                                   * name.
                                   */
                                  function symbol() public view returns (string memory) {
                                      return _symbol;
                                  }
                                  /**
                                   * @dev Returns the number of decimals used to get its user representation.
                                   * For example, if `decimals` equals `2`, a balance of `505` tokens should
                                   * be displayed to a user as `5,05` (`505 / 10 ** 2`).
                                   *
                                   * Tokens usually opt for a value of 18, imitating the relationship between
                                   * Ether and Wei. This is the value {ERC20} uses, unless {_setupDecimals} is
                                   * called.
                                   *
                                   * NOTE: This information is only used for _display_ purposes: it in
                                   * no way affects any of the arithmetic of the contract, including
                                   * {IERC20-balanceOf} and {IERC20-transfer}.
                                   */
                                  function decimals() public view returns (uint8) {
                                      return _decimals;
                                  }
                                  /**
                                   * @dev See {IERC20-totalSupply}.
                                   */
                                  function totalSupply() public view override returns (uint256) {
                                      return _totalSupply;
                                  }
                                  /**
                                   * @dev See {IERC20-balanceOf}.
                                   */
                                  function balanceOf(address account) public view override returns (uint256) {
                                      return _balances[account];
                                  }
                                  /**
                                   * @dev See {IERC20-transfer}.
                                   *
                                   * Requirements:
                                   *
                                   * - `recipient` cannot be the zero address.
                                   * - the caller must have a balance of at least `amount`.
                                   */
                                  function transfer(address recipient, uint256 amount) public virtual override returns (bool) {
                                      _transfer(_msgSender(), recipient, amount);
                                      return true;
                                  }
                                  /**
                                   * @dev See {IERC20-allowance}.
                                   */
                                  function allowance(address owner, address spender) public view virtual override returns (uint256) {
                                      return _allowances[owner][spender];
                                  }
                                  /**
                                   * @dev See {IERC20-approve}.
                                   *
                                   * Requirements:
                                   *
                                   * - `spender` cannot be the zero address.
                                   */
                                  function approve(address spender, uint256 amount) public virtual override returns (bool) {
                                      _approve(_msgSender(), spender, amount);
                                      return true;
                                  }
                                  /**
                                   * @dev See {IERC20-transferFrom}.
                                   *
                                   * Emits an {Approval} event indicating the updated allowance. This is not
                                   * required by the EIP. See the note at the beginning of {ERC20};
                                   *
                                   * Requirements:
                                   * - `sender` and `recipient` cannot be the zero address.
                                   * - `sender` must have a balance of at least `amount`.
                                   * - the caller must have allowance for ``sender``'s tokens of at least
                                   * `amount`.
                                   */
                                  function transferFrom(address sender, address recipient, uint256 amount) public virtual override returns (bool) {
                                      _transfer(sender, recipient, amount);
                                      _approve(sender, _msgSender(), _allowances[sender][_msgSender()].sub(amount, "ERC20: transfer amount exceeds allowance"));
                                      return true;
                                  }
                                  /**
                                   * @dev Atomically increases the allowance granted to `spender` by the caller.
                                   *
                                   * This is an alternative to {approve} that can be used as a mitigation for
                                   * problems described in {IERC20-approve}.
                                   *
                                   * Emits an {Approval} event indicating the updated allowance.
                                   *
                                   * Requirements:
                                   *
                                   * - `spender` cannot be the zero address.
                                   */
                                  function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
                                      _approve(_msgSender(), spender, _allowances[_msgSender()][spender].add(addedValue));
                                      return true;
                                  }
                                  /**
                                   * @dev Atomically decreases the allowance granted to `spender` by the caller.
                                   *
                                   * This is an alternative to {approve} that can be used as a mitigation for
                                   * problems described in {IERC20-approve}.
                                   *
                                   * Emits an {Approval} event indicating the updated allowance.
                                   *
                                   * Requirements:
                                   *
                                   * - `spender` cannot be the zero address.
                                   * - `spender` must have allowance for the caller of at least
                                   * `subtractedValue`.
                                   */
                                  function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
                                      _approve(_msgSender(), spender, _allowances[_msgSender()][spender].sub(subtractedValue, "ERC20: decreased allowance below zero"));
                                      return true;
                                  }
                                  /**
                                   * @dev Moves tokens `amount` from `sender` to `recipient`.
                                   *
                                   * This is internal function is equivalent to {transfer}, and can be used to
                                   * e.g. implement automatic token fees, slashing mechanisms, etc.
                                   *
                                   * Emits a {Transfer} event.
                                   *
                                   * Requirements:
                                   *
                                   * - `sender` cannot be the zero address.
                                   * - `recipient` cannot be the zero address.
                                   * - `sender` must have a balance of at least `amount`.
                                   */
                                  function _transfer(address sender, address recipient, uint256 amount) internal virtual {
                                      require(sender != address(0), "ERC20: transfer from the zero address");
                                      require(recipient != address(0), "ERC20: transfer to the zero address");
                                      _beforeTokenTransfer(sender, recipient, amount);
                                      _balances[sender] = _balances[sender].sub(amount, "ERC20: transfer amount exceeds balance");
                                      _balances[recipient] = _balances[recipient].add(amount);
                                      emit Transfer(sender, recipient, amount);
                                  }
                                  /** @dev Creates `amount` tokens and assigns them to `account`, increasing
                                   * the total supply.
                                   *
                                   * Emits a {Transfer} event with `from` set to the zero address.
                                   *
                                   * Requirements
                                   *
                                   * - `to` cannot be the zero address.
                                   */
                                  function _mint(address account, uint256 amount) internal virtual {
                                      require(account != address(0), "ERC20: mint to the zero address");
                                      _beforeTokenTransfer(address(0), account, amount);
                                      _totalSupply = _totalSupply.add(amount);
                                      _balances[account] = _balances[account].add(amount);
                                      emit Transfer(address(0), account, amount);
                                  }
                                  /**
                                   * @dev Destroys `amount` tokens from `account`, reducing the
                                   * total supply.
                                   *
                                   * Emits a {Transfer} event with `to` set to the zero address.
                                   *
                                   * Requirements
                                   *
                                   * - `account` cannot be the zero address.
                                   * - `account` must have at least `amount` tokens.
                                   */
                                  function _burn(address account, uint256 amount) internal virtual {
                                      require(account != address(0), "ERC20: burn from the zero address");
                                      _beforeTokenTransfer(account, address(0), amount);
                                      _balances[account] = _balances[account].sub(amount, "ERC20: burn amount exceeds balance");
                                      _totalSupply = _totalSupply.sub(amount);
                                      emit Transfer(account, address(0), amount);
                                  }
                                  /**
                                   * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
                                   *
                                   * This internal function is equivalent to `approve`, and can be used to
                                   * e.g. set automatic allowances for certain subsystems, etc.
                                   *
                                   * Emits an {Approval} event.
                                   *
                                   * Requirements:
                                   *
                                   * - `owner` cannot be the zero address.
                                   * - `spender` cannot be the zero address.
                                   */
                                  function _approve(address owner, address spender, uint256 amount) internal virtual {
                                      require(owner != address(0), "ERC20: approve from the zero address");
                                      require(spender != address(0), "ERC20: approve to the zero address");
                                      _allowances[owner][spender] = amount;
                                      emit Approval(owner, spender, amount);
                                  }
                                  /**
                                   * @dev Sets {decimals} to a value other than the default one of 18.
                                   *
                                   * WARNING: This function should only be called from the constructor. Most
                                   * applications that interact with token contracts will not expect
                                   * {decimals} to ever change, and may work incorrectly if it does.
                                   */
                                  function _setupDecimals(uint8 decimals_) internal {
                                      _decimals = decimals_;
                                  }
                                  /**
                                   * @dev Hook that is called before any transfer of tokens. This includes
                                   * minting and burning.
                                   *
                                   * Calling conditions:
                                   *
                                   * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
                                   * will be to transferred to `to`.
                                   * - when `from` is zero, `amount` tokens will be minted for `to`.
                                   * - when `to` is zero, `amount` of ``from``'s tokens will be burned.
                                   * - `from` and `to` are never both zero.
                                   *
                                   * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
                                   */
                                  function _beforeTokenTransfer(address from, address to, uint256 amount) internal virtual { }
                              }
                              // SPDX-License-Identifier: MIT
                              pragma solidity ^0.6.0;
                              /**
                               * @dev Interface of the ERC20 standard as defined in the EIP.
                               */
                              interface IERC20 {
                                  /**
                                   * @dev Returns the amount of tokens in existence.
                                   */
                                  function totalSupply() external view returns (uint256);
                                  /**
                                   * @dev Returns the amount of tokens owned by `account`.
                                   */
                                  function balanceOf(address account) external view returns (uint256);
                                  /**
                                   * @dev Moves `amount` tokens from the caller's account to `recipient`.
                                   *
                                   * Returns a boolean value indicating whether the operation succeeded.
                                   *
                                   * Emits a {Transfer} event.
                                   */
                                  function transfer(address recipient, uint256 amount) external returns (bool);
                                  /**
                                   * @dev Returns the remaining number of tokens that `spender` will be
                                   * allowed to spend on behalf of `owner` through {transferFrom}. This is
                                   * zero by default.
                                   *
                                   * This value changes when {approve} or {transferFrom} are called.
                                   */
                                  function allowance(address owner, address spender) external view returns (uint256);
                                  /**
                                   * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                                   *
                                   * Returns a boolean value indicating whether the operation succeeded.
                                   *
                                   * IMPORTANT: Beware that changing an allowance with this method brings the risk
                                   * that someone may use both the old and the new allowance by unfortunate
                                   * transaction ordering. One possible solution to mitigate this race
                                   * condition is to first reduce the spender's allowance to 0 and set the
                                   * desired value afterwards:
                                   * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                                   *
                                   * Emits an {Approval} event.
                                   */
                                  function approve(address spender, uint256 amount) external returns (bool);
                                  /**
                                   * @dev Moves `amount` tokens from `sender` to `recipient` using the
                                   * allowance mechanism. `amount` is then deducted from the caller's
                                   * allowance.
                                   *
                                   * Returns a boolean value indicating whether the operation succeeded.
                                   *
                                   * Emits a {Transfer} event.
                                   */
                                  function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
                                  /**
                                   * @dev Emitted when `value` tokens are moved from one account (`from`) to
                                   * another (`to`).
                                   *
                                   * Note that `value` may be zero.
                                   */
                                  event Transfer(address indexed from, address indexed to, uint256 value);
                                  /**
                                   * @dev Emitted when the allowance of a `spender` for an `owner` is set by
                                   * a call to {approve}. `value` is the new allowance.
                                   */
                                  event Approval(address indexed owner, address indexed spender, uint256 value);
                              }
                              // SPDX-License-Identifier: MIT
                              pragma solidity ^0.6.2;
                              /**
                               * @dev Collection of functions related to the address type
                               */
                              library Address {
                                  /**
                                   * @dev Returns true if `account` is a contract.
                                   *
                                   * [IMPORTANT]
                                   * ====
                                   * It is unsafe to assume that an address for which this function returns
                                   * false is an externally-owned account (EOA) and not a contract.
                                   *
                                   * Among others, `isContract` will return false for the following
                                   * types of addresses:
                                   *
                                   *  - an externally-owned account
                                   *  - a contract in construction
                                   *  - an address where a contract will be created
                                   *  - an address where a contract lived, but was destroyed
                                   * ====
                                   */
                                  function isContract(address account) internal view returns (bool) {
                                      // This method relies in extcodesize, which returns 0 for contracts in
                                      // construction, since the code is only stored at the end of the
                                      // constructor execution.
                                      uint256 size;
                                      // solhint-disable-next-line no-inline-assembly
                                      assembly { size := extcodesize(account) }
                                      return size > 0;
                                  }
                                  /**
                                   * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
                                   * `recipient`, forwarding all available gas and reverting on errors.
                                   *
                                   * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
                                   * of certain opcodes, possibly making contracts go over the 2300 gas limit
                                   * imposed by `transfer`, making them unable to receive funds via
                                   * `transfer`. {sendValue} removes this limitation.
                                   *
                                   * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
                                   *
                                   * IMPORTANT: because control is transferred to `recipient`, care must be
                                   * taken to not create reentrancy vulnerabilities. Consider using
                                   * {ReentrancyGuard} or the
                                   * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                                   */
                                  function sendValue(address payable recipient, uint256 amount) internal {
                                      require(address(this).balance >= amount, "Address: insufficient balance");
                                      // solhint-disable-next-line avoid-low-level-calls, avoid-call-value
                                      (bool success, ) = recipient.call{ value: amount }("");
                                      require(success, "Address: unable to send value, recipient may have reverted");
                                  }
                                  /**
                                   * @dev Performs a Solidity function call using a low level `call`. A
                                   * plain`call` is an unsafe replacement for a function call: use this
                                   * function instead.
                                   *
                                   * If `target` reverts with a revert reason, it is bubbled up by this
                                   * function (like regular Solidity function calls).
                                   *
                                   * Returns the raw returned data. To convert to the expected return value,
                                   * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
                                   *
                                   * Requirements:
                                   *
                                   * - `target` must be a contract.
                                   * - calling `target` with `data` must not revert.
                                   *
                                   * _Available since v3.1._
                                   */
                                  function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                                    return functionCall(target, data, "Address: low-level call failed");
                                  }
                                  /**
                                   * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                                   * `errorMessage` as a fallback revert reason when `target` reverts.
                                   *
                                   * _Available since v3.1._
                                   */
                                  function functionCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) {
                                      return _functionCallWithValue(target, data, 0, errorMessage);
                                  }
                                  /**
                                   * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                                   * but also transferring `value` wei to `target`.
                                   *
                                   * Requirements:
                                   *
                                   * - the calling contract must have an ETH balance of at least `value`.
                                   * - the called Solidity function must be `payable`.
                                   *
                                   * _Available since v3.1._
                                   */
                                  function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                                      return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                                  }
                                  /**
                                   * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                                   * with `errorMessage` as a fallback revert reason when `target` reverts.
                                   *
                                   * _Available since v3.1._
                                   */
                                  function functionCallWithValue(address target, bytes memory data, uint256 value, string memory errorMessage) internal returns (bytes memory) {
                                      require(address(this).balance >= value, "Address: insufficient balance for call");
                                      return _functionCallWithValue(target, data, value, errorMessage);
                                  }
                                  function _functionCallWithValue(address target, bytes memory data, uint256 weiValue, string memory errorMessage) private returns (bytes memory) {
                                      require(isContract(target), "Address: call to non-contract");
                                      // solhint-disable-next-line avoid-low-level-calls
                                      (bool success, bytes memory returndata) = target.call{ value: weiValue }(data);
                                      if (success) {
                                          return returndata;
                                      } else {
                                          // Look for revert reason and bubble it up if present
                                          if (returndata.length > 0) {
                                              // The easiest way to bubble the revert reason is using memory via assembly
                                              // solhint-disable-next-line no-inline-assembly
                                              assembly {
                                                  let returndata_size := mload(returndata)
                                                  revert(add(32, returndata), returndata_size)
                                              }
                                          } else {
                                              revert(errorMessage);
                                          }
                                      }
                                  }
                              }
                              

                              File 5 of 5: ChiToken
                              /*
                                                                                               ,╖╗#▒▓▓▓▓▓╣╬╣▓▓▓▓▒#╗╗╓,
                                                                                         ,╗@▓╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▓▓╗╖
                                                                                     ╓#▓╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╝▀╠╠▄╣╝╜"""╙╙▀╝╝╣╬╬╬╬▓▌╖
                                                                                  ╓▓╣╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▀`╓å▓▓▓╙  ,▄▓▓██▓▓▓▄▄▄▄▄╠╠╙╠▄▄
                                                                               ╓@╣╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▌  ê`'     *▀▀▀▀▀▀▀▓██████████████▄
                                                                             ╔▓╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬                      ╙▀████████████▌
                                                                           ╓▓╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬                         ╙████████████▌
                                                                         ,▓╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▓▀                          ╗▄█████████████▄
                                                                        é╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▓▌   #╙                       ╙▀█████████████▓
                                                                       ╣╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▀              ╙▓╬╣▓▄            ╙▀▀███████████µ
                                                                      ▓╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▀╣╝╙             ╒▓╬╬╬╬╬╬▓         ╙████████████████µ
                                                                     ▓╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▌              ╖╖╖▄▓╬╬╬╬╬╬╬▓         █████████████████µ
                                                                    ╣╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬        ,#▓╣╬╬▓╬╬╬╬╬╬╬╬╬╬╬╬▌         ▓█████████████████
                                                                   ]╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╓╖     ]╣╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╣╨          ██████████████████▌
                                                                   ▓╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▓▌╖,   ╙╠╠▓╬╬╬╬╬╬╬╬╬▓╝╙           ╫███████████████████
                                                                  ]╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╝▀╙              ▓████████████████████▌
                                                                  ║╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╝▀╙`                 ▄███████████████████████
                                                                  ╟╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▓╝▀╙                     ,▄█████████████████████████
                                                                  ╟╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╝╜`                       ▄▓████████████████████████████
                                                                  ║╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╣▀`                     ,▄▄▓████████████████████████████████
                                                                  ▐╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬▓╙                       ,,,▄╠▓██████████████████████████████▌
                                                                   ╣╬╬╬╬╬╬╬╬╬╬╬╬╬╬▓╙╔▒`               ╓▄▓████████████████████████████████████████⌐
                                                                   ╚╬╬╬╬╬╬╬╬╬╬╬╬╬▓▓╣▓              ▄▓████████████████████████████████████████████
                                                                    ▓╬╬╬╬╬╬╬╬╬╬╬╬╬╬╬             ▄██████████████████████████████████████████████⌐
                                                                     ╣╬╬╬╬╬╬╬╬╬╬╬╬╬╛           ▄███████████████████████████████████████████████▌
                                                                     └╣╬╬╬╬╬╬╬╬╬╬╬▓           ▄███████████████████████████████████████████████▌
                                                                      └▓╬╬╬╬╬╬╬╬╬╬Γ          ]███████████████████████████████████████████████▀
                                                                        ╣╬╬╬╬╬╬╬╬╬⌐          ╫████████████████████████████████▀▀▀▀▀▓████████╜
                                                                         ╙╬╬╬╬╬╬╬╬⌐          ╟███████████████████████████▀╙    ,▄▓▓▓▓▓████▓
                                                                           ╫╬╬╬╬╬╬b           ████████████████████████▀`    ,Φ▀▀█████████╙
                                                                             ╫╬╬╬╬▌╟           ██████████████████▀╓▀─       ▄▓█████████▀
                                                                               ╚╣╬▓╣▓           └▀████████████▀`         ╓▓█████████▓╙
                                                                                 ╙╝╬╬▓             .▄▄▓█▀▀▀`           ▄▓█████████▀
                                                                                    ╙▀▓▄                            ƒ,▓███████▀▀
                                                                                        "                        ╓▓█▓█████▀▀└
                                                                                                             ╓▄▓████▀▀╙└
                              
                               ██████╗██╗  ██╗██╗     ██████╗  █████╗ ███████╗████████╗ ██████╗ ██╗  ██╗███████╗███╗   ██╗    ██████╗ ██╗   ██╗     ██╗██╗███╗   ██╗ ██████╗██╗  ██╗
                              ██╔════╝██║  ██║██║    ██╔════╝ ██╔══██╗██╔════╝╚══██╔══╝██╔═══██╗██║ ██╔╝██╔════╝████╗  ██║    ██╔══██╗╚██╗ ██╔╝    ███║██║████╗  ██║██╔════╝██║  ██║
                              ██║     ███████║██║    ██║  ███╗███████║███████╗   ██║   ██║   ██║█████╔╝ █████╗  ██╔██╗ ██║    ██████╔╝ ╚████╔╝     ╚██║██║██╔██╗ ██║██║     ███████║
                              ██║     ██╔══██║██║    ██║   ██║██╔══██║╚════██║   ██║   ██║   ██║██╔═██╗ ██╔══╝  ██║╚██╗██║    ██╔══██╗  ╚██╔╝       ██║██║██║╚██╗██║██║     ██╔══██║
                              ╚██████╗██║  ██║██║    ╚██████╔╝██║  ██║███████║   ██║   ╚██████╔╝██║  ██╗███████╗██║ ╚████║    ██████╔╝   ██║        ██║██║██║ ╚████║╚██████╗██║  ██║
                               ╚═════╝╚═╝  ╚═╝╚═╝     ╚═════╝ ╚═╝  ╚═╝╚══════╝   ╚═╝    ╚═════╝ ╚═╝  ╚═╝╚══════╝╚═╝  ╚═══╝    ╚═════╝    ╚═╝        ╚═╝╚═╝╚═╝  ╚═══╝ ╚═════╝╚═╝  ╚═╝
                              
                                                                                           Copyright by 1inch Corporation
                                                                                               https://1inch.exchange
                              
                              ---
                              Deployer wallet address:
                              0x7E1E3334130355799F833ffec2D731BCa3E68aF6
                              
                              Signed raw transaction for chainId 1:
                              0xf90d7f808506fc23ac00830bd0fa8080b90d2c608060405234801561001057600080fd5b50610d0c806100206000396000f3fe608060405234801561001057600080fd5b506004361061010b5760003560e01c806370a08231116100a2578063a9059cbb11610071578063a9059cbb14610305578063b0ac19a014610331578063d89135cd1461036a578063d8ccd0f314610372578063dd62ed3e1461038f5761010b565b806370a08231146102b057806395d89b41146102d6578063a0712d68146102de578063a2309ff8146102fd5761010b565b806323b872dd116100de57806323b872dd14610213578063313ce567146102495780635f2e2b45146102675780636366b936146102935761010b565b806306fdde0314610110578063079d229f1461018d578063095ea7b3146101cb57806318160ddd1461020b575b600080fd5b6101186103bd565b6040805160208082528351818301528351919283929083019185019080838360005b8381101561015257818101518382015260200161013a565b50505050905090810190601f16801561017f5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6101b9600480360360408110156101a357600080fd5b506001600160a01b0381351690602001356103ee565b60408051918252519081900360200190f35b6101f7600480360360408110156101e157600080fd5b506001600160a01b03813516906020013561041f565b604080519115158252519081900360200190f35b6101b9610435565b6101f76004803603606081101561022957600080fd5b506001600160a01b03813581169160208101359091169060400135610453565b6102516104c2565b6040805160ff9092168252519081900360200190f35b6101b96004803603604081101561027d57600080fd5b506001600160a01b0381351690602001356104c7565b6101b9600480360360208110156102a957600080fd5b50356104e2565b6101b9600480360360208110156102c657600080fd5b50356001600160a01b03166104ff565b61011861051a565b6102fb600480360360208110156102f457600080fd5b5035610539565b005b6101b961070d565b6101f76004803603604081101561031b57600080fd5b506001600160a01b038135169060200135610713565b61034e6004803603602081101561034757600080fd5b5035610720565b604080516001600160a01b039092168252519081900360200190f35b6101b961078b565b6101b96004803603602081101561038857600080fd5b5035610791565b6101b9600480360360408110156103a557600080fd5b506001600160a01b03813581169160200135166107aa565b60405180604001604052806015815260200174086d0d2408ec2e6e8ded6cadc40c4f24062d2dcc6d605b1b81525081565b60006104188361041361040985610404886104ff565b6107d5565b61040487336107aa565b6104c7565b9392505050565b600061042c3384846107eb565b50600192915050565b600061044e60035460025461084d90919063ffffffff16565b905090565b600061046084848461088f565b6104b884336104b385604051806060016040528060288152602001610c8b602891396001600160a01b038a166000908152600160209081526040808320338452909152902054919063ffffffff61096116565b6107eb565b5060019392505050565b600081565b60006104d383836109f8565b6104dc82610a59565b50919050565b60006104f96104f483610404336104ff565b610791565b92915050565b6001600160a01b031660009081526020819052604090205490565b6040518060400160405280600381526020016243484960e81b81525081565b6002547f746d4946c0e9f43f4dee607b0ef1fa1c3318585733ff6000526015600bf30000600052602082045b80156106d85781601e600080f55060018201601e600080f55060028201601e600080f55060038201601e600080f55060048201601e600080f55060058201601e600080f55060068201601e600080f55060078201601e600080f55060088201601e600080f55060098201601e600080f550600a8201601e600080f550600b8201601e600080f550600c8201601e600080f550600d8201601e600080f550600e8201601e600080f550600f8201601e600080f55060108201601e600080f55060118201601e600080f55060128201601e600080f55060138201601e600080f55060148201601e600080f55060158201601e600080f55060168201601e600080f55060178201601e600080f55060188201601e600080f55060198201601e600080f550601a8201601e600080f550601b8201601e600080f550601c8201601e600080f550601d8201601e600080f550601e8201601e600080f550601f8201601e600080f5506020919091019060001901610565565b50601f82165b80156106fc5781601e600080f55060019190910190600019016106de565b506107073383610ad1565b60025550565b60025481565b600061042c33848461088f565b604080516001600160f81b03196020808301919091523060601b602183015260358201939093527f3c1644c68e5d6cb380c36d1bf847fdbc0c7ac28030025a2fc5e63cce23c16348605580830191909152825180830390910181526075909101909152805191012090565b60035481565b600061079d3383610b50565b6107a682610a59565b5090565b6001600160a01b03918216600090815260016020908152604080832093909416825291909152205490565b60008183106107e45781610418565b5090919050565b6001600160a01b03808416600081815260016020908152604080832094871680845294825291829020859055815185815291517f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259281900390910190a3505050565b600061041883836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f770000815250610961565b6108d281604051806060016040528060268152602001610c65602691396001600160a01b038616600090815260208190526040902054919063ffffffff61096116565b6001600160a01b038085166000908152602081905260408082209390935590841681522054610907908263ffffffff610be816565b6001600160a01b038084166000818152602081815260409182902094909455805185815290519193928716927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef92918290030190a3505050565b600081848411156109f05760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b838110156109b557818101518382015260200161099d565b50505050905090810190601f1680156109e25780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b505050900390565b610a028282610b50565b610a5582336104b384604051806060016040528060248152602001610cb3602491396001600160a01b0388166000908152600160209081526040808320338452909152902054919063ffffffff61096116565b5050565b60035460005b82811015610aca57610a72818301610720565b6040516001600160a01b039190911690600081818181865af19150503d8060008114610aba576040519150601f19603f3d011682016040523d82523d6000602084013e610abf565b606091505b505050600101610a5f565b5001600355565b6001600160a01b038216600090815260208190526040902054610afa908263ffffffff610be816565b6001600160a01b0383166000818152602081815260408083209490945583518581529351929391927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9281900390910190a35050565b610b9381604051806060016040528060228152602001610c43602291396001600160a01b038516600090815260208190526040902054919063ffffffff61096116565b6001600160a01b038316600081815260208181526040808320949094558351858152935191937fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef929081900390910190a35050565b600082820183811015610418576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fdfe45524332303a206275726e20616d6f756e7420657863656564732062616c616e636545524332303a207472616e7366657220616d6f756e7420657863656564732062616c616e636545524332303a207472616e7366657220616d6f756e74206578636565647320616c6c6f77616e636545524332303a206275726e20616d6f756e74206578636565647320616c6c6f77616e6365a2646970667358221220687f814fb4c0b3c6abd66ebdb1f1eabcf69becf92a382c3af453e0b21c3d15b564736f6c6343000608003325a00ed87a047b4e415bd7f8cf7a7ce5a1c204125df1cedc35c7bdcb71bd2a29a35ea02db2490337fa6c425f1b3d74b7b217de8b394adb3f571827629c06dc16364b66
                              ---
                              */
                              // File: @openzeppelin/contracts/math/Math.sol
                              
                              pragma solidity ^0.6.0;
                              
                              /**
                               * @dev Standard math utilities missing in the Solidity language.
                               */
                              library Math {
                                  /**
                                   * @dev Returns the largest of two numbers.
                                   */
                                  function max(uint256 a, uint256 b) internal pure returns (uint256) {
                                      return a >= b ? a : b;
                                  }
                              
                                  /**
                                   * @dev Returns the smallest of two numbers.
                                   */
                                  function min(uint256 a, uint256 b) internal pure returns (uint256) {
                                      return a < b ? a : b;
                                  }
                              
                                  /**
                                   * @dev Returns the average of two numbers. The result is rounded towards
                                   * zero.
                                   */
                                  function average(uint256 a, uint256 b) internal pure returns (uint256) {
                                      // (a + b) / 2 can overflow, so we distribute
                                      return (a / 2) + (b / 2) + ((a % 2 + b % 2) / 2);
                                  }
                              }
                              
                              // File: @openzeppelin/contracts/math/SafeMath.sol
                              
                              pragma solidity ^0.6.0;
                              
                              /**
                               * @dev Wrappers over Solidity's arithmetic operations with added overflow
                               * checks.
                               *
                               * Arithmetic operations in Solidity wrap on overflow. This can easily result
                               * in bugs, because programmers usually assume that an overflow raises an
                               * error, which is the standard behavior in high level programming languages.
                               * `SafeMath` restores this intuition by reverting the transaction when an
                               * operation overflows.
                               *
                               * Using this library instead of the unchecked operations eliminates an entire
                               * class of bugs, so it's recommended to use it always.
                               */
                              library SafeMath {
                                  /**
                                   * @dev Returns the addition of two unsigned integers, reverting on
                                   * overflow.
                                   *
                                   * Counterpart to Solidity's `+` operator.
                                   *
                                   * Requirements:
                                   * - Addition cannot overflow.
                                   */
                                  function add(uint256 a, uint256 b) internal pure returns (uint256) {
                                      uint256 c = a + b;
                                      require(c >= a, "SafeMath: addition overflow");
                              
                                      return c;
                                  }
                              
                                  /**
                                   * @dev Returns the subtraction of two unsigned integers, reverting on
                                   * overflow (when the result is negative).
                                   *
                                   * Counterpart to Solidity's `-` operator.
                                   *
                                   * Requirements:
                                   * - Subtraction cannot overflow.
                                   */
                                  function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                                      return sub(a, b, "SafeMath: subtraction overflow");
                                  }
                              
                                  /**
                                   * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
                                   * overflow (when the result is negative).
                                   *
                                   * Counterpart to Solidity's `-` operator.
                                   *
                                   * Requirements:
                                   * - Subtraction cannot overflow.
                                   */
                                  function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                                      require(b <= a, errorMessage);
                                      uint256 c = a - b;
                              
                                      return c;
                                  }
                              
                                  /**
                                   * @dev Returns the multiplication of two unsigned integers, reverting on
                                   * overflow.
                                   *
                                   * Counterpart to Solidity's `*` operator.
                                   *
                                   * Requirements:
                                   * - Multiplication cannot overflow.
                                   */
                                  function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                                      // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                                      // benefit is lost if 'b' is also tested.
                                      // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                                      if (a == 0) {
                                          return 0;
                                      }
                              
                                      uint256 c = a * b;
                                      require(c / a == b, "SafeMath: multiplication overflow");
                              
                                      return c;
                                  }
                              
                                  /**
                                   * @dev Returns the integer division of two unsigned integers. Reverts on
                                   * division by zero. The result is rounded towards zero.
                                   *
                                   * Counterpart to Solidity's `/` operator. Note: this function uses a
                                   * `revert` opcode (which leaves remaining gas untouched) while Solidity
                                   * uses an invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   * - The divisor cannot be zero.
                                   */
                                  function div(uint256 a, uint256 b) internal pure returns (uint256) {
                                      return div(a, b, "SafeMath: division by zero");
                                  }
                              
                                  /**
                                   * @dev Returns the integer division of two unsigned integers. Reverts with custom message on
                                   * division by zero. The result is rounded towards zero.
                                   *
                                   * Counterpart to Solidity's `/` operator. Note: this function uses a
                                   * `revert` opcode (which leaves remaining gas untouched) while Solidity
                                   * uses an invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   * - The divisor cannot be zero.
                                   */
                                  function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                                      // Solidity only automatically asserts when dividing by 0
                                      require(b > 0, errorMessage);
                                      uint256 c = a / b;
                                      // assert(a == b * c + a % b); // There is no case in which this doesn't hold
                              
                                      return c;
                                  }
                              
                                  /**
                                   * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
                                   * Reverts when dividing by zero.
                                   *
                                   * Counterpart to Solidity's `%` operator. This function uses a `revert`
                                   * opcode (which leaves remaining gas untouched) while Solidity uses an
                                   * invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   * - The divisor cannot be zero.
                                   */
                                  function mod(uint256 a, uint256 b) internal pure returns (uint256) {
                                      return mod(a, b, "SafeMath: modulo by zero");
                                  }
                              
                                  /**
                                   * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
                                   * Reverts with custom message when dividing by zero.
                                   *
                                   * Counterpart to Solidity's `%` operator. This function uses a `revert`
                                   * opcode (which leaves remaining gas untouched) while Solidity uses an
                                   * invalid opcode to revert (consuming all remaining gas).
                                   *
                                   * Requirements:
                                   * - The divisor cannot be zero.
                                   */
                                  function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                                      require(b != 0, errorMessage);
                                      return a % b;
                                  }
                              }
                              
                              // File: @openzeppelin/contracts/token/ERC20/IERC20.sol
                              
                              pragma solidity ^0.6.0;
                              
                              /**
                               * @dev Interface of the ERC20 standard as defined in the EIP.
                               */
                              interface IERC20 {
                                  /**
                                   * @dev Returns the amount of tokens in existence.
                                   */
                                  function totalSupply() external view returns (uint256);
                              
                                  /**
                                   * @dev Returns the amount of tokens owned by `account`.
                                   */
                                  function balanceOf(address account) external view returns (uint256);
                              
                                  /**
                                   * @dev Moves `amount` tokens from the caller's account to `recipient`.
                                   *
                                   * Returns a boolean value indicating whether the operation succeeded.
                                   *
                                   * Emits a {Transfer} event.
                                   */
                                  function transfer(address recipient, uint256 amount) external returns (bool);
                              
                                  /**
                                   * @dev Returns the remaining number of tokens that `spender` will be
                                   * allowed to spend on behalf of `owner` through {transferFrom}. This is
                                   * zero by default.
                                   *
                                   * This value changes when {approve} or {transferFrom} are called.
                                   */
                                  function allowance(address owner, address spender) external view returns (uint256);
                              
                                  /**
                                   * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                                   *
                                   * Returns a boolean value indicating whether the operation succeeded.
                                   *
                                   * IMPORTANT: Beware that changing an allowance with this method brings the risk
                                   * that someone may use both the old and the new allowance by unfortunate
                                   * transaction ordering. One possible solution to mitigate this race
                                   * condition is to first reduce the spender's allowance to 0 and set the
                                   * desired value afterwards:
                                   * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                                   *
                                   * Emits an {Approval} event.
                                   */
                                  function approve(address spender, uint256 amount) external returns (bool);
                              
                                  /**
                                   * @dev Moves `amount` tokens from `sender` to `recipient` using the
                                   * allowance mechanism. `amount` is then deducted from the caller's
                                   * allowance.
                                   *
                                   * Returns a boolean value indicating whether the operation succeeded.
                                   *
                                   * Emits a {Transfer} event.
                                   */
                                  function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
                              
                                  /**
                                   * @dev Emitted when `value` tokens are moved from one account (`from`) to
                                   * another (`to`).
                                   *
                                   * Note that `value` may be zero.
                                   */
                                  event Transfer(address indexed from, address indexed to, uint256 value);
                              
                                  /**
                                   * @dev Emitted when the allowance of a `spender` for an `owner` is set by
                                   * a call to {approve}. `value` is the new allowance.
                                   */
                                  event Approval(address indexed owner, address indexed spender, uint256 value);
                              }
                              
                              // File: contracts/ChiToken.sol
                              
                              pragma solidity ^0.6.0;
                              
                              
                              
                              
                              
                              abstract contract ERC20WithoutTotalSupply is IERC20 {
                                  using SafeMath for uint256;
                              
                                  mapping(address => uint256) private _balances;
                                  mapping(address => mapping(address => uint256)) private _allowances;
                              
                                  function balanceOf(address account) public view override returns (uint256) {
                                      return _balances[account];
                                  }
                              
                                  function allowance(address owner, address spender) public view override returns (uint256) {
                                      return _allowances[owner][spender];
                                  }
                              
                                  function transfer(address recipient, uint256 amount) public override returns (bool) {
                                      _transfer(msg.sender, recipient, amount);
                                      return true;
                                  }
                              
                                  function approve(address spender, uint256 amount) public override returns (bool) {
                                      _approve(msg.sender, spender, amount);
                                      return true;
                                  }
                              
                                  function transferFrom(address sender, address recipient, uint256 amount) public override returns (bool) {
                                      _transfer(sender, recipient, amount);
                                      _approve(sender, msg.sender, _allowances[sender][msg.sender].sub(amount, "ERC20: transfer amount exceeds allowance"));
                                      return true;
                                  }
                              
                                  function _transfer(address sender, address recipient, uint256 amount) internal {
                                      _balances[sender] = _balances[sender].sub(amount, "ERC20: transfer amount exceeds balance");
                                      _balances[recipient] = _balances[recipient].add(amount);
                                      emit Transfer(sender, recipient, amount);
                                  }
                              
                                  function _approve(address owner, address spender, uint256 amount) internal {
                                      _allowances[owner][spender] = amount;
                                      emit Approval(owner, spender, amount);
                                  }
                              
                                  function _mint(address account, uint256 amount) internal {
                                      _balances[account] = _balances[account].add(amount);
                                      emit Transfer(address(0), account, amount);
                                  }
                              
                                  function _burn(address account, uint256 amount) internal {
                                      _balances[account] = _balances[account].sub(amount, "ERC20: burn amount exceeds balance");
                                      emit Transfer(account, address(0), amount);
                                  }
                              
                                  function _burnFrom(address account, uint256 amount) internal {
                                      _burn(account, amount);
                                      _approve(account, msg.sender, _allowances[account][msg.sender].sub(amount, "ERC20: burn amount exceeds allowance"));
                                  }
                              }
                              
                              
                              contract ChiToken is IERC20, ERC20WithoutTotalSupply {
                                  string constant public name = "Chi Gastoken by 1inch";
                                  string constant public symbol = "CHI";
                                  uint8 constant public decimals = 0;
                              
                                  uint256 public totalMinted;
                                  uint256 public totalBurned;
                              
                                  function totalSupply() public view override returns(uint256) {
                                      return totalMinted.sub(totalBurned);
                                  }
                              
                                  function mint(uint256 value) public {
                                      uint256 offset = totalMinted;
                                      assembly {
                                          mstore(0, 0x746d4946c0e9F43F4Dee607b0eF1fA1c3318585733ff6000526015600bf30000)
                              
                                          for {let i := div(value, 32)} i {i := sub(i, 1)} {
                                              pop(create2(0, 0, 30, add(offset, 0))) pop(create2(0, 0, 30, add(offset, 1)))
                                              pop(create2(0, 0, 30, add(offset, 2))) pop(create2(0, 0, 30, add(offset, 3)))
                                              pop(create2(0, 0, 30, add(offset, 4))) pop(create2(0, 0, 30, add(offset, 5)))
                                              pop(create2(0, 0, 30, add(offset, 6))) pop(create2(0, 0, 30, add(offset, 7)))
                                              pop(create2(0, 0, 30, add(offset, 8))) pop(create2(0, 0, 30, add(offset, 9)))
                                              pop(create2(0, 0, 30, add(offset, 10))) pop(create2(0, 0, 30, add(offset, 11)))
                                              pop(create2(0, 0, 30, add(offset, 12))) pop(create2(0, 0, 30, add(offset, 13)))
                                              pop(create2(0, 0, 30, add(offset, 14))) pop(create2(0, 0, 30, add(offset, 15)))
                                              pop(create2(0, 0, 30, add(offset, 16))) pop(create2(0, 0, 30, add(offset, 17)))
                                              pop(create2(0, 0, 30, add(offset, 18))) pop(create2(0, 0, 30, add(offset, 19)))
                                              pop(create2(0, 0, 30, add(offset, 20))) pop(create2(0, 0, 30, add(offset, 21)))
                                              pop(create2(0, 0, 30, add(offset, 22))) pop(create2(0, 0, 30, add(offset, 23)))
                                              pop(create2(0, 0, 30, add(offset, 24))) pop(create2(0, 0, 30, add(offset, 25)))
                                              pop(create2(0, 0, 30, add(offset, 26))) pop(create2(0, 0, 30, add(offset, 27)))
                                              pop(create2(0, 0, 30, add(offset, 28))) pop(create2(0, 0, 30, add(offset, 29)))
                                              pop(create2(0, 0, 30, add(offset, 30))) pop(create2(0, 0, 30, add(offset, 31)))
                                              offset := add(offset, 32)
                                          }
                              
                                          for {let i := and(value, 0x1F)} i {i := sub(i, 1)} {
                                              pop(create2(0, 0, 30, offset))
                                              offset := add(offset, 1)
                                          }
                                      }
                              
                                      _mint(msg.sender, value);
                                      totalMinted = offset;
                                  }
                              
                                  function computeAddress2(uint256 salt) public view returns (address) {
                                      bytes32 _data = keccak256(
                                          abi.encodePacked(bytes1(0xff), address(this), salt, bytes32(0x3c1644c68e5d6cb380c36d1bf847fdbc0c7ac28030025a2fc5e63cce23c16348))
                                      );
                                      return address(uint256(_data));
                                  }
                              
                                  function _destroyChildren(uint256 value) internal {
                                      uint256 _totalBurned = totalBurned;
                                      for (uint256 i = 0; i < value; i++) {
                                          computeAddress2(_totalBurned + i).call("");
                                      }
                                      totalBurned = _totalBurned + value;
                                  }
                              
                                  function free(uint256 value) public returns (uint256)  {
                                      _burn(msg.sender, value);
                                      _destroyChildren(value);
                                      return value;
                                  }
                              
                                  function freeUpTo(uint256 value) public returns (uint256) {
                                      return free(Math.min(value, balanceOf(msg.sender)));
                                  }
                              
                                  function freeFrom(address from, uint256 value) public returns (uint256) {
                                      _burnFrom(from, value);
                                      _destroyChildren(value);
                                      return value;
                                  }
                              
                                  function freeFromUpTo(address from, uint256 value) public returns (uint256) {
                                      return freeFrom(from, Math.min(Math.min(value, balanceOf(from)), allowance(from, msg.sender)));
                                  }
                              }