Skip to content

Commit

Permalink
Solidity: rm duplicated code & stop inserting license & version headers
Browse files Browse the repository at this point in the history
- removes duplicated code for compiling contracts with solc
- stops silently inserting license and version headers into the input
  source (fixes: ethereum/act#161)
  • Loading branch information
d-xo committed Sep 27, 2023
1 parent fb7c953 commit 7ec64c3
Show file tree
Hide file tree
Showing 2 changed files with 16 additions and 94 deletions.
104 changes: 15 additions & 89 deletions src/EVM/Solidity.hs
Original file line number Diff line number Diff line change
@@ -1,13 +1,10 @@
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE DeriveAnyClass #-}
{-# LANGUAGE DerivingStrategies #-}
{-# LANGUAGE QuasiQuotes #-}

module EVM.Solidity
( solidity
, solcRuntime
, solidity'
, yul'
, yul
, yulRuntime
, JumpType (..)
Expand Down Expand Up @@ -72,20 +69,17 @@ import Data.List.NonEmpty qualified as NonEmpty
import Data.Maybe
import Data.Semigroup
import Data.Sequence (Seq)
import Data.String.Here qualified as Here
import Data.Text (pack, intercalate)
import Data.Text qualified as T
import Data.Text.Encoding (encodeUtf8, decodeUtf8)
import Data.Text.IO (readFile, writeFile)
import Data.Text.IO (readFile)
import Data.Vector (Vector)
import Data.Vector qualified as Vector
import Data.Word (Word8)
import Options.Generic
import Prelude hiding (readFile, writeFile)
import System.FilePattern.Directory
import System.FilePath.Posix
import System.IO hiding (readFile, writeFile)
import System.IO.Temp
import System.Process
import Text.Read (readMaybe)
import Witch (unsafeInto)
Expand Down Expand Up @@ -358,40 +352,40 @@ readSolc pt root fp =

yul :: Text -> Text -> IO (Maybe ByteString)
yul contractName src = do
(json, path) <- yul' src
let f = (json ^?! key "contracts") ^?! key (Key.fromText path)
json <- solc Yul src
let f = (json ^?! key "contracts") ^?! key (Key.fromText "hevm.sol")
c = f ^?! key (Key.fromText $ if T.null contractName then "object" else contractName)
bytecode = c ^?! key "evm" ^?! key "bytecode" ^?! key "object" % _String
pure $ (toCode contractName) <$> (Just bytecode)

yulRuntime :: Text -> Text -> IO (Maybe ByteString)
yulRuntime contractName src = do
(json, path) <- yul' src
let f = (json ^?! key "contracts") ^?! key (Key.fromText path)
json <- solc Yul src
let f = (json ^?! key "contracts") ^?! key (Key.fromText "hevm.sol")
c = f ^?! key (Key.fromText $ if T.null contractName then "object" else contractName)
bytecode = c ^?! key "evm" ^?! key "deployedBytecode" ^?! key "object" % _String
pure $ (toCode contractName) <$> (Just bytecode)

solidity :: Text -> Text -> IO (Maybe ByteString)
solidity contract src = do
(json, path) <- solidity' src
json <- solc Solidity src
let (Contracts sol, _, _) = fromJust $ readStdJSON json
pure $ Map.lookup (path <> ":" <> contract) sol <&> (.creationCode)
pure $ Map.lookup ("hevm.sol:" <> contract) sol <&> (.creationCode)

solcRuntime :: Text -> Text -> IO (Maybe ByteString)
solcRuntime contract src = do
(json, path) <- solidity' src
json <- solc Solidity src
case readStdJSON json of
Just (Contracts sol, _, _) -> pure $ Map.lookup (path <> ":" <> contract) sol <&> (.runtimeCode)
Just (Contracts sol, _, _) -> pure $ Map.lookup ("hevm.sol:" <> contract) sol <&> (.runtimeCode)
Nothing -> internalError $ "unable to parse solidity output:\n" <> (T.unpack json)

functionAbi :: Text -> IO Method
functionAbi f = do
(json, path) <- solidity' ("contract ABI { function " <> f <> " public {}}")
json <- solc Solidity ("contract ABI { function " <> f <> " public {}}")
let (Contracts sol, _, _) = fromMaybe
(internalError . T.unpack $ "unable to parse solc output:\n" <> json)
(readStdJSON json)
case Map.toList $ (fromJust (Map.lookup (path <> ":ABI") sol)).abiMap of
case Map.toList $ (fromJust (Map.lookup "hevm.sol:ABI" sol)).abiMap of
[(_,b)] -> pure b
_ -> internalError "unexpected abi format"

Expand Down Expand Up @@ -661,80 +655,12 @@ toCode contractName t = case BS16.decodeBase16 (encodeUtf8 t) of
then error $ T.unpack ("Error toCode: unlinked libraries detected in bytecode, in " <> contractName)
else error $ T.unpack ("Error toCode:" <> e <> ", in " <> contractName)

solidity' :: Text -> IO (Text, Text)
solidity' src = withSystemTempFile "hevm.sol" $ \path handle -> do
hClose handle
writeFile path ("//SPDX-License-Identifier: UNLICENSED\n" <> "pragma solidity ^0.8.6;\n" <> src)
writeFile (path <> ".json")
[Here.i|
{
"language": "Solidity",
"sources": {
${path}: {
"urls": [
${path}
]
}
},
"settings": {
"outputSelection": {
"*": {
"*": [
"metadata",
"evm.bytecode",
"evm.deployedBytecode",
"abi",
"storageLayout",
"evm.bytecode.sourceMap",
"evm.bytecode.linkReferences",
"evm.bytecode.generatedSources",
"evm.deployedBytecode.sourceMap",
"evm.deployedBytecode.linkReferences",
"evm.deployedBytecode.generatedSources"
],
"": [
"ast"
]
}
}
}
}
|]
x <- pack <$>
readProcess
"solc"
["--allow-paths", path, "--standard-json", (path <> ".json")]
""
pure (x, pack path)

yul' :: Text -> IO (Text, Text)
yul' src = withSystemTempFile "hevm.yul" $ \path handle -> do
hClose handle
writeFile path src
writeFile (path <> ".json")
[Here.i|
{
"language": "Yul",
"sources": { ${path}: { "urls": [ ${path} ] } },
"settings": { "outputSelection": { "*": { "*": ["*"], "": [ "*" ] } } }
}
|]
x <- pack <$>
readProcess
"solc"
["--allow-paths", path, "--standard-json", (path <> ".json")]
""
pure (x, pack path)

solc :: Language -> Text -> IO Text
solc lang src =
withSystemTempFile "hevm.sol" $ \path handle -> do
hClose handle
writeFile path (stdjson lang src)
T.pack <$> readProcess
"solc"
["--standard-json", path]
""
T.pack <$> readProcess
"solc"
["--standard-json"]
(T.unpack $ stdjson lang src)

data Language = Solidity | Yul
deriving (Show)
Expand Down
6 changes: 1 addition & 5 deletions test/test.hs
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,6 @@ import Witch (unsafeInto, into)

import Optics.Core hiding (pre, re, elements)
import Optics.State
import Optics.Operators.Unsafe

import EVM hiding (choose)
import EVM.ABI
Expand Down Expand Up @@ -683,10 +682,7 @@ tests = testGroup "hevm"
}
|]

(json, path') <- solidity' srccode
let (Contracts solc', _, _) = fromJust $ readStdJSON json
initCode = (solc' ^?! ix (path' <> ":A")).creationCode
-- add constructor arguments
Just initCode <- solidity "A" srccode
assertEqual "constructor args screwed up metadata stripping" (stripBytecodeMetadata (initCode <> encodeAbiValue (AbiUInt 256 1))) (stripBytecodeMetadata initCode)
]

Expand Down

0 comments on commit 7ec64c3

Please sign in to comment.