123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165 |
- /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
- file Copyright.txt or https://cmake.org/licensing for details. */
- #include "cmServerConnection.h"
- #include "cmConfigure.h"
- #include "cmServer.h"
- #include "cmServerDictionary.h"
- #include "cm_uv.h"
- #include <algorithm>
- #ifdef _WIN32
- #include "io.h"
- #else
- #include <unistd.h>
- #endif
- #include <cassert>
- cmStdIoConnection::cmStdIoConnection(
- cmConnectionBufferStrategy* bufferStrategy)
- : cmEventBasedConnection(bufferStrategy)
- {
- }
- cm::uv_stream_ptr cmStdIoConnection::SetupStream(int file_id)
- {
- switch (uv_guess_handle(file_id)) {
- case UV_TTY: {
- cm::uv_tty_ptr tty;
- tty.init(*this->Server->GetLoop(), file_id, file_id == 0,
- static_cast<cmEventBasedConnection*>(this));
- uv_tty_set_mode(tty, UV_TTY_MODE_NORMAL);
- return std::move(tty);
- }
- case UV_FILE:
- if (file_id == 0) {
- return nullptr;
- }
- // Intentional fallthrough; stdin can _not_ be treated as a named
- // pipe, however stdout can be.
- CM_FALLTHROUGH;
- case UV_NAMED_PIPE: {
- cm::uv_pipe_ptr pipe;
- pipe.init(*this->Server->GetLoop(), 0,
- static_cast<cmEventBasedConnection*>(this));
- uv_pipe_open(pipe, file_id);
- return std::move(pipe);
- }
- default:
- assert(false && "Unable to determine stream type");
- return nullptr;
- }
- }
- void cmStdIoConnection::SetServer(cmServerBase* s)
- {
- cmConnection::SetServer(s);
- if (!s) {
- return;
- }
- this->ReadStream = SetupStream(0);
- this->WriteStream = SetupStream(1);
- }
- void shutdown_connection(uv_prepare_t* prepare)
- {
- cmStdIoConnection* connection =
- static_cast<cmStdIoConnection*>(prepare->data);
- if (!uv_is_closing(reinterpret_cast<uv_handle_t*>(prepare))) {
- uv_close(reinterpret_cast<uv_handle_t*>(prepare),
- &cmEventBasedConnection::on_close_delete<uv_prepare_t>);
- }
- connection->OnDisconnect(0);
- }
- bool cmStdIoConnection::OnServeStart(std::string* pString)
- {
- Server->OnConnected(this);
- if (this->ReadStream.get()) {
- uv_read_start(this->ReadStream, on_alloc_buffer, on_read);
- } else if (uv_guess_handle(0) == UV_FILE) {
- char buffer[1024];
- while (auto len = read(0, buffer, sizeof(buffer))) {
- ReadData(std::string(buffer, buffer + len));
- }
- // We can't start the disconnect from here, add a prepare hook to do that
- // for us
- auto prepare = new uv_prepare_t();
- prepare->data = this;
- uv_prepare_init(Server->GetLoop(), prepare);
- uv_prepare_start(prepare, shutdown_connection);
- }
- return cmConnection::OnServeStart(pString);
- }
- bool cmStdIoConnection::OnConnectionShuttingDown()
- {
- if (ReadStream.get()) {
- uv_read_stop(ReadStream);
- ReadStream->data = nullptr;
- }
- this->ReadStream.reset();
- cmEventBasedConnection::OnConnectionShuttingDown();
- return true;
- }
- cmServerPipeConnection::cmServerPipeConnection(const std::string& name)
- : cmPipeConnection(name, new cmServerBufferStrategy)
- {
- }
- cmServerStdIoConnection::cmServerStdIoConnection()
- : cmStdIoConnection(new cmServerBufferStrategy)
- {
- }
- cmConnectionBufferStrategy::~cmConnectionBufferStrategy()
- {
- }
- void cmConnectionBufferStrategy::clear()
- {
- }
- std::string cmServerBufferStrategy::BufferOutMessage(
- const std::string& rawBuffer) const
- {
- return std::string("\n") + kSTART_MAGIC + std::string("\n") + rawBuffer +
- kEND_MAGIC + std::string("\n");
- }
- std::string cmServerBufferStrategy::BufferMessage(std::string& RawReadBuffer)
- {
- for (;;) {
- auto needle = RawReadBuffer.find('\n');
- if (needle == std::string::npos) {
- return "";
- }
- std::string line = RawReadBuffer.substr(0, needle);
- const auto ls = line.size();
- if (ls > 1 && line.at(ls - 1) == '\r') {
- line.erase(ls - 1, 1);
- }
- RawReadBuffer.erase(RawReadBuffer.begin(),
- RawReadBuffer.begin() + static_cast<long>(needle) + 1);
- if (line == kSTART_MAGIC) {
- RequestBuffer.clear();
- continue;
- }
- if (line == kEND_MAGIC) {
- std::string rtn;
- rtn.swap(this->RequestBuffer);
- return rtn;
- }
- this->RequestBuffer += line;
- this->RequestBuffer += "\n";
- }
- }
|