mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	Merge pull request #12994 from ethereum/lsp-include-path
LSP to resolve custom include paths (e.g. Hardhat)
This commit is contained in:
		
						commit
						f686294374
					
				| @ -30,7 +30,8 @@ Compiler Features: | |||||||
|  * LSP: Add rudimentary support for semantic highlighting. |  * LSP: Add rudimentary support for semantic highlighting. | ||||||
|  * Type Checker: Warn about assignments involving multiple pushes to storage ``bytes`` that may invalidate references. |  * Type Checker: Warn about assignments involving multiple pushes to storage ``bytes`` that may invalidate references. | ||||||
|  * Yul Optimizer: Improve inlining heuristics for via IR code generation and pure Yul compilation. |  * Yul Optimizer: Improve inlining heuristics for via IR code generation and pure Yul compilation. | ||||||
| 
 |  * Language Server: Always add ``{project_root}/node_modules`` to include search paths. | ||||||
|  |  * Language Server: Adds support for configuring ``include-paths`` JSON settings object that can be passed during LSP configuration stage. | ||||||
| 
 | 
 | ||||||
| Bugfixes: | Bugfixes: | ||||||
|  * ABI Encoder: When encoding an empty string coming from storage do not add a superfluous empty slot for data. |  * ABI Encoder: When encoding an empty string coming from storage do not add a superfluous empty slot for data. | ||||||
|  | |||||||
| @ -22,11 +22,14 @@ | |||||||
| #include <libsolutil/StringUtils.h> | #include <libsolutil/StringUtils.h> | ||||||
| #include <libsolutil/CommonIO.h> | #include <libsolutil/CommonIO.h> | ||||||
| 
 | 
 | ||||||
|  | #include <range/v3/algorithm/none_of.hpp> | ||||||
| #include <range/v3/range/conversion.hpp> | #include <range/v3/range/conversion.hpp> | ||||||
| #include <range/v3/view/transform.hpp> | #include <range/v3/view/transform.hpp> | ||||||
| 
 | 
 | ||||||
| #include <regex> | #include <regex> | ||||||
| 
 | 
 | ||||||
|  | #include <boost/algorithm/string/predicate.hpp> | ||||||
|  | 
 | ||||||
| using namespace std; | using namespace std; | ||||||
| using namespace solidity; | using namespace solidity; | ||||||
| using namespace solidity::lsp; | using namespace solidity::lsp; | ||||||
| @ -34,25 +37,49 @@ using namespace solidity::frontend; | |||||||
| 
 | 
 | ||||||
| using solidity::util::readFileAsString; | using solidity::util::readFileAsString; | ||||||
| using solidity::util::joinHumanReadable; | using solidity::util::joinHumanReadable; | ||||||
|  | using solidity::util::Result; | ||||||
| 
 | 
 | ||||||
| FileRepository::FileRepository(boost::filesystem::path _basePath): m_basePath(std::move(_basePath)) | FileRepository::FileRepository(boost::filesystem::path _basePath, std::vector<boost::filesystem::path> _includePaths): | ||||||
|  | 	m_basePath(std::move(_basePath)), | ||||||
|  | 	m_includePaths(std::move(_includePaths)) | ||||||
| { | { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void FileRepository::setIncludePaths(std::vector<boost::filesystem::path> _paths) | ||||||
|  | { | ||||||
|  | 	m_includePaths = std::move(_paths); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| string FileRepository::sourceUnitNameToUri(string const& _sourceUnitName) const | string FileRepository::sourceUnitNameToUri(string const& _sourceUnitName) const | ||||||
| { | { | ||||||
| 	regex const windowsDriveLetterPath("^[a-zA-Z]:/"); | 	regex const windowsDriveLetterPath("^[a-zA-Z]:/"); | ||||||
| 
 | 
 | ||||||
|  | 	auto const ensurePathIsUnixLike = [&](string inputPath) -> string { | ||||||
|  | 		if (!regex_search(inputPath, windowsDriveLetterPath)) | ||||||
|  | 			return inputPath; | ||||||
|  | 		else | ||||||
|  | 			return "/" + move(inputPath); | ||||||
|  | 	}; | ||||||
|  | 
 | ||||||
| 	if (m_sourceUnitNamesToUri.count(_sourceUnitName)) | 	if (m_sourceUnitNamesToUri.count(_sourceUnitName)) | ||||||
|  | 	{ | ||||||
|  | 		solAssert(boost::starts_with(m_sourceUnitNamesToUri.at(_sourceUnitName), "file://"), ""); | ||||||
| 		return m_sourceUnitNamesToUri.at(_sourceUnitName); | 		return m_sourceUnitNamesToUri.at(_sourceUnitName); | ||||||
|  | 	} | ||||||
| 	else if (_sourceUnitName.find("file://") == 0) | 	else if (_sourceUnitName.find("file://") == 0) | ||||||
| 		return _sourceUnitName; | 		return _sourceUnitName; | ||||||
| 	else if (regex_search(_sourceUnitName, windowsDriveLetterPath)) | 	else if (regex_search(_sourceUnitName, windowsDriveLetterPath)) | ||||||
| 		return "file:///" + _sourceUnitName; | 		return "file:///" + _sourceUnitName; | ||||||
| 	else if (_sourceUnitName.find("/") == 0) | 	else if ( | ||||||
| 		return "file://" + _sourceUnitName; | 		auto const resolvedPath = tryResolvePath(_sourceUnitName); | ||||||
| 	else | 		resolvedPath.message().empty() | ||||||
|  | 	) | ||||||
|  | 		return "file://" + ensurePathIsUnixLike(resolvedPath.get().generic_string()); | ||||||
|  | 	else if (m_basePath.generic_string() != "/") | ||||||
| 		return "file://" + m_basePath.generic_string() + "/" + _sourceUnitName; | 		return "file://" + m_basePath.generic_string() + "/" + _sourceUnitName; | ||||||
|  | 	else | ||||||
|  | 		// Avoid double-/ in case base-path itself is simply a UNIX root filesystem root.
 | ||||||
|  | 		return "file:///" + _sourceUnitName; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| string FileRepository::uriToSourceUnitName(string const& _path) const | string FileRepository::uriToSourceUnitName(string const& _path) const | ||||||
| @ -69,6 +96,52 @@ void FileRepository::setSourceByUri(string const& _uri, string _source) | |||||||
| 	m_sourceCodes[sourceUnitName] = std::move(_source); | 	m_sourceCodes[sourceUnitName] = std::move(_source); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | Result<boost::filesystem::path> FileRepository::tryResolvePath(std::string const& _strippedSourceUnitName) const | ||||||
|  | { | ||||||
|  | 	if ( | ||||||
|  | 		boost::filesystem::path(_strippedSourceUnitName).has_root_path() && | ||||||
|  | 		boost::filesystem::exists(_strippedSourceUnitName) | ||||||
|  | 	) | ||||||
|  | 		return boost::filesystem::path(_strippedSourceUnitName); | ||||||
|  | 
 | ||||||
|  | 	vector<boost::filesystem::path> candidates; | ||||||
|  | 	vector<reference_wrapper<boost::filesystem::path const>> prefixes = {m_basePath}; | ||||||
|  | 	prefixes += (m_includePaths | ranges::to<vector<reference_wrapper<boost::filesystem::path const>>>); | ||||||
|  | 	auto const defaultInclude = m_basePath / "node_modules"; | ||||||
|  | 	if (m_includePaths.empty()) | ||||||
|  | 		prefixes.emplace_back(defaultInclude); | ||||||
|  | 
 | ||||||
|  | 	auto const pathToQuotedString = [](boost::filesystem::path const& _path) { return "\"" + _path.string() + "\""; }; | ||||||
|  | 
 | ||||||
|  | 	for (auto const& prefix: prefixes) | ||||||
|  | 	{ | ||||||
|  | 		boost::filesystem::path canonicalPath = boost::filesystem::path(prefix) / boost::filesystem::path(_strippedSourceUnitName); | ||||||
|  | 
 | ||||||
|  | 		if (boost::filesystem::exists(canonicalPath)) | ||||||
|  | 			candidates.push_back(move(canonicalPath)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (candidates.empty()) | ||||||
|  | 		return Result<boost::filesystem::path>::err( | ||||||
|  | 			"File not found. Searched the following locations: " + | ||||||
|  | 			joinHumanReadable(prefixes | ranges::views::transform(pathToQuotedString), ", ") + | ||||||
|  | 			"." | ||||||
|  | 		); | ||||||
|  | 
 | ||||||
|  | 	if (candidates.size() >= 2) | ||||||
|  | 		return Result<boost::filesystem::path>::err( | ||||||
|  | 			"Ambiguous import. " | ||||||
|  | 			"Multiple matching files found inside base path and/or include paths: " + | ||||||
|  | 			joinHumanReadable(candidates | ranges::views::transform(pathToQuotedString), ", ") + | ||||||
|  | 			"." | ||||||
|  | 		); | ||||||
|  | 
 | ||||||
|  | 	if (!boost::filesystem::is_regular_file(candidates[0])) | ||||||
|  | 		return Result<boost::filesystem::path>::err("Not a valid file."); | ||||||
|  | 
 | ||||||
|  | 	return candidates[0]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| frontend::ReadCallback::Result FileRepository::readFile(string const& _kind, string const& _sourceUnitName) | frontend::ReadCallback::Result FileRepository::readFile(string const& _kind, string const& _sourceUnitName) | ||||||
| { | { | ||||||
| 	solAssert( | 	solAssert( | ||||||
| @ -83,53 +156,11 @@ frontend::ReadCallback::Result FileRepository::readFile(string const& _kind, str | |||||||
| 			return ReadCallback::Result{true, m_sourceCodes.at(_sourceUnitName)}; | 			return ReadCallback::Result{true, m_sourceCodes.at(_sourceUnitName)}; | ||||||
| 
 | 
 | ||||||
| 		string const strippedSourceUnitName = stripFileUriSchemePrefix(_sourceUnitName); | 		string const strippedSourceUnitName = stripFileUriSchemePrefix(_sourceUnitName); | ||||||
|  | 		Result<boost::filesystem::path> const resolvedPath = tryResolvePath(strippedSourceUnitName); | ||||||
|  | 		if (!resolvedPath.message().empty()) | ||||||
|  | 			return ReadCallback::Result{false, resolvedPath.message()}; | ||||||
| 
 | 
 | ||||||
| 		if ( | 		auto contents = readFileAsString(resolvedPath.get()); | ||||||
| 			boost::filesystem::path(strippedSourceUnitName).has_root_path() && |  | ||||||
| 			boost::filesystem::exists(strippedSourceUnitName) |  | ||||||
| 		) |  | ||||||
| 		{ |  | ||||||
| 			auto contents = readFileAsString(strippedSourceUnitName); |  | ||||||
| 			solAssert(m_sourceCodes.count(_sourceUnitName) == 0, ""); |  | ||||||
| 			m_sourceCodes[_sourceUnitName] = contents; |  | ||||||
| 			return ReadCallback::Result{true, move(contents)}; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		vector<boost::filesystem::path> candidates; |  | ||||||
| 		vector<reference_wrapper<boost::filesystem::path>> prefixes = {m_basePath}; |  | ||||||
| 		prefixes += (m_includePaths | ranges::to<vector<reference_wrapper<boost::filesystem::path>>>); |  | ||||||
| 
 |  | ||||||
| 		auto const pathToQuotedString = [](boost::filesystem::path const& _path) { return "\"" + _path.string() + "\""; }; |  | ||||||
| 
 |  | ||||||
| 		for (auto const& prefix: prefixes) |  | ||||||
| 		{ |  | ||||||
| 			boost::filesystem::path canonicalPath = boost::filesystem::path(prefix) / boost::filesystem::path(strippedSourceUnitName); |  | ||||||
| 
 |  | ||||||
| 			if (boost::filesystem::exists(canonicalPath)) |  | ||||||
| 				candidates.push_back(move(canonicalPath)); |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if (candidates.empty()) |  | ||||||
| 			return ReadCallback::Result{ |  | ||||||
| 				false, |  | ||||||
| 				"File not found. Searched the following locations: " + |  | ||||||
| 				joinHumanReadable(prefixes | ranges::views::transform(pathToQuotedString), ", ") + |  | ||||||
| 				"." |  | ||||||
| 			}; |  | ||||||
| 
 |  | ||||||
| 		if (candidates.size() >= 2) |  | ||||||
| 			return ReadCallback::Result{ |  | ||||||
| 				false, |  | ||||||
| 				"Ambiguous import. " |  | ||||||
| 				"Multiple matching files found inside base path and/or include paths: " + |  | ||||||
| 				joinHumanReadable(candidates | ranges::views::transform(pathToQuotedString), ", ") + |  | ||||||
| 				"." |  | ||||||
| 			}; |  | ||||||
| 
 |  | ||||||
| 		if (!boost::filesystem::is_regular_file(candidates[0])) |  | ||||||
| 			return ReadCallback::Result{false, "Not a valid file."}; |  | ||||||
| 
 |  | ||||||
| 		auto contents = readFileAsString(candidates[0]); |  | ||||||
| 		solAssert(m_sourceCodes.count(_sourceUnitName) == 0, ""); | 		solAssert(m_sourceCodes.count(_sourceUnitName) == 0, ""); | ||||||
| 		m_sourceCodes[_sourceUnitName] = contents; | 		m_sourceCodes[_sourceUnitName] = contents; | ||||||
| 		return ReadCallback::Result{true, move(contents)}; | 		return ReadCallback::Result{true, move(contents)}; | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ | |||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
| #include <libsolidity/interface/FileReader.h> | #include <libsolidity/interface/FileReader.h> | ||||||
|  | #include <libsolutil/Result.h> | ||||||
| 
 | 
 | ||||||
| #include <string> | #include <string> | ||||||
| #include <map> | #include <map> | ||||||
| @ -28,7 +29,10 @@ namespace solidity::lsp | |||||||
| class FileRepository | class FileRepository | ||||||
| { | { | ||||||
| public: | public: | ||||||
| 	explicit FileRepository(boost::filesystem::path _basePath); | 	FileRepository(boost::filesystem::path _basePath, std::vector<boost::filesystem::path> _includePaths); | ||||||
|  | 
 | ||||||
|  | 	std::vector<boost::filesystem::path> const& includePaths() const noexcept { return m_includePaths; } | ||||||
|  | 	void setIncludePaths(std::vector<boost::filesystem::path> _paths); | ||||||
| 
 | 
 | ||||||
| 	boost::filesystem::path const& basePath() const { return m_basePath; } | 	boost::filesystem::path const& basePath() const { return m_basePath; } | ||||||
| 
 | 
 | ||||||
| @ -51,6 +55,8 @@ public: | |||||||
| 		return [this](std::string const& _kind, std::string const& _path) { return readFile(_kind, _path); }; | 		return [this](std::string const& _kind, std::string const& _path) { return readFile(_kind, _path); }; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	util::Result<boost::filesystem::path> tryResolvePath(std::string const& _sourceUnitName) const; | ||||||
|  | 
 | ||||||
| private: | private: | ||||||
| 	/// Base path without URI scheme.
 | 	/// Base path without URI scheme.
 | ||||||
| 	boost::filesystem::path m_basePath; | 	boost::filesystem::path m_basePath; | ||||||
|  | |||||||
| @ -130,7 +130,7 @@ LanguageServer::LanguageServer(Transport& _transport): | |||||||
| 		{"textDocument/semanticTokens/full", bind(&LanguageServer::semanticTokensFull, this, _1, _2)}, | 		{"textDocument/semanticTokens/full", bind(&LanguageServer::semanticTokensFull, this, _1, _2)}, | ||||||
| 		{"workspace/didChangeConfiguration", bind(&LanguageServer::handleWorkspaceDidChangeConfiguration, this, _2)}, | 		{"workspace/didChangeConfiguration", bind(&LanguageServer::handleWorkspaceDidChangeConfiguration, this, _2)}, | ||||||
| 	}, | 	}, | ||||||
| 	m_fileRepository("/" /* basePath */), | 	m_fileRepository("/" /* basePath */, {} /* no search paths */), | ||||||
| 	m_compilerStack{m_fileRepository.reader()} | 	m_compilerStack{m_fileRepository.reader()} | ||||||
| { | { | ||||||
| } | } | ||||||
| @ -148,6 +148,26 @@ Json::Value LanguageServer::toJson(SourceLocation const& _location) | |||||||
| void LanguageServer::changeConfiguration(Json::Value const& _settings) | void LanguageServer::changeConfiguration(Json::Value const& _settings) | ||||||
| { | { | ||||||
| 	m_settingsObject = _settings; | 	m_settingsObject = _settings; | ||||||
|  | 	Json::Value jsonIncludePaths = _settings["include-paths"]; | ||||||
|  | 	int typeFailureCount = 0; | ||||||
|  | 
 | ||||||
|  | 	if (jsonIncludePaths && jsonIncludePaths.isArray()) | ||||||
|  | 	{ | ||||||
|  | 		vector<boost::filesystem::path> includePaths; | ||||||
|  | 		for (Json::Value const& jsonPath: jsonIncludePaths) | ||||||
|  | 		{ | ||||||
|  | 			if (jsonPath.isString()) | ||||||
|  | 				includePaths.emplace_back(boost::filesystem::path(jsonPath.asString())); | ||||||
|  | 			else | ||||||
|  | 				typeFailureCount++; | ||||||
|  | 		} | ||||||
|  | 		m_fileRepository.setIncludePaths(move(includePaths)); | ||||||
|  | 	} | ||||||
|  | 	else | ||||||
|  | 		++typeFailureCount; | ||||||
|  | 
 | ||||||
|  | 	if (typeFailureCount) | ||||||
|  | 		m_client.trace("Invalid JSON configuration passed. \"include-paths\" must be an array of strings."); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void LanguageServer::compile() | void LanguageServer::compile() | ||||||
| @ -155,7 +175,7 @@ void LanguageServer::compile() | |||||||
| 	// For files that are not open, we have to take changes on disk into account,
 | 	// For files that are not open, we have to take changes on disk into account,
 | ||||||
| 	// so we just remove all non-open files.
 | 	// so we just remove all non-open files.
 | ||||||
| 
 | 
 | ||||||
| 	FileRepository oldRepository(m_fileRepository.basePath()); | 	FileRepository oldRepository(m_fileRepository.basePath(), m_fileRepository.includePaths()); | ||||||
| 	swap(oldRepository, m_fileRepository); | 	swap(oldRepository, m_fileRepository); | ||||||
| 
 | 
 | ||||||
| 	for (string const& fileName: m_openFiles) | 	for (string const& fileName: m_openFiles) | ||||||
| @ -302,7 +322,7 @@ void LanguageServer::handleInitialize(MessageID _id, Json::Value const& _args) | |||||||
| 	else if (Json::Value rootPath = _args["rootPath"]) | 	else if (Json::Value rootPath = _args["rootPath"]) | ||||||
| 		rootPath = rootPath.asString(); | 		rootPath = rootPath.asString(); | ||||||
| 
 | 
 | ||||||
| 	m_fileRepository = FileRepository(rootPath); | 	m_fileRepository = FileRepository(rootPath, {}); | ||||||
| 	if (_args["initializationOptions"].isObject()) | 	if (_args["initializationOptions"].isObject()) | ||||||
| 		changeConfiguration(_args["initializationOptions"]); | 		changeConfiguration(_args["initializationOptions"]); | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										14
									
								
								test/libsolidity/lsp/include-paths/default_include.sol
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								test/libsolidity/lsp/include-paths/default_include.sol
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,14 @@ | |||||||
|  | // SPDX-License-Identifier: UNLICENSED | ||||||
|  | pragma solidity >=0.8.0; | ||||||
|  | 
 | ||||||
|  | import "my-module/test.sol"; | ||||||
|  | 
 | ||||||
|  | contract MyContract | ||||||
|  | { | ||||||
|  |     function f(uint a, uint b) public pure returns (uint) | ||||||
|  |     { | ||||||
|  |         return MyModule.add(a, b); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | // ---- | ||||||
|  | // test: | ||||||
							
								
								
									
										10
									
								
								test/libsolidity/lsp/include-paths/file_at_include_path.sol
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								test/libsolidity/lsp/include-paths/file_at_include_path.sol
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | // SPDX-License-Identifier: UNLICENSED | ||||||
|  | pragma solidity >=0.8.0; | ||||||
|  | 
 | ||||||
|  | import "rootlib.sol"; | ||||||
|  | 
 | ||||||
|  | contract MyContract | ||||||
|  | { | ||||||
|  | } | ||||||
|  | // ---- | ||||||
|  | // rootlib: | ||||||
| @ -0,0 +1,11 @@ | |||||||
|  | // SPDX-License-Identifier: UNLICENSED | ||||||
|  | pragma solidity >=0.8.0; | ||||||
|  | 
 | ||||||
|  |    import "test.sol"; | ||||||
|  | // ^^^^^^^^^^^^^^^^^^ @IncludeLocation | ||||||
|  | 
 | ||||||
|  | contract SomeContract | ||||||
|  | { | ||||||
|  | } | ||||||
|  | // ---- | ||||||
|  | // file_not_found_in_searchpath: @IncludeLocation 6275 | ||||||
							
								
								
									
										11
									
								
								test/libsolidity/lsp/include-paths/using-custom-includes.sol
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								test/libsolidity/lsp/include-paths/using-custom-includes.sol
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | // SPDX-License-Identifier: UNLICENSED | ||||||
|  | pragma solidity >=0.8.0; | ||||||
|  | 
 | ||||||
|  |    import "otherlib/otherlib.sol"; | ||||||
|  | // ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ @NotFound | ||||||
|  | 
 | ||||||
|  | contract MyContract | ||||||
|  | { | ||||||
|  | } | ||||||
|  | // ---- | ||||||
|  | // using-custom-includes: @NotFound 6275 | ||||||
							
								
								
									
										10
									
								
								test/libsolidity/lsp/node_modules/my-module/test.sol
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								test/libsolidity/lsp/node_modules/my-module/test.sol
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | // SPDX-License-Identifier: UNLICENSED | ||||||
|  | pragma solidity >=0.8.0; | ||||||
|  | 
 | ||||||
|  | library MyModule | ||||||
|  | { | ||||||
|  |     function add(uint a, uint b) public pure returns (uint) | ||||||
|  |     { | ||||||
|  |         return a + b; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										6
									
								
								test/libsolidity/lsp/node_modules/rootlib.sol
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								test/libsolidity/lsp/node_modules/rootlib.sol
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,6 @@ | |||||||
|  | // SPDX-License-Identifier: UNLICENSED | ||||||
|  | pragma solidity >=0.8.0; | ||||||
|  | 
 | ||||||
|  | library MyRootLib | ||||||
|  | { | ||||||
|  | } | ||||||
| @ -0,0 +1,7 @@ | |||||||
|  | // SPDX-License-Identifier: UNLICENSED | ||||||
|  | pragma solidity >=0.8.0; | ||||||
|  | 
 | ||||||
|  | library OtherLib | ||||||
|  | { | ||||||
|  |     function f(uint n) public returns (uint) { return n + 1; } | ||||||
|  | } | ||||||
							
								
								
									
										30
									
								
								test/lsp.py
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								test/lsp.py
									
									
									
									
									
								
							| @ -1381,6 +1381,34 @@ class SolidityLSPTestSuite: # {{{ | |||||||
|         return markers |         return markers | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |     def test_custom_includes(self, solc: JsonRpcProcess) -> None: | ||||||
|  |         self.setup_lsp(solc, expose_project_root=False) | ||||||
|  |         solc.send_notification( | ||||||
|  |             'workspace/didChangeConfiguration', { | ||||||
|  |                 'settings': { | ||||||
|  |                     'include-paths': [ | ||||||
|  |                         f"{self.project_root_dir}/other-include-dir" | ||||||
|  |                     ] | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         ) | ||||||
|  |         published_diagnostics = self.open_file_and_wait_for_diagnostics(solc, 'include-paths/using-custom-includes') | ||||||
|  | 
 | ||||||
|  |         self.expect_equal(len(published_diagnostics), 2, "Diagnostic reports for 2 files") | ||||||
|  | 
 | ||||||
|  |         # test file | ||||||
|  |         report = published_diagnostics[0] | ||||||
|  |         self.expect_equal(report['uri'], self.get_test_file_uri('using-custom-includes', 'include-paths')) | ||||||
|  |         diagnostics = report['diagnostics'] | ||||||
|  |         self.expect_equal(len(diagnostics), 0, "no diagnostics") | ||||||
|  | 
 | ||||||
|  |         # imported file | ||||||
|  |         report = published_diagnostics[1] | ||||||
|  |         self.expect_equal(report['uri'], f"{self.project_root_uri}/other-include-dir/otherlib/otherlib.sol") | ||||||
|  |         diagnostics = report['diagnostics'] | ||||||
|  |         self.expect_equal(len(diagnostics), 1, "no diagnostics") | ||||||
|  |         self.expect_diagnostic(diagnostics[0], code=2018, lineNo=5, startEndColumns=(4, 62)) | ||||||
|  | 
 | ||||||
|     def test_didChange_in_A_causing_error_in_B(self, solc: JsonRpcProcess) -> None: |     def test_didChange_in_A_causing_error_in_B(self, solc: JsonRpcProcess) -> None: | ||||||
|         # Reusing another test but now change some file that generates an error in the other. |         # Reusing another test but now change some file that generates an error in the other. | ||||||
|         self.test_textDocument_didOpen_with_relative_import(solc) |         self.test_textDocument_didOpen_with_relative_import(solc) | ||||||
| @ -1450,7 +1478,7 @@ class SolidityLSPTestSuite: # {{{ | |||||||
|         self.setup_lsp(solc) |         self.setup_lsp(solc) | ||||||
| 
 | 
 | ||||||
|         sub_dirs = filter( |         sub_dirs = filter( | ||||||
|             lambda filepath: filepath.is_dir(), |             lambda filepath: filepath.is_dir() and filepath.name != 'node_modules', | ||||||
|             os.scandir(self.project_root_dir) |             os.scandir(self.project_root_dir) | ||||||
|         ) |         ) | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user