|
12 | 12 | along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
13 | 13 | */
|
14 | 14 |
|
15 |
| -#include <test/libsolidity/SemanticTest.h> |
16 |
| -#include <test/libsolidity/util/BytesUtils.h> |
17 |
| -#include <libsolutil/Whiskers.h> |
18 |
| -#include <libyul/Exceptions.h> |
19 |
| -#include <test/Common.h> |
20 | 15 | #include <boost/algorithm/string.hpp>
|
21 | 16 | #include <boost/algorithm/string/predicate.hpp>
|
22 | 17 | #include <boost/algorithm/string/trim.hpp>
|
23 | 18 | #include <boost/throw_exception.hpp>
|
| 19 | +#include <libsolutil/Whiskers.h> |
| 20 | +#include <libyul/Exceptions.h> |
| 21 | +#include <test/Common.h> |
| 22 | +#include <test/libsolidity/SemanticTest.h> |
| 23 | +#include <test/libsolidity/util/BytesUtils.h> |
| 24 | + |
| 25 | +#include <test/libsolidity/Builtin.h> |
| 26 | +#include <test/libsolidity/TestHook.h> |
| 27 | +#include <test/libsolidity/hooks/SmokeHook.h> |
24 | 28 |
|
25 | 29 | #include <algorithm>
|
26 | 30 | #include <cctype>
|
@@ -50,13 +54,23 @@ SemanticTest::SemanticTest(string const& _filename, langutil::EVMVersion _evmVer
|
50 | 54 | m_enforceViaYul(enforceViaYul)
|
51 | 55 | {
|
52 | 56 | using namespace std::placeholders;
|
53 |
| - m_builtins |
54 |
| - = {{"smoke", |
55 |
| - { |
56 |
| - {"test0", {std::bind(&SemanticTest::builtinSmokeTest, this, _1)}}, |
57 |
| - {"test1", {std::bind(&SemanticTest::builtinSmokeTest, this, _1)}}, |
58 |
| - {"test2", {std::bind(&SemanticTest::builtinSmokeTest, this, _1)}}, |
59 |
| - }}}; |
| 57 | + // clang-format off |
| 58 | + auto simpleSmokeBuiltin = std::make_shared<Builtin>( |
| 59 | + std::bind(&SemanticTest::builtinSmokeTest, this, _1) |
| 60 | + ); |
| 61 | + m_builtins = |
| 62 | + { |
| 63 | + {"smoke", |
| 64 | + {{"test0", simpleSmokeBuiltin}, |
| 65 | + {"test1", simpleSmokeBuiltin}, |
| 66 | + {"test2", simpleSmokeBuiltin}} |
| 67 | + } |
| 68 | + }; |
| 69 | + m_testHooks = |
| 70 | + { |
| 71 | + std::make_shared<SmokeHook>() |
| 72 | + }; |
| 73 | + // clang-format on |
60 | 74 |
|
61 | 75 | string choice = m_reader.stringSetting("compileViaYul", "default");
|
62 | 76 | if (choice == "also")
|
@@ -130,7 +144,13 @@ TestCase::TestResult SemanticTest::run(ostream& _stream, string const& _linePref
|
130 | 144 | return result;
|
131 | 145 | }
|
132 | 146 |
|
133 |
| -TestCase::TestResult SemanticTest::runTest(ostream& _stream, string const& _linePrefix, bool _formatted, bool _compileViaYul, bool _compileToEwasm) |
| 147 | +void SemanticTest::addBuiltin(std::string _module, std::string _function, std::shared_ptr<Builtin> _builtin) |
| 148 | +{ |
| 149 | + m_builtins[_module][_function] = _builtin; |
| 150 | +} |
| 151 | + |
| 152 | +TestCase::TestResult SemanticTest::runTest( |
| 153 | + ostream& _stream, string const& _linePrefix, bool _formatted, bool _compileViaYul, bool _compileToEwasm) |
134 | 154 | {
|
135 | 155 | bool success = true;
|
136 | 156 |
|
@@ -160,11 +180,28 @@ TestCase::TestResult SemanticTest::runTest(ostream& _stream, string const& _line
|
160 | 180 |
|
161 | 181 | bool constructed = false;
|
162 | 182 |
|
| 183 | + // Iterate through the test calls and set the previous call. |
| 184 | + TestFunctionCall* previousCall{nullptr}; |
| 185 | + for (auto& test: m_tests) |
| 186 | + { |
| 187 | + test.setPreviousCall(previousCall); |
| 188 | + test.setTestHooks(&m_testHooks); |
| 189 | + previousCall = &test; |
| 190 | + } |
| 191 | + |
| 192 | + for (auto& hook: m_testHooks) |
| 193 | + hook->beginTestCase(); |
| 194 | + |
163 | 195 | for (auto& test: m_tests)
|
164 | 196 | {
|
| 197 | + for (auto& hook: m_testHooks) |
| 198 | + hook->beforeFunctionCall(test); |
| 199 | + |
165 | 200 | if (constructed)
|
166 | 201 | {
|
167 |
| - soltestAssert(test.call().kind != FunctionCall::Kind::Library, "Libraries have to be deployed before any other call."); |
| 202 | + soltestAssert( |
| 203 | + test.call().kind != FunctionCall::Kind::Library, |
| 204 | + "Libraries have to be deployed before any other call."); |
168 | 205 | soltestAssert(
|
169 | 206 | test.call().kind != FunctionCall::Kind::Constructor,
|
170 | 207 | "Constructor has to be the first function call expect for library deployments.");
|
@@ -214,7 +251,7 @@ TestCase::TestResult SemanticTest::runTest(ostream& _stream, string const& _line
|
214 | 251 | boost::split(builtinPath, test.call().signature, boost::is_any_of("."));
|
215 | 252 | soltestAssert(builtinPath.size() == 2, "");
|
216 | 253 | auto builtin = m_builtins[builtinPath.front()][builtinPath.back()];
|
217 |
| - std::optional<bytes> builtinOutput{builtin.function(test.call())}; |
| 254 | + std::optional<bytes> builtinOutput{builtin->builtin(test.call())}; |
218 | 255 | if (builtinOutput.has_value())
|
219 | 256 | {
|
220 | 257 | test.setFailure(false);
|
@@ -243,9 +280,9 @@ TestCase::TestResult SemanticTest::runTest(ostream& _stream, string const& _line
|
243 | 280 | {
|
244 | 281 | std::vector<string> builtinPath;
|
245 | 282 | boost::split(builtinPath, test.call().expectations.builtin->signature, boost::is_any_of("."));
|
246 |
| - assert(builtinPath.size() == 2); |
| 283 | + soltestAssert(builtinPath.size() == 2, ""); |
247 | 284 | auto builtin = m_builtins[builtinPath.front()][builtinPath.back()];
|
248 |
| - std::optional<bytes> builtinResult = builtin.function(*test.call().expectations.builtin); |
| 285 | + std::optional<bytes> builtinResult = builtin->builtin(*test.call().expectations.builtin); |
249 | 286 | if (builtinResult.has_value())
|
250 | 287 | expectationOutput = builtinResult.value();
|
251 | 288 | else
|
@@ -273,8 +310,29 @@ TestCase::TestResult SemanticTest::runTest(ostream& _stream, string const& _line
|
273 | 310 | test.setRawBytes(std::move(output));
|
274 | 311 | test.setContractABI(m_compiler.contractABI(m_compiler.lastContractName()));
|
275 | 312 | }
|
| 313 | + |
| 314 | + for (auto& hook: m_testHooks) |
| 315 | + hook->afterFunctionCall(test); |
| 316 | + } |
| 317 | + |
| 318 | + // The artificialFunctionCall is an artificially created function call, |
| 319 | + // where it's previous call is pointing to the last call of the test. |
| 320 | + TestFunctionCall artificialFunctionCall(FunctionCall{}); |
| 321 | + artificialFunctionCall.setTestHooks(&m_testHooks); |
| 322 | + artificialFunctionCall.setPreviousCall(previousCall); |
| 323 | + for (auto& hook: m_testHooks) |
| 324 | + { |
| 325 | + hook->beforeFunctionCall(artificialFunctionCall); |
| 326 | + hook->afterFunctionCall(artificialFunctionCall); |
276 | 327 | }
|
277 | 328 |
|
| 329 | + for (auto& hook: m_testHooks) |
| 330 | + hook->endTestCase(); |
| 331 | + |
| 332 | + for (auto& test: m_tests) |
| 333 | + for (auto& hook: m_testHooks) |
| 334 | + success &= hook->verifyFunctionCall(test); |
| 335 | + |
278 | 336 | if (!m_runWithYul && _compileViaYul)
|
279 | 337 | {
|
280 | 338 | m_compileViaYulCanBeSet = success;
|
|
0 commit comments