1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
11 #include "mojo/public/c/system/macros.h"
12 #include "mojo/public/cpp/bindings/interface_impl.h"
13 #include "mojo/public/cpp/bindings/interface_ptr.h"
14 #include "mojo/public/cpp/bindings/lib/connector.h"
15 #include "mojo/public/cpp/bindings/lib/filter_chain.h"
16 #include "mojo/public/cpp/bindings/lib/message_header_validator.h"
17 #include "mojo/public/cpp/bindings/lib/router.h"
18 #include "mojo/public/cpp/bindings/lib/validation_errors.h"
19 #include "mojo/public/cpp/bindings/message.h"
20 #include "mojo/public/cpp/bindings/tests/validation_test_input_parser.h"
21 #include "mojo/public/cpp/environment/environment.h"
22 #include "mojo/public/cpp/system/core.h"
23 #include "mojo/public/cpp/test_support/test_support.h"
24 #include "mojo/public/cpp/utility/run_loop.h"
25 #include "mojo/public/interfaces/bindings/tests/validation_test_interfaces.mojom.h"
26 #include "testing/gtest/include/gtest/gtest.h"
33 void Append(std::vector
<uint8_t>* data_vector
, T data
) {
34 size_t pos
= data_vector
->size();
35 data_vector
->resize(pos
+ sizeof(T
));
36 memcpy(&(*data_vector
)[pos
], &data
, sizeof(T
));
39 bool TestInputParser(const std::string
& input
,
41 const std::vector
<uint8_t>& expected_data
,
42 size_t expected_num_handles
) {
43 std::vector
<uint8_t> data
;
45 std::string error_message
;
47 bool result
= ParseValidationTestInput(input
, &data
, &num_handles
,
49 if (expected_result
) {
50 if (result
&& error_message
.empty() &&
51 expected_data
== data
&& expected_num_handles
== num_handles
) {
55 // Compare with an empty string instead of checking |error_message.empty()|,
56 // so that the message will be printed out if the two are not equal.
57 EXPECT_EQ(std::string(), error_message
);
58 EXPECT_EQ(expected_data
, data
);
59 EXPECT_EQ(expected_num_handles
, num_handles
);
63 EXPECT_FALSE(error_message
.empty());
64 return !result
&& !error_message
.empty();
67 std::vector
<std::string
> GetMatchingTests(const std::vector
<std::string
>& names
,
68 const std::string
& prefix
) {
69 const std::string suffix
= ".data";
70 std::vector
<std::string
> tests
;
71 for (size_t i
= 0; i
< names
.size(); ++i
) {
72 if (names
[i
].size() >= suffix
.size() &&
73 names
[i
].substr(0, prefix
.size()) == prefix
&&
74 names
[i
].substr(names
[i
].size() - suffix
.size()) == suffix
)
75 tests
.push_back(names
[i
].substr(0, names
[i
].size() - suffix
.size()));
80 bool ReadFile(const std::string
& path
, std::string
* result
) {
81 FILE* fp
= OpenSourceRootRelativeFile(path
.c_str());
83 ADD_FAILURE() << "File not found: " << path
;
86 fseek(fp
, 0, SEEK_END
);
87 size_t size
= static_cast<size_t>(ftell(fp
));
93 fseek(fp
, 0, SEEK_SET
);
95 size_t size_read
= fread(&result
->at(0), 1, size
, fp
);
97 return size
== size_read
;
100 bool ReadAndParseDataFile(const std::string
& path
,
101 std::vector
<uint8_t>* data
,
102 size_t* num_handles
) {
104 if (!ReadFile(path
, &input
))
107 std::string error_message
;
108 if (!ParseValidationTestInput(input
, data
, num_handles
, &error_message
)) {
109 ADD_FAILURE() << error_message
;
116 bool ReadResultFile(const std::string
& path
, std::string
* result
) {
117 if (!ReadFile(path
, result
))
120 // Result files are new-line delimited text files. Remove any CRs.
121 result
->erase(std::remove(result
->begin(), result
->end(), '\r'),
124 // Remove trailing LFs.
125 size_t pos
= result
->find_last_not_of('\n');
126 if (pos
== std::string::npos
)
129 result
->resize(pos
+ 1);
134 std::string
GetPath(const std::string
& root
, const std::string
& suffix
) {
135 return "mojo/public/interfaces/bindings/tests/data/validation/" +
139 // |message| should be a newly created object.
140 bool ReadTestCase(const std::string
& test
,
142 std::string
* expected
) {
143 std::vector
<uint8_t> data
;
145 if (!ReadAndParseDataFile(GetPath(test
, ".data"), &data
, &num_handles
) ||
146 !ReadResultFile(GetPath(test
, ".expected"), expected
)) {
150 message
->AllocUninitializedData(static_cast<uint32_t>(data
.size()));
152 memcpy(message
->mutable_data(), &data
[0], data
.size());
153 message
->mutable_handles()->resize(num_handles
);
158 void RunValidationTests(const std::string
& prefix
,
159 MessageReceiver
* test_message_receiver
) {
160 std::vector
<std::string
> names
=
161 EnumerateSourceRootRelativeDirectory(GetPath("", ""));
162 std::vector
<std::string
> tests
= GetMatchingTests(names
, prefix
);
164 for (size_t i
= 0; i
< tests
.size(); ++i
) {
166 std::string expected
;
167 ASSERT_TRUE(ReadTestCase(tests
[i
], &message
, &expected
));
170 mojo::internal::ValidationErrorObserverForTesting observer
;
171 bool unused MOJO_ALLOW_UNUSED
= test_message_receiver
->Accept(&message
);
172 if (observer
.last_error() == mojo::internal::VALIDATION_ERROR_NONE
)
175 result
= mojo::internal::ValidationErrorToString(observer
.last_error());
177 EXPECT_EQ(expected
, result
) << "failed test: " << tests
[i
];
181 class DummyMessageReceiver
: public MessageReceiver
{
183 virtual bool Accept(Message
* message
) MOJO_OVERRIDE
{
184 return true; // Any message is OK.
188 class ValidationTest
: public testing::Test
{
190 virtual ~ValidationTest() {
197 class ValidationIntegrationTest
: public ValidationTest
{
199 ValidationIntegrationTest() : test_message_receiver_(NULL
) {
202 virtual ~ValidationIntegrationTest() {
205 virtual void SetUp() MOJO_OVERRIDE
{
206 ScopedMessagePipeHandle tester_endpoint
;
207 ASSERT_EQ(MOJO_RESULT_OK
,
208 CreateMessagePipe(NULL
, &tester_endpoint
, &testee_endpoint_
));
209 test_message_receiver_
=
210 new TestMessageReceiver(this, tester_endpoint
.Pass());
213 virtual void TearDown() MOJO_OVERRIDE
{
214 delete test_message_receiver_
;
215 test_message_receiver_
= NULL
;
217 // Make sure that the other end receives the OnConnectionError()
222 MessageReceiver
* test_message_receiver() {
223 return test_message_receiver_
;
226 ScopedMessagePipeHandle
testee_endpoint() {
227 return testee_endpoint_
.Pass();
231 class TestMessageReceiver
: public MessageReceiver
{
233 TestMessageReceiver(ValidationIntegrationTest
* owner
,
234 ScopedMessagePipeHandle handle
)
236 connector_(handle
.Pass()) {
238 virtual ~TestMessageReceiver() {
241 virtual bool Accept(Message
* message
) MOJO_OVERRIDE
{
242 bool rv
= connector_
.Accept(message
);
243 owner_
->PumpMessages();
248 ValidationIntegrationTest
* owner_
;
249 mojo::internal::Connector connector_
;
252 void PumpMessages() {
253 loop_
.RunUntilIdle();
257 TestMessageReceiver
* test_message_receiver_
;
258 ScopedMessagePipeHandle testee_endpoint_
;
261 class IntegrationTestInterface1Client
: public IntegrationTestInterface1
{
263 virtual ~IntegrationTestInterface1Client() {
266 virtual void Method0(BasicStructPtr param0
) MOJO_OVERRIDE
{
270 class IntegrationTestInterface1Impl
271 : public InterfaceImpl
<IntegrationTestInterface1
> {
273 virtual ~IntegrationTestInterface1Impl() {
276 virtual void Method0(BasicStructPtr param0
) MOJO_OVERRIDE
{
280 TEST_F(ValidationTest
, InputParser
) {
282 // The parser, as well as Append() defined above, assumes that this code is
283 // running on a little-endian platform. Test whether that is true.
285 ASSERT_EQ(1, *(reinterpret_cast<char*>(&x
)));
290 std::vector
<uint8_t> expected
;
292 EXPECT_TRUE(TestInputParser(input
, true, expected
, 0));
295 // Test input that only consists of comments and whitespaces.
296 std::string input
= " \t // hello world \n\r \t// the answer is 42 ";
297 std::vector
<uint8_t> expected
;
299 EXPECT_TRUE(TestInputParser(input
, true, expected
, 0));
302 std::string input
= "[u1]0x10// hello world !! \n\r \t [u2]65535 \n"
303 "[u4]65536 [u8]0xFFFFFFFFFFFFFFFF 0 0Xff";
304 std::vector
<uint8_t> expected
;
305 Append(&expected
, static_cast<uint8_t>(0x10));
306 Append(&expected
, static_cast<uint16_t>(65535));
307 Append(&expected
, static_cast<uint32_t>(65536));
308 Append(&expected
, static_cast<uint64_t>(0xffffffffffffffff));
309 Append(&expected
, static_cast<uint8_t>(0));
310 Append(&expected
, static_cast<uint8_t>(0xff));
312 EXPECT_TRUE(TestInputParser(input
, true, expected
, 0));
315 std::string input
= "[s8]-0x800 [s1]-128\t[s2]+0 [s4]-40";
316 std::vector
<uint8_t> expected
;
317 Append(&expected
, -static_cast<int64_t>(0x800));
318 Append(&expected
, static_cast<int8_t>(-128));
319 Append(&expected
, static_cast<int16_t>(0));
320 Append(&expected
, static_cast<int32_t>(-40));
322 EXPECT_TRUE(TestInputParser(input
, true, expected
, 0));
325 std::string input
= "[b]00001011 [b]10000000 // hello world\r [b]00000000";
326 std::vector
<uint8_t> expected
;
327 Append(&expected
, static_cast<uint8_t>(11));
328 Append(&expected
, static_cast<uint8_t>(128));
329 Append(&expected
, static_cast<uint8_t>(0));
331 EXPECT_TRUE(TestInputParser(input
, true, expected
, 0));
334 std::string input
= "[f]+.3e9 [d]-10.03";
335 std::vector
<uint8_t> expected
;
336 Append(&expected
, +.3e9f
);
337 Append(&expected
, -10.03);
339 EXPECT_TRUE(TestInputParser(input
, true, expected
, 0));
342 std::string input
= "[dist4]foo 0 [dist8]bar 0 [anchr]foo [anchr]bar";
343 std::vector
<uint8_t> expected
;
344 Append(&expected
, static_cast<uint32_t>(14));
345 Append(&expected
, static_cast<uint8_t>(0));
346 Append(&expected
, static_cast<uint64_t>(9));
347 Append(&expected
, static_cast<uint8_t>(0));
349 EXPECT_TRUE(TestInputParser(input
, true, expected
, 0));
352 std::string input
= "// This message has handles! \n[handles]50 [u8]2";
353 std::vector
<uint8_t> expected
;
354 Append(&expected
, static_cast<uint64_t>(2));
356 EXPECT_TRUE(TestInputParser(input
, true, expected
, 50));
359 // Test some failure cases.
361 const char* error_inputs
[] = {
370 "[anchr]hello [dist8]hello",
371 "[dist4]a [dist4]a [anchr]a",
372 "[dist4]a [anchr]a [dist4]a [anchr]a",
377 for (size_t i
= 0; error_inputs
[i
]; ++i
) {
378 std::vector
<uint8_t> expected
;
379 if (!TestInputParser(error_inputs
[i
], false, expected
, 0))
380 ADD_FAILURE() << "Unexpected test result for: " << error_inputs
[i
];
385 TEST_F(ValidationTest
, Conformance
) {
386 DummyMessageReceiver dummy_receiver
;
387 mojo::internal::FilterChain
validators(&dummy_receiver
);
388 validators
.Append
<mojo::internal::MessageHeaderValidator
>();
389 validators
.Append
<ConformanceTestInterface::RequestValidator_
>();
391 RunValidationTests("conformance_", validators
.GetHead());
394 TEST_F(ValidationIntegrationTest
, InterfacePtr
) {
395 // Test that InterfacePtr<X> applies the correct validators and they don't
396 // conflict with each other:
397 // - MessageHeaderValidator
398 // - X::Client::RequestValidator_
399 // - X::ResponseValidator_
401 IntegrationTestInterface1Client interface1_client
;
402 IntegrationTestInterface2Ptr interface2_ptr
=
403 MakeProxy
<IntegrationTestInterface2
>(testee_endpoint().Pass());
404 interface2_ptr
.set_client(&interface1_client
);
405 interface2_ptr
.internal_state()->router_for_testing()->EnableTestingMode();
407 RunValidationTests("integration_", test_message_receiver());
410 TEST_F(ValidationIntegrationTest
, InterfaceImpl
) {
411 // Test that InterfaceImpl<X> applies the correct validators and they don't
412 // conflict with each other:
413 // - MessageHeaderValidator
414 // - X::RequestValidator_
415 // - X::Client::ResponseValidator_
417 // |interface1_impl| will delete itself when the pipe is closed.
418 IntegrationTestInterface1Impl
* interface1_impl
=
419 BindToPipe(new IntegrationTestInterface1Impl(), testee_endpoint().Pass());
420 interface1_impl
->internal_state()->router()->EnableTestingMode();
422 RunValidationTests("integration_", test_message_receiver());