You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			180 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			180 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===-- TestFS.cpp ----------------------------------------------*- C++ -*-===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | #include "TestFS.h"
 | ||
|  | #include "llvm/Support/Errc.h"
 | ||
|  | #include "gtest/gtest.h"
 | ||
|  | 
 | ||
|  | namespace clang { | ||
|  | namespace clangd { | ||
|  | namespace { | ||
|  | 
 | ||
|  | /// An implementation of vfs::FileSystem that only allows access to
 | ||
|  | /// files and folders inside a set of whitelisted directories.
 | ||
|  | ///
 | ||
|  | /// FIXME(ibiryukov): should it also emulate access to parents of whitelisted
 | ||
|  | /// directories with only whitelisted contents?
 | ||
|  | class FilteredFileSystem : public vfs::FileSystem { | ||
|  | public: | ||
|  |   /// The paths inside \p WhitelistedDirs should be absolute
 | ||
|  |   FilteredFileSystem(std::vector<std::string> WhitelistedDirs, | ||
|  |                      IntrusiveRefCntPtr<vfs::FileSystem> InnerFS) | ||
|  |       : WhitelistedDirs(std::move(WhitelistedDirs)), InnerFS(InnerFS) { | ||
|  |     assert(std::all_of(WhitelistedDirs.begin(), WhitelistedDirs.end(), | ||
|  |                        [](const std::string &Path) -> bool { | ||
|  |                          return llvm::sys::path::is_absolute(Path); | ||
|  |                        }) && | ||
|  |            "Not all WhitelistedDirs are absolute"); | ||
|  |   } | ||
|  | 
 | ||
|  |   virtual llvm::ErrorOr<vfs::Status> status(const Twine &Path) { | ||
|  |     if (!isInsideWhitelistedDir(Path)) | ||
|  |       return llvm::errc::no_such_file_or_directory; | ||
|  |     return InnerFS->status(Path); | ||
|  |   } | ||
|  | 
 | ||
|  |   virtual llvm::ErrorOr<std::unique_ptr<vfs::File>> | ||
|  |   openFileForRead(const Twine &Path) { | ||
|  |     if (!isInsideWhitelistedDir(Path)) | ||
|  |       return llvm::errc::no_such_file_or_directory; | ||
|  |     return InnerFS->openFileForRead(Path); | ||
|  |   } | ||
|  | 
 | ||
|  |   llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> | ||
|  |   getBufferForFile(const Twine &Name, int64_t FileSize = -1, | ||
|  |                    bool RequiresNullTerminator = true, | ||
|  |                    bool IsVolatile = false) { | ||
|  |     if (!isInsideWhitelistedDir(Name)) | ||
|  |       return llvm::errc::no_such_file_or_directory; | ||
|  |     return InnerFS->getBufferForFile(Name, FileSize, RequiresNullTerminator, | ||
|  |                                      IsVolatile); | ||
|  |   } | ||
|  | 
 | ||
|  |   virtual vfs::directory_iterator dir_begin(const Twine &Dir, | ||
|  |                                             std::error_code &EC) { | ||
|  |     if (!isInsideWhitelistedDir(Dir)) { | ||
|  |       EC = llvm::errc::no_such_file_or_directory; | ||
|  |       return vfs::directory_iterator(); | ||
|  |     } | ||
|  |     return InnerFS->dir_begin(Dir, EC); | ||
|  |   } | ||
|  | 
 | ||
|  |   virtual std::error_code setCurrentWorkingDirectory(const Twine &Path) { | ||
|  |     return InnerFS->setCurrentWorkingDirectory(Path); | ||
|  |   } | ||
|  | 
 | ||
|  |   virtual llvm::ErrorOr<std::string> getCurrentWorkingDirectory() const { | ||
|  |     return InnerFS->getCurrentWorkingDirectory(); | ||
|  |   } | ||
|  | 
 | ||
|  |   bool exists(const Twine &Path) { | ||
|  |     if (!isInsideWhitelistedDir(Path)) | ||
|  |       return false; | ||
|  |     return InnerFS->exists(Path); | ||
|  |   } | ||
|  | 
 | ||
|  |   std::error_code makeAbsolute(SmallVectorImpl<char> &Path) const { | ||
|  |     return InnerFS->makeAbsolute(Path); | ||
|  |   } | ||
|  | 
 | ||
|  | private: | ||
|  |   bool isInsideWhitelistedDir(const Twine &InputPath) const { | ||
|  |     SmallString<128> Path; | ||
|  |     InputPath.toVector(Path); | ||
|  | 
 | ||
|  |     if (makeAbsolute(Path)) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     for (const auto &Dir : WhitelistedDirs) { | ||
|  |       if (Path.startswith(Dir)) | ||
|  |         return true; | ||
|  |     } | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   std::vector<std::string> WhitelistedDirs; | ||
|  |   IntrusiveRefCntPtr<vfs::FileSystem> InnerFS; | ||
|  | }; | ||
|  | 
 | ||
|  | /// Create a vfs::FileSystem that has access only to temporary directories
 | ||
|  | /// (obtained by calling system_temp_directory).
 | ||
|  | IntrusiveRefCntPtr<vfs::FileSystem> getTempOnlyFS() { | ||
|  |   llvm::SmallString<128> TmpDir1; | ||
|  |   llvm::sys::path::system_temp_directory(/*erasedOnReboot=*/false, TmpDir1); | ||
|  |   llvm::SmallString<128> TmpDir2; | ||
|  |   llvm::sys::path::system_temp_directory(/*erasedOnReboot=*/true, TmpDir2); | ||
|  | 
 | ||
|  |   std::vector<std::string> TmpDirs; | ||
|  |   TmpDirs.push_back(TmpDir1.str()); | ||
|  |   if (TmpDir1 != TmpDir2) | ||
|  |     TmpDirs.push_back(TmpDir2.str()); | ||
|  |   return new FilteredFileSystem(std::move(TmpDirs), vfs::getRealFileSystem()); | ||
|  | } | ||
|  | 
 | ||
|  | } // namespace
 | ||
|  | 
 | ||
|  | IntrusiveRefCntPtr<vfs::FileSystem> | ||
|  | buildTestFS(llvm::StringMap<std::string> const &Files) { | ||
|  |   IntrusiveRefCntPtr<vfs::InMemoryFileSystem> MemFS( | ||
|  |       new vfs::InMemoryFileSystem); | ||
|  |   for (auto &FileAndContents : Files) | ||
|  |     MemFS->addFile(FileAndContents.first(), time_t(), | ||
|  |                    llvm::MemoryBuffer::getMemBuffer(FileAndContents.second, | ||
|  |                                                     FileAndContents.first())); | ||
|  | 
 | ||
|  |   auto OverlayFS = IntrusiveRefCntPtr<vfs::OverlayFileSystem>( | ||
|  |       new vfs::OverlayFileSystem(getTempOnlyFS())); | ||
|  |   OverlayFS->pushOverlay(std::move(MemFS)); | ||
|  |   return OverlayFS; | ||
|  | } | ||
|  | 
 | ||
|  | Tagged<IntrusiveRefCntPtr<vfs::FileSystem>> | ||
|  | MockFSProvider::getTaggedFileSystem(PathRef File) { | ||
|  |   if (ExpectedFile) { | ||
|  |     EXPECT_EQ(*ExpectedFile, File); | ||
|  |   } | ||
|  | 
 | ||
|  |   auto FS = buildTestFS(Files); | ||
|  |   return make_tagged(FS, Tag); | ||
|  | } | ||
|  | 
 | ||
|  | MockCompilationDatabase::MockCompilationDatabase() | ||
|  |     : ExtraClangFlags({"-ffreestanding"}) {} // Avoid implicit stdc-predef.h.
 | ||
|  | 
 | ||
|  | llvm::Optional<tooling::CompileCommand> | ||
|  | MockCompilationDatabase::getCompileCommand(PathRef File) const { | ||
|  |   if (ExtraClangFlags.empty()) | ||
|  |     return llvm::None; | ||
|  | 
 | ||
|  |   auto CommandLine = ExtraClangFlags; | ||
|  |   CommandLine.insert(CommandLine.begin(), "clang"); | ||
|  |   CommandLine.insert(CommandLine.end(), File.str()); | ||
|  |   return {tooling::CompileCommand(llvm::sys::path::parent_path(File), | ||
|  |                                   llvm::sys::path::filename(File), | ||
|  |                                   std::move(CommandLine), "")}; | ||
|  | } | ||
|  | 
 | ||
|  | static const char *getVirtualTestRoot() { | ||
|  | #ifdef LLVM_ON_WIN32
 | ||
|  |   return "C:\\clangd-test"; | ||
|  | #else
 | ||
|  |   return "/clangd-test"; | ||
|  | #endif
 | ||
|  | } | ||
|  | 
 | ||
|  | llvm::SmallString<32> getVirtualTestFilePath(PathRef File) { | ||
|  |   assert(llvm::sys::path::is_relative(File) && "FileName should be relative"); | ||
|  | 
 | ||
|  |   llvm::SmallString<32> Path; | ||
|  |   llvm::sys::path::append(Path, getVirtualTestRoot(), File); | ||
|  |   return Path; | ||
|  | } | ||
|  | 
 | ||
|  | } // namespace clangd
 | ||
|  | } // namespace clang
 |