2017-06-21 17:29:13 +08:00
|
|
|
#include <gflags/gflags.h>
|
|
|
|
#include <glog/logging.h>
|
2017-03-06 20:37:51 +08:00
|
|
|
|
2017-06-07 21:23:08 +08:00
|
|
|
#include "bolt_common.hpp"
|
2017-03-06 20:37:51 +08:00
|
|
|
#include "communication/bolt/v1/session.hpp"
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
|
|
|
|
using communication::bolt::ClientError;
|
|
|
|
using communication::bolt::Session;
|
|
|
|
using communication::bolt::SessionException;
|
|
|
|
using communication::bolt::State;
|
2018-07-24 21:11:18 +08:00
|
|
|
using communication::bolt::Value;
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
|
|
|
|
static const char *kInvalidQuery = "invalid query";
|
|
|
|
static const char *kQueryReturn42 = "RETURN 42";
|
|
|
|
static const char *kQueryEmpty = "no results";
|
|
|
|
|
|
|
|
class TestSessionData {};
|
|
|
|
|
|
|
|
class TestSession : public Session<TestInputStream, TestOutputStream> {
|
|
|
|
public:
|
2018-07-18 16:40:06 +08:00
|
|
|
using Session<TestInputStream, TestOutputStream>::TEncoder;
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
|
|
|
|
TestSession(TestSessionData &data, TestInputStream &input_stream,
|
|
|
|
TestOutputStream &output_stream)
|
|
|
|
: Session<TestInputStream, TestOutputStream>(input_stream,
|
|
|
|
output_stream) {}
|
|
|
|
|
2018-07-18 16:40:06 +08:00
|
|
|
std::vector<std::string> Interpret(
|
|
|
|
const std::string &query,
|
2018-07-24 21:11:18 +08:00
|
|
|
const std::map<std::string, Value> ¶ms) override {
|
2018-07-18 16:40:06 +08:00
|
|
|
if (query == kQueryReturn42 || query == kQueryEmpty) {
|
|
|
|
query_ = query;
|
|
|
|
return {"result_name"};
|
|
|
|
} else {
|
|
|
|
query_ = "";
|
|
|
|
throw ClientError("client sent invalid query");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-24 21:11:18 +08:00
|
|
|
std::map<std::string, Value> PullAll(TEncoder *encoder) override {
|
2018-07-18 16:40:06 +08:00
|
|
|
if (query_ == kQueryReturn42) {
|
2018-07-24 21:11:18 +08:00
|
|
|
encoder->MessageRecord(std::vector<Value>{42});
|
2018-07-18 16:40:06 +08:00
|
|
|
return {};
|
|
|
|
} else if (query_ == kQueryEmpty) {
|
|
|
|
return {};
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
} else {
|
|
|
|
throw ClientError("client sent invalid query");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Abort() override {}
|
|
|
|
|
2018-07-18 16:40:06 +08:00
|
|
|
private:
|
|
|
|
std::string query_;
|
|
|
|
};
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-08-03 21:53:41 +08:00
|
|
|
// TODO: This could be done in fixture.
|
2017-04-15 21:14:12 +08:00
|
|
|
// Shortcuts for writing variable initializations in tests
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
#define INIT_VARS \
|
|
|
|
TestInputStream input_stream; \
|
|
|
|
TestOutputStream output_stream; \
|
|
|
|
TestSessionData session_data; \
|
|
|
|
TestSession session(session_data, input_stream, output_stream); \
|
2018-03-23 23:32:17 +08:00
|
|
|
std::vector<uint8_t> &output = output_stream.output;
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
// Sample testdata that has correct inputs and outputs.
|
2017-06-07 21:23:08 +08:00
|
|
|
const uint8_t handshake_req[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x00, 0x00,
|
|
|
|
0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
2017-04-15 21:14:12 +08:00
|
|
|
const uint8_t handshake_resp[] = {0x00, 0x00, 0x00, 0x01};
|
|
|
|
const uint8_t init_req[] = {
|
2017-06-07 21:23:08 +08:00
|
|
|
0xb2, 0x01, 0xd0, 0x15, 0x6c, 0x69, 0x62, 0x6e, 0x65, 0x6f, 0x34,
|
|
|
|
0x6a, 0x2d, 0x63, 0x6c, 0x69, 0x65, 0x6e, 0x74, 0x2f, 0x31, 0x2e,
|
|
|
|
0x32, 0x2e, 0x31, 0xa3, 0x86, 0x73, 0x63, 0x68, 0x65, 0x6d, 0x65,
|
|
|
|
0x85, 0x62, 0x61, 0x73, 0x69, 0x63, 0x89, 0x70, 0x72, 0x69, 0x6e,
|
|
|
|
0x63, 0x69, 0x70, 0x61, 0x6c, 0x80, 0x8b, 0x63, 0x72, 0x65, 0x64,
|
|
|
|
0x65, 0x6e, 0x74, 0x69, 0x61, 0x6c, 0x73, 0x80};
|
2017-04-15 21:14:12 +08:00
|
|
|
const uint8_t init_resp[] = {0x00, 0x03, 0xb1, 0x70, 0xa0, 0x00, 0x00};
|
|
|
|
const uint8_t run_req_header[] = {0xb2, 0x10, 0xd1};
|
|
|
|
const uint8_t pullall_req[] = {0xb0, 0x3f};
|
|
|
|
const uint8_t discardall_req[] = {0xb0, 0x2f};
|
|
|
|
const uint8_t reset_req[] = {0xb0, 0x0f};
|
|
|
|
const uint8_t ackfailure_req[] = {0xb0, 0x0e};
|
|
|
|
const uint8_t success_resp[] = {0x00, 0x03, 0xb1, 0x70, 0xa0, 0x00, 0x00};
|
|
|
|
const uint8_t ignored_resp[] = {0x00, 0x02, 0xb0, 0x7e, 0x00, 0x00};
|
|
|
|
|
|
|
|
// Write bolt chunk header (length)
|
2018-03-23 23:32:17 +08:00
|
|
|
void WriteChunkHeader(TestInputStream &input_stream, uint16_t len) {
|
2018-04-22 14:31:09 +08:00
|
|
|
len = utils::Bswap(len);
|
2018-03-23 23:32:17 +08:00
|
|
|
input_stream.Write(reinterpret_cast<uint8_t *>(&len), sizeof(len));
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write bolt chunk tail (two zeros)
|
2018-03-23 23:32:17 +08:00
|
|
|
void WriteChunkTail(TestInputStream &input_stream) {
|
|
|
|
WriteChunkHeader(input_stream, 0);
|
|
|
|
}
|
2017-03-22 23:36:48 +08:00
|
|
|
|
2017-08-03 21:53:41 +08:00
|
|
|
// Check that the server responded with a failure message.
|
2017-04-15 21:14:12 +08:00
|
|
|
void CheckFailureMessage(std::vector<uint8_t> &output) {
|
|
|
|
ASSERT_GE(output.size(), 6);
|
|
|
|
// skip the first two bytes because they are the chunk header
|
2017-06-07 21:23:08 +08:00
|
|
|
ASSERT_EQ(output[2], 0xB1); // tiny struct 1
|
|
|
|
ASSERT_EQ(output[3], 0x7F); // signature failure
|
2017-08-03 21:53:41 +08:00
|
|
|
output.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that the server responded with a success message.
|
|
|
|
void CheckSuccessMessage(std::vector<uint8_t> &output, bool clear = true) {
|
|
|
|
ASSERT_GE(output.size(), 6);
|
|
|
|
// skip the first two bytes because they are the chunk header
|
|
|
|
ASSERT_EQ(output[2], 0xB1); // tiny struct 1
|
|
|
|
ASSERT_EQ(output[3], 0x70); // signature success
|
|
|
|
if (clear) {
|
|
|
|
output.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that the server responded with a ignore message.
|
|
|
|
void CheckIgnoreMessage(std::vector<uint8_t> &output) {
|
|
|
|
ASSERT_GE(output.size(), 6);
|
|
|
|
// skip the first two bytes because they are the chunk header
|
|
|
|
ASSERT_EQ(output[2], 0xB0);
|
|
|
|
ASSERT_EQ(output[3], 0x7E); // signature ignore
|
|
|
|
output.clear();
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Execute and check a correct handshake
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
void ExecuteHandshake(TestInputStream &input_stream, TestSession &session,
|
2018-03-23 23:32:17 +08:00
|
|
|
std::vector<uint8_t> &output) {
|
|
|
|
input_stream.Write(handshake_req, 20);
|
2017-04-15 21:14:12 +08:00
|
|
|
session.Execute();
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Init);
|
2017-03-28 18:42:04 +08:00
|
|
|
PrintOutput(output);
|
|
|
|
CheckOutput(output, handshake_resp, 4);
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write bolt chunk and execute command
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
void ExecuteCommand(TestInputStream &input_stream, TestSession &session,
|
2018-03-23 23:32:17 +08:00
|
|
|
const uint8_t *data, size_t len, bool chunk = true) {
|
|
|
|
if (chunk) WriteChunkHeader(input_stream, len);
|
|
|
|
input_stream.Write(data, len);
|
|
|
|
if (chunk) WriteChunkTail(input_stream);
|
2017-04-15 21:14:12 +08:00
|
|
|
session.Execute();
|
|
|
|
}
|
2017-03-06 20:37:51 +08:00
|
|
|
|
2017-04-15 21:14:12 +08:00
|
|
|
// Execute and check a correct init
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
void ExecuteInit(TestInputStream &input_stream, TestSession &session,
|
2018-03-23 23:32:17 +08:00
|
|
|
std::vector<uint8_t> &output) {
|
|
|
|
ExecuteCommand(input_stream, session, init_req, sizeof(init_req));
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Idle);
|
2017-03-28 18:42:04 +08:00
|
|
|
PrintOutput(output);
|
|
|
|
CheckOutput(output, init_resp, 7);
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write bolt encoded run request
|
2018-03-23 23:32:17 +08:00
|
|
|
void WriteRunRequest(TestInputStream &input_stream, const char *str) {
|
2017-04-15 21:14:12 +08:00
|
|
|
// write chunk header
|
|
|
|
auto len = strlen(str);
|
2018-03-23 23:32:17 +08:00
|
|
|
WriteChunkHeader(input_stream, 3 + 2 + len + 1);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
// write string header
|
2018-03-23 23:32:17 +08:00
|
|
|
input_stream.Write(run_req_header, 3);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
// write string length
|
2018-03-23 23:32:17 +08:00
|
|
|
WriteChunkHeader(input_stream, len);
|
2017-03-06 20:37:51 +08:00
|
|
|
|
2017-04-15 21:14:12 +08:00
|
|
|
// write string
|
2018-03-23 23:32:17 +08:00
|
|
|
input_stream.Write(str, len);
|
2017-03-28 18:42:04 +08:00
|
|
|
|
2017-04-15 21:14:12 +08:00
|
|
|
// write empty map for parameters
|
2018-03-23 23:32:17 +08:00
|
|
|
input_stream.Write("\xA0", 1); // TinyMap0
|
2017-03-06 20:37:51 +08:00
|
|
|
|
2017-04-15 21:14:12 +08:00
|
|
|
// write chunk tail
|
2018-03-23 23:32:17 +08:00
|
|
|
WriteChunkTail(input_stream);
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, HandshakeWrongPreamble) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
// write 0x00000001 five times
|
|
|
|
for (int i = 0; i < 5; ++i) input_stream.Write(handshake_req + 4, 4);
|
2018-02-22 23:17:45 +08:00
|
|
|
ASSERT_THROW(session.Execute(), SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
PrintOutput(output);
|
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, HandshakeInTwoPackets) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
input_stream.Write(handshake_req, 10);
|
2017-04-15 21:14:12 +08:00
|
|
|
session.Execute();
|
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Handshake);
|
2017-03-06 20:37:51 +08:00
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
input_stream.Write(handshake_req + 10, 10);
|
2017-04-15 21:14:12 +08:00
|
|
|
session.Execute();
|
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Init);
|
2017-04-15 21:14:12 +08:00
|
|
|
PrintOutput(output);
|
|
|
|
CheckOutput(output, handshake_resp, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, HandshakeWriteFail) {
|
|
|
|
INIT_VARS;
|
2018-03-23 23:32:17 +08:00
|
|
|
output_stream.SetWriteSuccess(false);
|
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, handshake_req,
|
|
|
|
sizeof(handshake_req), false),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
ASSERT_EQ(output.size(), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, HandshakeOK) {
|
|
|
|
INIT_VARS;
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, InitWrongSignature) {
|
|
|
|
INIT_VARS;
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, run_req_header,
|
|
|
|
sizeof(run_req_header)),
|
2018-02-22 23:17:45 +08:00
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, InitWrongMarker) {
|
|
|
|
INIT_VARS;
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
// wrong marker, good signature
|
|
|
|
uint8_t data[2] = {0x00, init_req[1]};
|
2018-03-23 23:32:17 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, data, 2),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, InitMissingData) {
|
|
|
|
// test lengths, they test the following situations:
|
|
|
|
// missing header data, missing client name, missing metadata
|
|
|
|
int len[] = {1, 2, 25};
|
|
|
|
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
INIT_VARS;
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, init_req, len[i]),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, InitWriteFail) {
|
|
|
|
INIT_VARS;
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
output_stream.SetWriteSuccess(false);
|
|
|
|
ASSERT_THROW(
|
|
|
|
ExecuteCommand(input_stream, session, init_req, sizeof(init_req)),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
ASSERT_EQ(output.size(), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, InitOK) {
|
|
|
|
INIT_VARS;
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ExecuteRunWrongMarker) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
// wrong marker, good signature
|
|
|
|
uint8_t data[2] = {0x00, run_req_header[1]};
|
2018-03-23 23:32:17 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, data, sizeof(data)),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ExecuteRunMissingData) {
|
|
|
|
// test lengths, they test the following situations:
|
|
|
|
// missing header data, missing query data, missing parameters
|
|
|
|
int len[] = {1, 2, 37};
|
|
|
|
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
INIT_VARS;
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, run_req_header, len[i]),
|
2018-02-22 23:17:45 +08:00
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ExecuteRunBasicException) {
|
|
|
|
// first test with socket write success, then with socket write fail
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
output_stream.SetWriteSuccess(i == 0);
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kInvalidQuery);
|
2018-02-22 23:17:45 +08:00
|
|
|
if (i == 0) {
|
|
|
|
session.Execute();
|
|
|
|
} else {
|
|
|
|
ASSERT_THROW(session.Execute(), SessionException);
|
|
|
|
}
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
if (i == 0) {
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Error);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
} else {
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
ASSERT_EQ(output.size(), 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-18 23:14:43 +08:00
|
|
|
TEST(BoltSession, ExecuteRunWithoutPullAll) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-07-18 23:14:43 +08:00
|
|
|
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kQueryReturn42);
|
2017-07-18 23:14:43 +08:00
|
|
|
session.Execute();
|
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Result);
|
2017-07-18 23:14:43 +08:00
|
|
|
}
|
|
|
|
|
2017-04-15 21:14:12 +08:00
|
|
|
TEST(BoltSession, ExecutePullAllDiscardAllResetWrongMarker) {
|
|
|
|
// This test first tests PULL_ALL then DISCARD_ALL and then RESET
|
|
|
|
// It tests for missing data in the message header
|
|
|
|
const uint8_t *dataset[3] = {pullall_req, discardall_req, reset_req};
|
|
|
|
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
// wrong marker, good signature
|
|
|
|
uint8_t data[2] = {0x00, dataset[i][1]};
|
2018-03-23 23:32:17 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, data, sizeof(data)),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ExecutePullAllBufferEmpty) {
|
|
|
|
// first test with socket write success, then with socket write fail
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
output_stream.SetWriteSuccess(i == 0);
|
|
|
|
ASSERT_THROW(
|
|
|
|
ExecuteCommand(input_stream, session, pullall_req, sizeof(pullall_req)),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
if (i == 0) {
|
|
|
|
CheckFailureMessage(output);
|
|
|
|
} else {
|
|
|
|
ASSERT_EQ(output.size(), 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ExecutePullAllDiscardAllReset) {
|
|
|
|
// This test first tests PULL_ALL then DISCARD_ALL and then RESET
|
|
|
|
// It tests a good message
|
|
|
|
const uint8_t *dataset[3] = {pullall_req, discardall_req, reset_req};
|
|
|
|
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
// first test with socket write success, then with socket write fail
|
|
|
|
for (int j = 0; j < 2; ++j) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kQueryReturn42);
|
2017-04-15 21:14:12 +08:00
|
|
|
session.Execute();
|
|
|
|
|
|
|
|
if (j == 1) output.clear();
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
output_stream.SetWriteSuccess(j == 0);
|
2018-02-22 23:17:45 +08:00
|
|
|
if (j == 0) {
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteCommand(input_stream, session, dataset[i], 2);
|
2018-02-22 23:17:45 +08:00
|
|
|
} else {
|
2018-03-23 23:32:17 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, dataset[i], 2),
|
|
|
|
SessionException);
|
2018-02-22 23:17:45 +08:00
|
|
|
}
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
if (j == 0) {
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Idle);
|
2017-04-15 21:14:12 +08:00
|
|
|
ASSERT_FALSE(session.encoder_buffer_.HasData());
|
|
|
|
PrintOutput(output);
|
|
|
|
} else {
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
ASSERT_EQ(output.size(), 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ExecuteInvalidMessage) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
|
|
|
ASSERT_THROW(
|
|
|
|
ExecuteCommand(input_stream, session, init_req, sizeof(init_req)),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ErrorIgnoreMessage) {
|
|
|
|
// first test with socket write success, then with socket write fail
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kInvalidQuery);
|
2017-04-15 21:14:12 +08:00
|
|
|
session.Execute();
|
|
|
|
|
|
|
|
output.clear();
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
output_stream.SetWriteSuccess(i == 0);
|
2018-02-22 23:17:45 +08:00
|
|
|
if (i == 0) {
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteCommand(input_stream, session, init_req, sizeof(init_req));
|
2018-02-22 23:17:45 +08:00
|
|
|
} else {
|
2018-03-23 23:32:17 +08:00
|
|
|
ASSERT_THROW(
|
|
|
|
ExecuteCommand(input_stream, session, init_req, sizeof(init_req)),
|
|
|
|
SessionException);
|
2018-02-22 23:17:45 +08:00
|
|
|
}
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
// assert that all data from the init message was cleaned up
|
|
|
|
ASSERT_EQ(session.decoder_buffer_.Size(), 0);
|
|
|
|
|
|
|
|
if (i == 0) {
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Error);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckOutput(output, ignored_resp, sizeof(ignored_resp));
|
|
|
|
} else {
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
ASSERT_EQ(output.size(), 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-21 00:35:25 +08:00
|
|
|
TEST(BoltSession, ErrorRunAfterRun) {
|
|
|
|
// first test with socket write success, then with socket write fail
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-07-21 00:35:25 +08:00
|
|
|
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kQueryReturn42);
|
2017-07-21 00:35:25 +08:00
|
|
|
session.Execute();
|
|
|
|
|
|
|
|
output.clear();
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
output_stream.SetWriteSuccess(true);
|
2017-07-21 00:35:25 +08:00
|
|
|
|
|
|
|
// Session holds results of last run.
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Result);
|
2017-07-21 00:35:25 +08:00
|
|
|
|
|
|
|
// New run request.
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kQueryReturn42);
|
2018-02-22 23:17:45 +08:00
|
|
|
ASSERT_THROW(session.Execute(), SessionException);
|
2017-07-21 00:35:25 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-07-21 00:35:25 +08:00
|
|
|
}
|
|
|
|
|
2017-04-15 21:14:12 +08:00
|
|
|
TEST(BoltSession, ErrorCantCleanup) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kInvalidQuery);
|
2017-04-15 21:14:12 +08:00
|
|
|
session.Execute();
|
|
|
|
|
|
|
|
output.clear();
|
|
|
|
|
|
|
|
// there is data missing in the request, cleanup should fail
|
2018-03-23 23:32:17 +08:00
|
|
|
ASSERT_THROW(
|
|
|
|
ExecuteCommand(input_stream, session, init_req, sizeof(init_req) - 10),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ErrorWrongMarker) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kInvalidQuery);
|
2017-04-15 21:14:12 +08:00
|
|
|
session.Execute();
|
|
|
|
|
|
|
|
output.clear();
|
|
|
|
|
|
|
|
// wrong marker, good signature
|
|
|
|
uint8_t data[2] = {0x00, init_req[1]};
|
2018-03-23 23:32:17 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, data, sizeof(data)),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ErrorOK) {
|
|
|
|
// test ACK_FAILURE and RESET
|
|
|
|
const uint8_t *dataset[] = {ackfailure_req, reset_req};
|
|
|
|
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
2017-06-07 21:23:08 +08:00
|
|
|
// first test with socket write success, then with socket write fail
|
2017-04-15 21:14:12 +08:00
|
|
|
for (int j = 0; j < 2; ++j) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kInvalidQuery);
|
2017-04-15 21:14:12 +08:00
|
|
|
session.Execute();
|
|
|
|
|
|
|
|
output.clear();
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
output_stream.SetWriteSuccess(j == 0);
|
2018-02-22 23:17:45 +08:00
|
|
|
if (j == 0) {
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteCommand(input_stream, session, dataset[i], 2);
|
2018-02-22 23:17:45 +08:00
|
|
|
} else {
|
2018-03-23 23:32:17 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, dataset[i], 2),
|
|
|
|
SessionException);
|
2018-02-22 23:17:45 +08:00
|
|
|
}
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
// assert that all data from the init message was cleaned up
|
|
|
|
ASSERT_EQ(session.decoder_buffer_.Size(), 0);
|
|
|
|
|
|
|
|
if (j == 0) {
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Idle);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckOutput(output, success_resp, sizeof(success_resp));
|
|
|
|
} else {
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
ASSERT_EQ(output.size(), 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ErrorMissingData) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kInvalidQuery);
|
2017-04-15 21:14:12 +08:00
|
|
|
session.Execute();
|
|
|
|
|
|
|
|
output.clear();
|
|
|
|
|
|
|
|
// some marker, missing signature
|
|
|
|
uint8_t data[1] = {0x00};
|
2018-03-23 23:32:17 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, data, sizeof(data)),
|
|
|
|
SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, MultipleChunksInOneExecute) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
Extract communication to static library
Summary:
Session specifics have been move out of the Bolt `executing` state, and
are accessed via pure virtual Session type. Our server is templated on
the session and we are setting the concrete type, so there should be no
virtual call overhead. Abstract Session is used to indicate the
interface, this could have also been templated, but the explicit
interface definition makes it clearer.
Specific session implementation for running Memgraph is now implemented
in memgraph_bolt, which instantiates the concrete session type. This may
not be 100% appropriate place, but Memgraph specific session isn't
needed anywhere else.
Bolt/communication tests now use a dummy session and depend only on
communication, which significantly improves test run times.
All these changes make the communication a library which doesn't depend
on storage nor the database. Only shared connection points, which aren't
part of the base communication library are:
* glue/conversion -- which converts between storage and bolt types, and
* communication/result_stream_faker -- templated, but used in tests and query/repl
Depends on D1453
Reviewers: mferencevic, buda, mtomic, msantl
Reviewed By: mferencevic, mtomic
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1456
2018-07-10 22:18:19 +08:00
|
|
|
WriteRunRequest(input_stream, kQueryReturn42);
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteCommand(input_stream, session, pullall_req, sizeof(pullall_req));
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Idle);
|
2017-04-15 21:14:12 +08:00
|
|
|
PrintOutput(output);
|
|
|
|
|
|
|
|
// Count chunks in output
|
|
|
|
int len, num = 0;
|
2017-06-07 21:23:08 +08:00
|
|
|
while (output.size() > 0) {
|
2017-04-15 21:14:12 +08:00
|
|
|
len = (output[0] << 8) + output[1];
|
|
|
|
output.erase(output.begin(), output.begin() + len + 4);
|
|
|
|
++num;
|
|
|
|
}
|
|
|
|
|
|
|
|
// there should be 3 chunks in the output
|
|
|
|
// the first is a success with the query headers
|
|
|
|
// the second is a record message
|
|
|
|
// and the last is a success message with query run metadata
|
|
|
|
ASSERT_EQ(num, 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, PartialChunk) {
|
|
|
|
INIT_VARS;
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
WriteChunkHeader(input_stream, sizeof(discardall_req));
|
|
|
|
input_stream.Write(discardall_req, sizeof(discardall_req));
|
2017-04-15 21:14:12 +08:00
|
|
|
|
|
|
|
// missing chunk tail
|
|
|
|
session.Execute();
|
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Idle);
|
2017-04-15 21:14:12 +08:00
|
|
|
ASSERT_EQ(output.size(), 0);
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
WriteChunkTail(input_stream);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2018-02-22 23:17:45 +08:00
|
|
|
ASSERT_THROW(session.Execute(), SessionException);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
2017-04-15 21:14:12 +08:00
|
|
|
ASSERT_GT(output.size(), 0);
|
|
|
|
PrintOutput(output);
|
|
|
|
}
|
|
|
|
|
2017-03-28 18:42:04 +08:00
|
|
|
int main(int argc, char **argv) {
|
2017-06-21 17:29:13 +08:00
|
|
|
google::InitGoogleLogging(argv[0]);
|
2017-03-06 20:37:51 +08:00
|
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
|
|
return RUN_ALL_TESTS();
|
|
|
|
}
|