#include "LibLsp/lsp/ProcessIoService.h"
#include <boost/program_options.hpp>
#include "LibLsp/lsp/general/exit.h"
#include "LibLsp/lsp/textDocument/declaration_definition.h"
#include "LibLsp/lsp/textDocument/signature_help.h"
#include "LibLsp/lsp/general/initialize.h"
#include "LibLsp/lsp/ProtocolJsonHandler.h"
#include "LibLsp/lsp/textDocument/typeHierarchy.h"
#include "LibLsp/lsp/AbsolutePath.h"
#include "LibLsp/lsp/textDocument/resolveCompletionItem.h"
#include <network/uri.hpp>
#include "LibLsp/JsonRpc/Endpoint.h"
#include "LibLsp/JsonRpc/stream.h"
#include "LibLsp/JsonRpc/TcpServer.h"
#include "LibLsp/lsp/textDocument/document_symbol.h"
#include "LibLsp/lsp/workspace/execute_command.h"
#include <boost/process.hpp>
#include <boost/filesystem.hpp>
#include <boost/asio.hpp>
#include <iostream>
#include <thread>
using namespace boost::asio::ip;
using namespace std;
class DummyLog :public lsp::Log
{
public:
void log(Level level, std::wstring&& msg)
{
std::wcerr << msg << std::endl;
};
void log(Level level, const std::wstring& msg)
{
std::wcerr << msg << std::endl;
};
void log(Level level, std::string&& msg)
{
std::cerr << msg << std::endl;
};
void log(Level level, const std::string& msg)
{
std::cerr << msg << std::endl;
};
};
struct boost_process_ipstream : lsp::base_istream< boost::process::ipstream >
{
explicit boost_process_ipstream(boost::process::ipstream& _t)
: base_istream<boost::process::ipstream>(_t)
{
}
std::string what() override
{
return {};
}
void clear() override
{
}
};
struct boost_process_opstream : lsp::base_ostream< boost::process::opstream >
{
explicit boost_process_opstream(boost::process::opstream& _t)
: lsp::base_ostream<boost::process::opstream>(_t)
{
}
std::string what() override
{
return {};
}
void clear() override
{
}
};
class Client
{
public:
Client(std::string& execPath,const std::vector<std::string>& args)
:point(protocol_json_handler, endpoint, _log)
{
std::error_code ec;
namespace bp = boost::process;
c = std::make_shared<bp::child>(asio_io.getIOService(), execPath,
bp::args = args,
ec,
bp::std_out > read_from_service,
bp::std_in < write_to_service,
bp::on_exit([&](int exit_code, const std::error_code& ec_in){
}));
if (ec)
{
// fail
_log.log(lsp::Log::Level::SEVERE, "Start server failed.");
}
else {
//success
point.startProcessingMessages(read_from_service_proxy, write_to_service_proxy);
}
}
~Client()
{
point.stop();
std::this_thread::sleep_for(std::chrono::milliseconds (1000));
}
lsp::ProcessIoService asio_io;
std::shared_ptr < lsp::ProtocolJsonHandler > protocol_json_handler = std::make_shared< lsp::ProtocolJsonHandler>();
DummyLog _log;
std::shared_ptr<GenericEndpoint> endpoint = std::make_shared<GenericEndpoint>(_log);
boost::process::opstream write_to_service;
boost::process::ipstream read_from_service;
std::shared_ptr<lsp::ostream> write_to_service_proxy = std::make_shared<boost_process_opstream>(write_to_service);
std::shared_ptr<lsp::istream> read_from_service_proxy = std::make_shared< boost_process_ipstream >(read_from_service);
std::shared_ptr<boost::process::child> c;
RemoteEndPoint point;
};
int main(int argc, char* argv[])
{
using namespace boost::program_options;
options_description desc("Allowed options");
desc.add_options()
("help,h", "produce help message")
("execPath", value<string>(), "LSP server's path");
variables_map vm;
try {
store(parse_command_line(argc, argv, desc), vm);
}
catch (std::exception& e) {
std::cout << "Undefined input.Reason:" << e.what() << std::endl;
return 0;
}
notify(vm);
if (vm.count("help"))
{
cout << desc << endl;
return 1;
}
string execPath;
if (vm.count("execPath"))
{
execPath = vm["execPath"].as<string>();
}
else
{
execPath = "StdIOServerExample";
}
Client client(execPath, {});
{
td_initialize::request req;
auto rsp = client.point.waitResponse(req);
if (rsp)
{
std::cerr << rsp->ToJson() << std::endl;
}
else
{
std::cerr << "get initialze response time out" << std::endl;
}
}
{
td_definition::request req;
auto future_rsp = client.point.send(req);
auto state = future_rsp.wait_for(std::chrono::seconds(4));
if (lsp::future_status::timeout == state)
{
std::cerr << "get textDocument/definition response time out" << std::endl;
return 0;
}
auto rsp = future_rsp.get();
if (rsp.is_error)
{
std::cerr << "get textDocument/definition response error" << std::endl;
}
else
{
std::cerr << rsp.response.ToJson() << std::endl;
}
}
{
td_initialize::request req;
auto rsp = client.point.waitResponse(req);
if (rsp)
{
std::cerr << rsp->ToJson() << std::endl;
}
else
{
std::cerr << "get initialze response time out" << std::endl;
}
}
Notify_Exit::notify notify;
client.point.send(notify);
return 0;
}