1 // Protocol Buffers - Google's data interchange format
2 // Copyright 2008 Google Inc. All rights reserved.
3 // https://developers.google.com/protocol-buffers/
4 //
5 // Redistribution and use in source and binary forms, with or without
6 // modification, are permitted provided that the following conditions are
7 // met:
8 //
9 // * Redistributions of source code must retain the above copyright
10 // notice, this list of conditions and the following disclaimer.
11 // * Redistributions in binary form must reproduce the above
12 // copyright notice, this list of conditions and the following disclaimer
13 // in the documentation and/or other materials provided with the
14 // distribution.
15 // * Neither the name of Google Inc. nor the names of its
16 // contributors may be used to endorse or promote products derived from
17 // this software without specific prior written permission.
18 //
19 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
31 // Author: kenton@google.com (Kenton Varda)
32 // Based on original Protocol Buffers design by
33 // Sanjay Ghemawat, Jeff Dean, and others.
34
35 #include <google/protobuf/compiler/importer.h>
36
37 #include <memory>
38 #include <unordered_map>
39
40 #include <google/protobuf/stubs/logging.h>
41 #include <google/protobuf/stubs/common.h>
42 #include <google/protobuf/testing/file.h>
43 #include <google/protobuf/testing/file.h>
44 #include <google/protobuf/testing/file.h>
45 #include <google/protobuf/io/zero_copy_stream_impl.h>
46 #include <google/protobuf/descriptor.h>
47 #include <google/protobuf/stubs/substitute.h>
48 #include <google/protobuf/testing/googletest.h>
49 #include <gtest/gtest.h>
50 #include <google/protobuf/stubs/map_util.h>
51 #include <google/protobuf/stubs/strutil.h>
52
53 namespace google {
54 namespace protobuf {
55 namespace compiler {
56
57 namespace {
58
FileExists(const std::string & path)59 bool FileExists(const std::string& path) {
60 return File::Exists(path);
61 }
62
63 #define EXPECT_SUBSTRING(needle, haystack) \
64 EXPECT_PRED_FORMAT2(testing::IsSubstring, (needle), (haystack))
65
66 class MockErrorCollector : public MultiFileErrorCollector {
67 public:
MockErrorCollector()68 MockErrorCollector() {}
~MockErrorCollector()69 ~MockErrorCollector() {}
70
71 std::string text_;
72 std::string warning_text_;
73
74 // implements ErrorCollector ---------------------------------------
AddError(const std::string & filename,int line,int column,const std::string & message)75 void AddError(const std::string& filename, int line, int column,
76 const std::string& message) {
77 strings::SubstituteAndAppend(&text_, "$0:$1:$2: $3\n", filename, line,
78 column, message);
79 }
80
AddWarning(const std::string & filename,int line,int column,const std::string & message)81 void AddWarning(const std::string& filename, int line, int column,
82 const std::string& message) {
83 strings::SubstituteAndAppend(&warning_text_, "$0:$1:$2: $3\n", filename,
84 line, column, message);
85 }
86 };
87
88 // -------------------------------------------------------------------
89
90 // A dummy implementation of SourceTree backed by a simple map.
91 class MockSourceTree : public SourceTree {
92 public:
MockSourceTree()93 MockSourceTree() {}
~MockSourceTree()94 ~MockSourceTree() {}
95
AddFile(const std::string & name,const char * contents)96 void AddFile(const std::string& name, const char* contents) {
97 files_[name] = contents;
98 }
99
100 // implements SourceTree -------------------------------------------
Open(const std::string & filename)101 io::ZeroCopyInputStream* Open(const std::string& filename) {
102 const char* contents = FindPtrOrNull(files_, filename);
103 if (contents == NULL) {
104 return NULL;
105 } else {
106 return new io::ArrayInputStream(contents, strlen(contents));
107 }
108 }
109
GetLastErrorMessage()110 std::string GetLastErrorMessage() { return "File not found."; }
111
112 private:
113 std::unordered_map<std::string, const char*> files_;
114 };
115
116 // ===================================================================
117
118 class ImporterTest : public testing::Test {
119 protected:
ImporterTest()120 ImporterTest() : importer_(&source_tree_, &error_collector_) {}
121
AddFile(const std::string & filename,const char * text)122 void AddFile(const std::string& filename, const char* text) {
123 source_tree_.AddFile(filename, text);
124 }
125
126 // Return the collected error text
warning() const127 std::string warning() const { return error_collector_.warning_text_; }
128
129 MockErrorCollector error_collector_;
130 MockSourceTree source_tree_;
131 Importer importer_;
132 };
133
TEST_F(ImporterTest,Import)134 TEST_F(ImporterTest, Import) {
135 // Test normal importing.
136 AddFile("foo.proto",
137 "syntax = \"proto2\";\n"
138 "message Foo {}\n");
139
140 const FileDescriptor* file = importer_.Import("foo.proto");
141 EXPECT_EQ("", error_collector_.text_);
142 ASSERT_TRUE(file != NULL);
143
144 ASSERT_EQ(1, file->message_type_count());
145 EXPECT_EQ("Foo", file->message_type(0)->name());
146
147 // Importing again should return same object.
148 EXPECT_EQ(file, importer_.Import("foo.proto"));
149 }
150
TEST_F(ImporterTest,ImportNested)151 TEST_F(ImporterTest, ImportNested) {
152 // Test that importing a file which imports another file works.
153 AddFile("foo.proto",
154 "syntax = \"proto2\";\n"
155 "import \"bar.proto\";\n"
156 "message Foo {\n"
157 " optional Bar bar = 1;\n"
158 "}\n");
159 AddFile("bar.proto",
160 "syntax = \"proto2\";\n"
161 "message Bar {}\n");
162
163 // Note that both files are actually parsed by the first call to Import()
164 // here, since foo.proto imports bar.proto. The second call just returns
165 // the same ProtoFile for bar.proto which was constructed while importing
166 // foo.proto. We test that this is the case below by checking that bar
167 // is among foo's dependencies (by pointer).
168 const FileDescriptor* foo = importer_.Import("foo.proto");
169 const FileDescriptor* bar = importer_.Import("bar.proto");
170 EXPECT_EQ("", error_collector_.text_);
171 ASSERT_TRUE(foo != NULL);
172 ASSERT_TRUE(bar != NULL);
173
174 // Check that foo's dependency is the same object as bar.
175 ASSERT_EQ(1, foo->dependency_count());
176 EXPECT_EQ(bar, foo->dependency(0));
177
178 // Check that foo properly cross-links bar.
179 ASSERT_EQ(1, foo->message_type_count());
180 ASSERT_EQ(1, bar->message_type_count());
181 ASSERT_EQ(1, foo->message_type(0)->field_count());
182 ASSERT_EQ(FieldDescriptor::TYPE_MESSAGE,
183 foo->message_type(0)->field(0)->type());
184 EXPECT_EQ(bar->message_type(0),
185 foo->message_type(0)->field(0)->message_type());
186 }
187
TEST_F(ImporterTest,FileNotFound)188 TEST_F(ImporterTest, FileNotFound) {
189 // Error: Parsing a file that doesn't exist.
190 EXPECT_TRUE(importer_.Import("foo.proto") == NULL);
191 EXPECT_EQ("foo.proto:-1:0: File not found.\n", error_collector_.text_);
192 }
193
TEST_F(ImporterTest,ImportNotFound)194 TEST_F(ImporterTest, ImportNotFound) {
195 // Error: Importing a file that doesn't exist.
196 AddFile("foo.proto",
197 "syntax = \"proto2\";\n"
198 "import \"bar.proto\";\n");
199
200 EXPECT_TRUE(importer_.Import("foo.proto") == NULL);
201 EXPECT_EQ(
202 "bar.proto:-1:0: File not found.\n"
203 "foo.proto:1:0: Import \"bar.proto\" was not found or had errors.\n",
204 error_collector_.text_);
205 }
206
TEST_F(ImporterTest,RecursiveImport)207 TEST_F(ImporterTest, RecursiveImport) {
208 // Error: Recursive import.
209 AddFile("recursive1.proto",
210 "syntax = \"proto2\";\n"
211 "\n"
212 "import \"recursive2.proto\";\n");
213 AddFile("recursive2.proto",
214 "syntax = \"proto2\";\n"
215 "import \"recursive1.proto\";\n");
216
217 EXPECT_TRUE(importer_.Import("recursive1.proto") == NULL);
218 EXPECT_EQ(
219 "recursive1.proto:2:0: File recursively imports itself: "
220 "recursive1.proto "
221 "-> recursive2.proto -> recursive1.proto\n"
222 "recursive2.proto:1:0: Import \"recursive1.proto\" was not found "
223 "or had errors.\n"
224 "recursive1.proto:2:0: Import \"recursive2.proto\" was not found "
225 "or had errors.\n",
226 error_collector_.text_);
227 }
228
TEST_F(ImporterTest,RecursiveImportSelf)229 TEST_F(ImporterTest, RecursiveImportSelf) {
230 // Error: Recursive import.
231 AddFile("recursive.proto",
232 "syntax = \"proto2\";\n"
233 "\n"
234 "import \"recursive.proto\";\n");
235
236 EXPECT_TRUE(importer_.Import("recursive.proto") == nullptr);
237 EXPECT_EQ(
238 "recursive.proto:2:0: File recursively imports itself: "
239 "recursive.proto -> recursive.proto\n",
240 error_collector_.text_);
241 }
242
TEST_F(ImporterTest,LiteRuntimeImport)243 TEST_F(ImporterTest, LiteRuntimeImport) {
244 // Error: Recursive import.
245 AddFile("bar.proto",
246 "syntax = \"proto2\";\n"
247 "option optimize_for = LITE_RUNTIME;\n");
248 AddFile("foo.proto",
249 "syntax = \"proto2\";\n"
250 "import \"bar.proto\";\n");
251
252 EXPECT_TRUE(importer_.Import("foo.proto") == nullptr);
253 EXPECT_EQ(
254 "foo.proto:1:0: Files that do not use optimize_for = LITE_RUNTIME "
255 "cannot import files which do use this option. This file is not "
256 "lite, but it imports \"bar.proto\" which is.\n",
257 error_collector_.text_);
258 }
259
260
261 // ===================================================================
262
263 class DiskSourceTreeTest : public testing::Test {
264 protected:
SetUp()265 virtual void SetUp() {
266 dirnames_.push_back(TestTempDir() + "/test_proto2_import_path_1");
267 dirnames_.push_back(TestTempDir() + "/test_proto2_import_path_2");
268
269 for (int i = 0; i < dirnames_.size(); i++) {
270 if (FileExists(dirnames_[i])) {
271 File::DeleteRecursively(dirnames_[i], NULL, NULL);
272 }
273 GOOGLE_CHECK_OK(File::CreateDir(dirnames_[i], 0777));
274 }
275 }
276
TearDown()277 virtual void TearDown() {
278 for (int i = 0; i < dirnames_.size(); i++) {
279 if (FileExists(dirnames_[i])) {
280 File::DeleteRecursively(dirnames_[i], NULL, NULL);
281 }
282 }
283 }
284
AddFile(const std::string & filename,const char * contents)285 void AddFile(const std::string& filename, const char* contents) {
286 GOOGLE_CHECK_OK(File::SetContents(filename, contents, true));
287 }
288
AddSubdir(const std::string & dirname)289 void AddSubdir(const std::string& dirname) {
290 GOOGLE_CHECK_OK(File::CreateDir(dirname, 0777));
291 }
292
ExpectFileContents(const std::string & filename,const char * expected_contents)293 void ExpectFileContents(const std::string& filename,
294 const char* expected_contents) {
295 std::unique_ptr<io::ZeroCopyInputStream> input(source_tree_.Open(filename));
296
297 ASSERT_FALSE(input == NULL);
298
299 // Read all the data from the file.
300 std::string file_contents;
301 const void* data;
302 int size;
303 while (input->Next(&data, &size)) {
304 file_contents.append(reinterpret_cast<const char*>(data), size);
305 }
306
307 EXPECT_EQ(expected_contents, file_contents);
308 }
309
ExpectCannotOpenFile(const std::string & filename,const std::string & error_message)310 void ExpectCannotOpenFile(const std::string& filename,
311 const std::string& error_message) {
312 std::unique_ptr<io::ZeroCopyInputStream> input(source_tree_.Open(filename));
313 EXPECT_TRUE(input == NULL);
314 EXPECT_EQ(error_message, source_tree_.GetLastErrorMessage());
315 }
316
317 DiskSourceTree source_tree_;
318
319 // Paths of two on-disk directories to use during the test.
320 std::vector<std::string> dirnames_;
321 };
322
TEST_F(DiskSourceTreeTest,MapRoot)323 TEST_F(DiskSourceTreeTest, MapRoot) {
324 // Test opening a file in a directory that is mapped to the root of the
325 // source tree.
326 AddFile(dirnames_[0] + "/foo", "Hello World!");
327 source_tree_.MapPath("", dirnames_[0]);
328
329 ExpectFileContents("foo", "Hello World!");
330 ExpectCannotOpenFile("bar", "File not found.");
331 }
332
TEST_F(DiskSourceTreeTest,MapDirectory)333 TEST_F(DiskSourceTreeTest, MapDirectory) {
334 // Test opening a file in a directory that is mapped to somewhere other
335 // than the root of the source tree.
336
337 AddFile(dirnames_[0] + "/foo", "Hello World!");
338 source_tree_.MapPath("baz", dirnames_[0]);
339
340 ExpectFileContents("baz/foo", "Hello World!");
341 ExpectCannotOpenFile("baz/bar", "File not found.");
342 ExpectCannotOpenFile("foo", "File not found.");
343 ExpectCannotOpenFile("bar", "File not found.");
344
345 // Non-canonical file names should not work.
346 ExpectCannotOpenFile("baz//foo",
347 "Backslashes, consecutive slashes, \".\", or \"..\" are "
348 "not allowed in the virtual path");
349 ExpectCannotOpenFile("baz/../baz/foo",
350 "Backslashes, consecutive slashes, \".\", or \"..\" are "
351 "not allowed in the virtual path");
352 ExpectCannotOpenFile("baz/./foo",
353 "Backslashes, consecutive slashes, \".\", or \"..\" are "
354 "not allowed in the virtual path");
355 ExpectCannotOpenFile("baz/foo/", "File not found.");
356 }
357
TEST_F(DiskSourceTreeTest,NoParent)358 TEST_F(DiskSourceTreeTest, NoParent) {
359 // Test that we cannot open files in a parent of a mapped directory.
360
361 AddFile(dirnames_[0] + "/foo", "Hello World!");
362 AddSubdir(dirnames_[0] + "/bar");
363 AddFile(dirnames_[0] + "/bar/baz", "Blah.");
364 source_tree_.MapPath("", dirnames_[0] + "/bar");
365
366 ExpectFileContents("baz", "Blah.");
367 ExpectCannotOpenFile("../foo",
368 "Backslashes, consecutive slashes, \".\", or \"..\" are "
369 "not allowed in the virtual path");
370 ExpectCannotOpenFile("../bar/baz",
371 "Backslashes, consecutive slashes, \".\", or \"..\" are "
372 "not allowed in the virtual path");
373 }
374
TEST_F(DiskSourceTreeTest,MapFile)375 TEST_F(DiskSourceTreeTest, MapFile) {
376 // Test opening a file that is mapped directly into the source tree.
377
378 AddFile(dirnames_[0] + "/foo", "Hello World!");
379 source_tree_.MapPath("foo", dirnames_[0] + "/foo");
380
381 ExpectFileContents("foo", "Hello World!");
382 ExpectCannotOpenFile("bar", "File not found.");
383 }
384
TEST_F(DiskSourceTreeTest,SearchMultipleDirectories)385 TEST_F(DiskSourceTreeTest, SearchMultipleDirectories) {
386 // Test mapping and searching multiple directories.
387
388 AddFile(dirnames_[0] + "/foo", "Hello World!");
389 AddFile(dirnames_[1] + "/foo", "This file should be hidden.");
390 AddFile(dirnames_[1] + "/bar", "Goodbye World!");
391 source_tree_.MapPath("", dirnames_[0]);
392 source_tree_.MapPath("", dirnames_[1]);
393
394 ExpectFileContents("foo", "Hello World!");
395 ExpectFileContents("bar", "Goodbye World!");
396 ExpectCannotOpenFile("baz", "File not found.");
397 }
398
TEST_F(DiskSourceTreeTest,OrderingTrumpsSpecificity)399 TEST_F(DiskSourceTreeTest, OrderingTrumpsSpecificity) {
400 // Test that directories are always searched in order, even when a latter
401 // directory is more-specific than a former one.
402
403 // Create the "bar" directory so we can put a file in it.
404 GOOGLE_CHECK_OK(File::CreateDir(dirnames_[0] + "/bar", 0777));
405
406 // Add files and map paths.
407 AddFile(dirnames_[0] + "/bar/foo", "Hello World!");
408 AddFile(dirnames_[1] + "/foo", "This file should be hidden.");
409 source_tree_.MapPath("", dirnames_[0]);
410 source_tree_.MapPath("bar", dirnames_[1]);
411
412 // Check.
413 ExpectFileContents("bar/foo", "Hello World!");
414 }
415
TEST_F(DiskSourceTreeTest,DiskFileToVirtualFile)416 TEST_F(DiskSourceTreeTest, DiskFileToVirtualFile) {
417 // Test DiskFileToVirtualFile.
418
419 AddFile(dirnames_[0] + "/foo", "Hello World!");
420 AddFile(dirnames_[1] + "/foo", "This file should be hidden.");
421 source_tree_.MapPath("bar", dirnames_[0]);
422 source_tree_.MapPath("bar", dirnames_[1]);
423
424 std::string virtual_file;
425 std::string shadowing_disk_file;
426
427 EXPECT_EQ(DiskSourceTree::NO_MAPPING,
428 source_tree_.DiskFileToVirtualFile("/foo", &virtual_file,
429 &shadowing_disk_file));
430
431 EXPECT_EQ(DiskSourceTree::SHADOWED,
432 source_tree_.DiskFileToVirtualFile(
433 dirnames_[1] + "/foo", &virtual_file, &shadowing_disk_file));
434 EXPECT_EQ("bar/foo", virtual_file);
435 EXPECT_EQ(dirnames_[0] + "/foo", shadowing_disk_file);
436
437 EXPECT_EQ(DiskSourceTree::CANNOT_OPEN,
438 source_tree_.DiskFileToVirtualFile(
439 dirnames_[1] + "/baz", &virtual_file, &shadowing_disk_file));
440 EXPECT_EQ("bar/baz", virtual_file);
441
442 EXPECT_EQ(DiskSourceTree::SUCCESS,
443 source_tree_.DiskFileToVirtualFile(
444 dirnames_[0] + "/foo", &virtual_file, &shadowing_disk_file));
445 EXPECT_EQ("bar/foo", virtual_file);
446 }
447
TEST_F(DiskSourceTreeTest,DiskFileToVirtualFileCanonicalization)448 TEST_F(DiskSourceTreeTest, DiskFileToVirtualFileCanonicalization) {
449 // Test handling of "..", ".", etc. in DiskFileToVirtualFile().
450
451 source_tree_.MapPath("dir1", "..");
452 source_tree_.MapPath("dir2", "../../foo");
453 source_tree_.MapPath("dir3", "./foo/bar/.");
454 source_tree_.MapPath("dir4", ".");
455 source_tree_.MapPath("", "/qux");
456 source_tree_.MapPath("dir5", "/quux/");
457
458 std::string virtual_file;
459 std::string shadowing_disk_file;
460
461 // "../.." should not be considered to be under "..".
462 EXPECT_EQ(DiskSourceTree::NO_MAPPING,
463 source_tree_.DiskFileToVirtualFile("../../baz", &virtual_file,
464 &shadowing_disk_file));
465
466 // "/foo" is not mapped (it should not be misintepreted as being under ".").
467 EXPECT_EQ(DiskSourceTree::NO_MAPPING,
468 source_tree_.DiskFileToVirtualFile("/foo", &virtual_file,
469 &shadowing_disk_file));
470
471 #ifdef WIN32
472 // "C:\foo" is not mapped (it should not be misintepreted as being under ".").
473 EXPECT_EQ(DiskSourceTree::NO_MAPPING,
474 source_tree_.DiskFileToVirtualFile("C:\\foo", &virtual_file,
475 &shadowing_disk_file));
476 #endif // WIN32
477
478 // But "../baz" should be.
479 EXPECT_EQ(DiskSourceTree::CANNOT_OPEN,
480 source_tree_.DiskFileToVirtualFile("../baz", &virtual_file,
481 &shadowing_disk_file));
482 EXPECT_EQ("dir1/baz", virtual_file);
483
484 // "../../foo/baz" is under "../../foo".
485 EXPECT_EQ(DiskSourceTree::CANNOT_OPEN,
486 source_tree_.DiskFileToVirtualFile("../../foo/baz", &virtual_file,
487 &shadowing_disk_file));
488 EXPECT_EQ("dir2/baz", virtual_file);
489
490 // "foo/./bar/baz" is under "./foo/bar/.".
491 EXPECT_EQ(DiskSourceTree::CANNOT_OPEN,
492 source_tree_.DiskFileToVirtualFile("foo/bar/baz", &virtual_file,
493 &shadowing_disk_file));
494 EXPECT_EQ("dir3/baz", virtual_file);
495
496 // "bar" is under ".".
497 EXPECT_EQ(DiskSourceTree::CANNOT_OPEN,
498 source_tree_.DiskFileToVirtualFile("bar", &virtual_file,
499 &shadowing_disk_file));
500 EXPECT_EQ("dir4/bar", virtual_file);
501
502 // "/qux/baz" is under "/qux".
503 EXPECT_EQ(DiskSourceTree::CANNOT_OPEN,
504 source_tree_.DiskFileToVirtualFile("/qux/baz", &virtual_file,
505 &shadowing_disk_file));
506 EXPECT_EQ("baz", virtual_file);
507
508 // "/quux/bar" is under "/quux".
509 EXPECT_EQ(DiskSourceTree::CANNOT_OPEN,
510 source_tree_.DiskFileToVirtualFile("/quux/bar", &virtual_file,
511 &shadowing_disk_file));
512 EXPECT_EQ("dir5/bar", virtual_file);
513 }
514
TEST_F(DiskSourceTreeTest,VirtualFileToDiskFile)515 TEST_F(DiskSourceTreeTest, VirtualFileToDiskFile) {
516 // Test VirtualFileToDiskFile.
517
518 AddFile(dirnames_[0] + "/foo", "Hello World!");
519 AddFile(dirnames_[1] + "/foo", "This file should be hidden.");
520 AddFile(dirnames_[1] + "/quux", "This file should not be hidden.");
521 source_tree_.MapPath("bar", dirnames_[0]);
522 source_tree_.MapPath("bar", dirnames_[1]);
523
524 // Existent files, shadowed and non-shadowed case.
525 std::string disk_file;
526 EXPECT_TRUE(source_tree_.VirtualFileToDiskFile("bar/foo", &disk_file));
527 EXPECT_EQ(dirnames_[0] + "/foo", disk_file);
528 EXPECT_TRUE(source_tree_.VirtualFileToDiskFile("bar/quux", &disk_file));
529 EXPECT_EQ(dirnames_[1] + "/quux", disk_file);
530
531 // Nonexistent file in existent directory and vice versa.
532 std::string not_touched = "not touched";
533 EXPECT_FALSE(source_tree_.VirtualFileToDiskFile("bar/baz", ¬_touched));
534 EXPECT_EQ("not touched", not_touched);
535 EXPECT_FALSE(source_tree_.VirtualFileToDiskFile("baz/foo", ¬_touched));
536 EXPECT_EQ("not touched", not_touched);
537
538 // Accept NULL as output parameter.
539 EXPECT_TRUE(source_tree_.VirtualFileToDiskFile("bar/foo", NULL));
540 EXPECT_FALSE(source_tree_.VirtualFileToDiskFile("baz/foo", NULL));
541 }
542
543 } // namespace
544
545 } // namespace compiler
546 } // namespace protobuf
547 } // namespace google
548