mirror of
https://github.com/SpartanJ/eepp.git
synced 2026-05-28 17:16:29 +03:00
396 lines
12 KiB
C++
396 lines
12 KiB
C++
#include "lspclientservermanager.hpp"
|
|
#include "../../projectsearch.hpp"
|
|
#include "lspclientplugin.hpp"
|
|
#include <algorithm>
|
|
#include <eepp/system/filesystem.hpp>
|
|
#include <eepp/system/luapattern.hpp>
|
|
|
|
namespace ecode {
|
|
|
|
LSPClientServerManager::LSPClientServerManager() {}
|
|
|
|
void LSPClientServerManager::load( LSPClientPlugin* plugin, PluginManager* pluginManager,
|
|
std::vector<LSPDefinition>&& lsps ) {
|
|
mPlugin = plugin;
|
|
mPluginManager = pluginManager;
|
|
mThreadPool = pluginManager->getThreadPool();
|
|
mLSPs = lsps;
|
|
}
|
|
|
|
std::vector<LSPDefinition>
|
|
LSPClientServerManager::supportsLSP( const std::shared_ptr<TextDocument>& doc ) {
|
|
if ( !doc->hasFilepath() && doc->getLoadingFilePath().empty() )
|
|
return {};
|
|
std::string fileName( FileSystem::fileNameFromPath(
|
|
doc->getFilePath().empty() ? doc->getLoadingFilePath() : doc->getFilePath() ) );
|
|
const auto& def = doc->getSyntaxDefinition();
|
|
std::vector<LSPDefinition> lsps;
|
|
|
|
for ( auto& lsp : mLSPs ) {
|
|
for ( auto& ext : lsp.filePatterns ) {
|
|
if ( LuaPattern::find( fileName, ext ).isValid() ) {
|
|
lsps.push_back( lsp );
|
|
break;
|
|
}
|
|
auto& files = def.getFiles();
|
|
if ( std::find( files.begin(), files.end(), ext ) != files.end() ) {
|
|
lsps.push_back( lsp );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return lsps;
|
|
}
|
|
|
|
std::unique_ptr<LSPClientServer>
|
|
LSPClientServerManager::runLSPServer( const String::HashType& id, const LSPDefinition& lsp,
|
|
const std::string& rootPath ) {
|
|
auto server = std::make_unique<LSPClientServer>( this, id, lsp, rootPath );
|
|
server->start();
|
|
return server;
|
|
}
|
|
|
|
std::string LSPClientServerManager::findRootPath( const LSPDefinition& lsp,
|
|
const std::shared_ptr<TextDocument>& doc ) {
|
|
if ( lsp.rootIndicationFileNames.empty() || !doc->hasFilepath() )
|
|
return "";
|
|
std::string rootPath( doc->getFileInfo().getDirectoryPath() );
|
|
std::string lRootPath;
|
|
FileSystem::dirAddSlashAtEnd( rootPath );
|
|
|
|
while ( rootPath != lRootPath ) {
|
|
for ( const auto& fileName : lsp.rootIndicationFileNames ) {
|
|
if ( FileSystem::fileExists( rootPath + fileName ) )
|
|
return rootPath;
|
|
|
|
if ( fileName.find_first_of( "$%" ) != std::string::npos &&
|
|
!LuaPattern::matchesAny( FileSystem::filesGetInPath( rootPath ), fileName )
|
|
.empty() )
|
|
return rootPath;
|
|
}
|
|
lRootPath = rootPath;
|
|
rootPath = FileSystem::removeLastFolderFromPath( rootPath );
|
|
}
|
|
|
|
return "";
|
|
}
|
|
|
|
void LSPClientServerManager::tryRunServer( const std::shared_ptr<TextDocument>& doc ) {
|
|
auto lsps = supportsLSP( doc );
|
|
if ( lsps.empty() )
|
|
return;
|
|
|
|
for ( const auto& lsp : lsps ) {
|
|
auto rootPath = findRootPath( lsp, doc );
|
|
auto lspName = lsp.name.empty() ? lsp.command : lsp.name;
|
|
String::HashType id = String::hash( lspName + "|" + lsp.language + "|" + rootPath );
|
|
Lock l( mClientsMutex );
|
|
auto clientIt = mClients.find( id );
|
|
LSPClientServer* server = nullptr;
|
|
if ( clientIt == mClients.end() ) {
|
|
std::unique_ptr<LSPClientServer> serverUP = runLSPServer( id, lsp, rootPath );
|
|
if ( ( server = serverUP.get() ) ) {
|
|
mClients[id] = std::move( serverUP );
|
|
if ( !mLSPWorkspaceFolder.uri.empty() )
|
|
server->didChangeWorkspaceFolders( { mLSPWorkspaceFolder }, {} );
|
|
}
|
|
} else {
|
|
server = clientIt->second.get();
|
|
}
|
|
if ( server ) {
|
|
server->registerDoc( doc );
|
|
}
|
|
}
|
|
}
|
|
|
|
void LSPClientServerManager::closeLSPServer( const String::HashType& id ) {
|
|
if ( mErasingClients.find( id ) != mErasingClients.end() )
|
|
return;
|
|
mErasingClients.insert( id );
|
|
mThreadPool->run( [this, id]() {
|
|
Lock l( mClientsMutex );
|
|
auto it = mClients.find( id );
|
|
if ( it != mClients.end() ) {
|
|
const auto& def = it->second->getDefinition();
|
|
Log::debug( "Closing LSP server: %s for language %s", def.name.c_str(),
|
|
def.language.c_str() );
|
|
mClients.erase( it );
|
|
mLSPsToClose.erase( id );
|
|
mErasingClients.erase( id );
|
|
}
|
|
} );
|
|
}
|
|
|
|
void LSPClientServerManager::goToLocation( const LSPLocation& loc ) {
|
|
UICodeEditorSplitter* splitter = mPlugin->getManager()->getSplitter();
|
|
if ( nullptr == splitter )
|
|
return;
|
|
splitter->getUISceneNode()->runOnMainThread( [this, splitter, loc]() {
|
|
UITab* tab = splitter->isDocumentOpen( loc.uri.toString() );
|
|
if ( !tab ) {
|
|
std::string path( loc.uri.getPath() );
|
|
FileInfo fileInfo( path );
|
|
if ( fileInfo.exists() && fileInfo.isRegularFile() ) {
|
|
splitter->loadAsyncFileFromPathInNewTab(
|
|
path, mThreadPool, [loc]( UICodeEditor* editor, auto ) {
|
|
if ( loc.range.isValid() )
|
|
editor->goToLine( loc.range.start() );
|
|
editor->setFocus();
|
|
} );
|
|
}
|
|
} else {
|
|
tab->getTabWidget()->setTabSelected( tab );
|
|
if ( loc.range.isValid() )
|
|
splitter->editorFromTab( tab )->goToLine( loc.range.start() );
|
|
splitter->editorFromTab( tab )->setFocus();
|
|
}
|
|
} );
|
|
}
|
|
|
|
void LSPClientServerManager::run( const std::shared_ptr<TextDocument>& doc ) {
|
|
mThreadPool->run( [&, doc]() { tryRunServer( doc ); }, []() {} );
|
|
}
|
|
|
|
size_t LSPClientServerManager::clientCount() const {
|
|
return mClients.size();
|
|
}
|
|
|
|
size_t LSPClientServerManager::lspCount() const {
|
|
return mLSPs.size();
|
|
}
|
|
|
|
const std::shared_ptr<ThreadPool>& LSPClientServerManager::getThreadPool() const {
|
|
return mThreadPool;
|
|
}
|
|
|
|
void LSPClientServerManager::updateDirty() {
|
|
// Run the check only once per second
|
|
if ( mUpdateClock.getElapsedTime() < Seconds( 1 ) )
|
|
return;
|
|
|
|
mUpdateClock.restart();
|
|
|
|
{
|
|
Lock l( mClientsMutex );
|
|
for ( auto& server : mClients ) {
|
|
if ( !server.second->hasDocuments() &&
|
|
mLSPsToClose.find( server.first ) == mLSPsToClose.end() )
|
|
mLSPsToClose.insert( { server.first, std::make_unique<Clock>() } );
|
|
}
|
|
}
|
|
|
|
if ( !mLSPsToClose.empty() ) {
|
|
std::vector<String::HashType> removed;
|
|
std::vector<String::HashType> invalidatedClose;
|
|
|
|
for ( const auto& server : mLSPsToClose ) {
|
|
// Kill server only after N seconds of inactivity
|
|
if ( server.second->getElapsedTime() > mLSPDecayTime ) {
|
|
// If a document was opened while waiting, remove the server from the queue
|
|
auto clientServer = mClients.find( server.first );
|
|
if ( clientServer != mClients.end() && clientServer->second->hasDocuments() ) {
|
|
invalidatedClose.push_back( server.first );
|
|
} else {
|
|
removed.push_back( server.first );
|
|
}
|
|
}
|
|
}
|
|
if ( !invalidatedClose.empty() ) {
|
|
for ( auto invalided : invalidatedClose )
|
|
mLSPsToClose.erase( invalided );
|
|
}
|
|
if ( !removed.empty() ) {
|
|
for ( auto& remove : removed )
|
|
closeLSPServer( remove );
|
|
}
|
|
}
|
|
}
|
|
|
|
void LSPClientServerManager::getAndGoToLocation( const std::shared_ptr<TextDocument>& doc,
|
|
const std::string& search ) {
|
|
auto* server = getOneLSPClientServer( doc );
|
|
if ( server )
|
|
server->getAndGoToLocation( doc->getURI(), doc->getSelection().start(), search );
|
|
}
|
|
|
|
PluginManager* LSPClientServerManager::getPluginManager() const {
|
|
return mPluginManager;
|
|
}
|
|
|
|
LSPClientPlugin* LSPClientServerManager::getPlugin() const {
|
|
return mPlugin;
|
|
}
|
|
|
|
void LSPClientServerManager::findAndOpenClosestURI( const std::vector<URI>& uris ) {
|
|
json data;
|
|
data["uri"] = json::array();
|
|
for ( const auto& uri : uris )
|
|
data["uri"].push_back( uri.toString() );
|
|
mPluginManager->sendRequest( mPlugin, PluginMessageType::FindAndOpenClosestURI,
|
|
PluginMessageFormat::JSON, &data );
|
|
}
|
|
|
|
const Time& LSPClientServerManager::getLSPDecayTime() const {
|
|
return mLSPDecayTime;
|
|
}
|
|
|
|
void LSPClientServerManager::setLSPDecayTime( const Time& lSPDecayTime ) {
|
|
mLSPDecayTime = lSPDecayTime;
|
|
}
|
|
|
|
void LSPClientServerManager::sendSymbolReferenceBroadcast( const std::vector<LSPLocation>& resp ) {
|
|
if ( resp.empty() )
|
|
return;
|
|
|
|
std::map<std::string, ProjectSearch::ResultData> res;
|
|
|
|
for ( auto& r : resp ) {
|
|
auto& rd = res[r.uri.getPath()];
|
|
if ( rd.file.empty() )
|
|
rd.file = r.uri.getPath();
|
|
auto curDoc = mPluginManager->getSplitter()->findDocFromPath( r.uri.getPath() );
|
|
if ( !curDoc )
|
|
continue;
|
|
|
|
ProjectSearch::ResultData::Result rs( curDoc->line( r.range.start().line() ).getText(),
|
|
r.range, -1, -1 );
|
|
|
|
rd.results.emplace_back( std::move( rs ) );
|
|
}
|
|
|
|
ProjectSearch::Result result;
|
|
for ( auto& r : res ) {
|
|
if ( !r.second.results.empty() )
|
|
result.emplace_back( std::move( r.second ) );
|
|
}
|
|
|
|
mPluginManager->sendBroadcast( PluginMessageType::SymbolReference,
|
|
PluginMessageFormat::ProjectSearchResult, &result );
|
|
}
|
|
|
|
void LSPClientServerManager::getSymbolReferences( std::shared_ptr<TextDocument> doc ) {
|
|
auto* server = getOneLSPClientServer( doc );
|
|
if ( !server )
|
|
return;
|
|
server->documentReferences(
|
|
doc->getURI(), doc->getSelection().start(), true,
|
|
[this]( const PluginIDType&, const std::vector<LSPLocation>& resp ) {
|
|
sendSymbolReferenceBroadcast( resp );
|
|
} );
|
|
}
|
|
|
|
void LSPClientServerManager::memoryUsage( std::shared_ptr<TextDocument> doc ) {
|
|
auto* server = getOneLSPClientServer( doc );
|
|
if ( !server )
|
|
return;
|
|
server->memoryUsage();
|
|
}
|
|
|
|
const std::vector<LSPDefinition>& LSPClientServerManager::getLSPs() const {
|
|
return mLSPs;
|
|
}
|
|
|
|
LSPDefinition
|
|
LSPClientServerManager::getLSPForLang( const std::string& lang,
|
|
const std::vector<std::string>& extensions ) const {
|
|
for ( const auto& lsp : mLSPs ) {
|
|
if ( lsp.language == lang ) {
|
|
return lsp;
|
|
}
|
|
|
|
if ( !lsp.filePatterns.empty() ) {
|
|
for ( const auto& file : lsp.filePatterns ) {
|
|
for ( const auto& ext : extensions ) {
|
|
if ( ext == file ) {
|
|
return lsp;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return {};
|
|
}
|
|
|
|
void LSPClientServerManager::didChangeWorkspaceFolders( const std::string& folder ) {
|
|
mLSPWorkspaceFolder = { "file://" + folder, FileSystem::fileNameFromPath( folder ) };
|
|
Lock l( mClientsMutex );
|
|
for ( auto& server : mClients )
|
|
server.second->didChangeWorkspaceFolders( { mLSPWorkspaceFolder }, {} );
|
|
}
|
|
|
|
const LSPWorkspaceFolder& LSPClientServerManager::getLSPWorkspaceFolder() const {
|
|
return mLSPWorkspaceFolder;
|
|
}
|
|
|
|
std::vector<LSPClientServer*> LSPClientServerManager::getLSPClientServers( UICodeEditor* editor ) {
|
|
return getLSPClientServers( editor->getDocumentRef() );
|
|
}
|
|
|
|
std::vector<LSPClientServer*>
|
|
LSPClientServerManager::getLSPClientServers( const std::shared_ptr<TextDocument>& doc ) {
|
|
std::vector<LSPClientServer*> servers;
|
|
Lock l( mClientsMutex );
|
|
for ( auto& server : mClients ) {
|
|
if ( server.second->hasDocument( doc.get() ) )
|
|
servers.push_back( server.second.get() );
|
|
}
|
|
return servers;
|
|
}
|
|
|
|
std::vector<LSPClientServer*> LSPClientServerManager::getLSPClientServers( const URI& uri ) {
|
|
std::vector<LSPClientServer*> servers;
|
|
Lock l( mClientsMutex );
|
|
for ( auto& server : mClients ) {
|
|
if ( server.second->hasDocument( uri ) )
|
|
servers.push_back( server.second.get() );
|
|
}
|
|
return servers;
|
|
}
|
|
|
|
LSPClientServer* LSPClientServerManager::getOneLSPClientServer( UICodeEditor* editor ) {
|
|
return getOneLSPClientServer( editor->getDocumentRef() );
|
|
}
|
|
|
|
LSPClientServer*
|
|
LSPClientServerManager::getOneLSPClientServer( const std::shared_ptr<TextDocument>& doc ) {
|
|
Lock l( mClientsMutex );
|
|
for ( auto& server : mClients ) {
|
|
if ( server.second->hasDocument( doc.get() ) )
|
|
return server.second.get();
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
LSPClientServer* LSPClientServerManager::getOneLSPClientServer( const URI& uri ) {
|
|
Lock l( mClientsMutex );
|
|
for ( auto& server : mClients ) {
|
|
if ( server.second->hasDocument( uri ) )
|
|
return server.second.get();
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
LSPClientServer* LSPClientServerManager::getOneLSPClientServer( const std::string& language ) {
|
|
std::vector<LSPClientServer*> servers;
|
|
Lock l( mClientsMutex );
|
|
for ( auto& server : mClients ) {
|
|
if ( server.second->getDefinition().language == language )
|
|
return server.second.get();
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
std::vector<LSPClientServer*>
|
|
LSPClientServerManager::getLSPClientServers( const std::string& language ) {
|
|
std::vector<LSPClientServer*> servers;
|
|
Lock l( mClientsMutex );
|
|
for ( auto& server : mClients ) {
|
|
if ( server.second->getDefinition().language == language )
|
|
servers.push_back( server.second.get() );
|
|
}
|
|
return servers;
|
|
}
|
|
|
|
} // namespace ecode
|