Skip to content

Commit 419ab92

Browse files
authoredFeb 17, 2017
Merge pull request ethereum#1703 from ethereum/fuzzer
Executable for use with AFL
2 parents 7bdc4dd + f66ebbc commit 419ab92

File tree

3 files changed

+115
-22
lines changed

3 files changed

+115
-22
lines changed
 

‎test/CMakeLists.txt

+4-18
Original file line numberDiff line numberDiff line change
@@ -7,23 +7,9 @@ aux_source_directory(libsolidity SRC_LIST)
77
aux_source_directory(contracts SRC_LIST)
88
aux_source_directory(liblll SRC_LIST)
99

10-
get_filename_component(TESTS_DIR "${CMAKE_CURRENT_SOURCE_DIR}" ABSOLUTE)
10+
list(REMOVE_ITEM SRC_LIST "./fuzzer.cpp")
1111

12-
# search for test names and create ctest tests
13-
enable_testing()
14-
foreach(file ${SRC_LIST})
15-
file(STRINGS ${CMAKE_CURRENT_SOURCE_DIR}/${file} test_list_raw REGEX "BOOST_.*TEST_(SUITE|CASE)")
16-
set(TestSuite "DEFAULT")
17-
foreach(test_raw ${test_list_raw})
18-
string(REGEX REPLACE ".*TEST_(SUITE|CASE)\\(([^ ,\\)]*).*" "\\1 \\2" test ${test_raw})
19-
if(test MATCHES "^SUITE .*")
20-
string(SUBSTRING ${test} 6 -1 TestSuite)
21-
elseif(test MATCHES "^CASE .*")
22-
string(SUBSTRING ${test} 5 -1 TestCase)
23-
add_test(NAME ${TestSuite}/${TestCase} WORKING_DIRECTORY ${CMAKE_BINARY_DIR}/test COMMAND test -t ${TestSuite}/${TestCase})
24-
endif(test MATCHES "^SUITE .*")
25-
endforeach(test_raw)
26-
endforeach(file)
12+
get_filename_component(TESTS_DIR "${CMAKE_CURRENT_SOURCE_DIR}" ABSOLUTE)
2713

2814
file(GLOB HEADERS "*.h" "*/*.h")
2915
set(EXECUTABLE soltest)
@@ -34,5 +20,5 @@ eth_use(${EXECUTABLE} REQUIRED Solidity::solidity Solidity::lll)
3420
include_directories(BEFORE ..)
3521
target_link_libraries(${EXECUTABLE} ${Boost_UNIT_TEST_FRAMEWORK_LIBRARIES})
3622

37-
enable_testing()
38-
set(CTEST_OUTPUT_ON_FAILURE TRUE)
23+
add_executable(solfuzzer fuzzer.cpp)
24+
target_link_libraries(solfuzzer soljson)

‎test/cmdlineTests.sh

+19-4
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ set -e
3131
REPO_ROOT="$(dirname "$0")"/..
3232
SOLC="$REPO_ROOT/build/solc/solc"
3333

34-
# Compile all files in std and examples.
34+
# Compile all files in std and examples.
3535

3636
for f in "$REPO_ROOT"/std/*.sol
3737
do
@@ -46,6 +46,21 @@ do
4646
test -z "$output" -a "$failed" -eq 0
4747
done
4848

49-
# Test library checksum
50-
echo 'contact C {}' | "$SOLC" --link --libraries a:0x90f20564390eAe531E810af625A22f51385Cd222
51-
! echo 'contract C {}' | "$SOLC" --link --libraries a:0x80f20564390eAe531E810af625A22f51385Cd222 2>/dev/null
49+
echo "Testing library checksum..."
50+
echo '' | "$SOLC" --link --libraries a:0x90f20564390eAe531E810af625A22f51385Cd222
51+
! echo '' | "$SOLC" --link --libraries a:0x80f20564390eAe531E810af625A22f51385Cd222 2>/dev/null
52+
53+
echo "Testing soljson via the fuzzer..."
54+
TMPDIR=$(mktemp -d)
55+
(
56+
cd "$REPO_ROOT"
57+
REPO_ROOT=$(pwd) # make it absolute
58+
cd "$TMPDIR"
59+
"$REPO_ROOT"/scripts/isolate_tests.py "$REPO_ROOT"/test/contracts/* "$REPO_ROOT"/test/libsolidity/*EndToEnd*
60+
for f in *.sol
61+
do
62+
"$REPO_ROOT"/build/test/solfuzzer < "$f"
63+
done
64+
)
65+
rm -rf "$TMPDIR"
66+
echo "Done."

‎test/fuzzer.cpp

+92
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
/*
2+
This file is part of solidity.
3+
4+
solidity is free software: you can redistribute it and/or modify
5+
it under the terms of the GNU General Public License as published by
6+
the Free Software Foundation, either version 3 of the License, or
7+
(at your option) any later version.
8+
9+
solidity is distributed in the hope that it will be useful,
10+
but WITHOUT ANY WARRANTY; without even the implied warranty of
11+
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12+
GNU General Public License for more details.
13+
14+
You should have received a copy of the GNU General Public License
15+
along with solidity. If not, see <http://www.gnu.org/licenses/>.
16+
*/
17+
/**
18+
* Executable for use with AFL <http://lcamtuf.coredump.cx/afl>.
19+
* Reads a single source from stdin and signals a failure for internal errors.
20+
*/
21+
22+
#include <json/json.h>
23+
24+
#include <string>
25+
#include <iostream>
26+
27+
using namespace std;
28+
29+
extern "C"
30+
{
31+
extern char const* compileJSON(char const* _input, bool _optimize);
32+
}
33+
34+
string contains(string const& _haystack, vector<string> const& _needles)
35+
{
36+
for (string const& needle: _needles)
37+
if (_haystack.find(needle) != string::npos)
38+
return needle;
39+
return "";
40+
}
41+
42+
int main()
43+
{
44+
string input;
45+
while (!cin.eof())
46+
{
47+
string s;
48+
getline(cin, s);
49+
input += s + '\n';
50+
}
51+
52+
bool optimize = true;
53+
string outputString(compileJSON(input.c_str(), optimize));
54+
Json::Value outputJson;
55+
if (!Json::Reader().parse(outputString, outputJson))
56+
{
57+
cout << "Compiler produced invalid JSON output." << endl;
58+
abort();
59+
}
60+
if (outputJson.isMember("errors"))
61+
{
62+
if (!outputJson["errors"].isArray())
63+
{
64+
cout << "Output JSON has \"errors\" but it is not an array." << endl;
65+
abort();
66+
}
67+
for (Json::Value const& error: outputJson["errors"])
68+
{
69+
string invalid = contains(error.asString(), vector<string>{
70+
"Compiler error",
71+
"Internal compiler error",
72+
"Exception during compilation",
73+
"Unknown exception during compilation",
74+
"Unknown exception while generating contract data output",
75+
"Unknown exception while generating formal method output",
76+
"Unknown exception while generating source name output",
77+
"Unknown error while generating JSON"
78+
});
79+
if (!invalid.empty())
80+
{
81+
cout << "Invalid error: \"" << invalid << "\"" << endl;
82+
abort();
83+
}
84+
}
85+
}
86+
else if (!outputJson.isMember("contracts"))
87+
{
88+
cout << "Output JSON has neither \"errors\" nor \"contracts\"." << endl;
89+
abort();
90+
}
91+
return 0;
92+
}

0 commit comments

Comments
 (0)