2022-02-22 20:33:45 +08:00
|
|
|
// Copyright 2022 Memgraph Ltd.
|
2021-10-26 14:53:56 +08:00
|
|
|
//
|
|
|
|
// Use of this software is governed by the Business Source License
|
|
|
|
// included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
|
|
|
|
// License, and you may not use this file except in compliance with the Business Source License.
|
|
|
|
//
|
|
|
|
// As of the Change Date specified in that file, in accordance with
|
|
|
|
// the Business Source License, use of this software will be governed
|
|
|
|
// by the Apache License, Version 2.0, included in the file
|
|
|
|
// licenses/APL.txt.
|
|
|
|
|
2022-08-26 19:19:27 +08:00
|
|
|
#include <string>
|
|
|
|
|
2017-06-21 17:29:13 +08:00
|
|
|
#include <gflags/gflags.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"
|
2020-10-16 18:49:33 +08:00
|
|
|
#include "communication/exceptions.hpp"
|
2021-01-21 22:47:56 +08:00
|
|
|
#include "utils/logging.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
|
|
|
|
2022-02-22 20:33:45 +08:00
|
|
|
using memgraph::communication::bolt::ClientError;
|
|
|
|
using memgraph::communication::bolt::Session;
|
|
|
|
using memgraph::communication::bolt::SessionException;
|
|
|
|
using memgraph::communication::bolt::State;
|
|
|
|
using memgraph::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";
|
2020-10-16 18:49:33 +08:00
|
|
|
static const char *kQueryReturnMultiple = "UNWIND [1,2,3] as n RETURN n";
|
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 *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
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
TestSession(TestSessionData *data, TestInputStream *input_stream, TestOutputStream *output_stream)
|
|
|
|
: Session<TestInputStream, TestOutputStream>(input_stream, output_stream) {}
|
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
|
|
|
|
2020-10-16 18:49:33 +08:00
|
|
|
std::pair<std::vector<std::string>, std::optional<int>> Interpret(
|
2021-02-18 22:32:43 +08:00
|
|
|
const std::string &query, const std::map<std::string, Value> ¶ms) override {
|
|
|
|
if (query == kQueryReturn42 || query == kQueryEmpty || query == kQueryReturnMultiple) {
|
2018-07-18 16:40:06 +08:00
|
|
|
query_ = query;
|
2020-10-16 18:49:33 +08:00
|
|
|
return {{"result_name"}, {}};
|
2018-07-18 16:40:06 +08:00
|
|
|
} else {
|
|
|
|
query_ = "";
|
|
|
|
throw ClientError("client sent invalid query");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
std::map<std::string, Value> Pull(TEncoder *encoder, std::optional<int> n, std::optional<int> qid) override {
|
2018-07-18 16:40:06 +08:00
|
|
|
if (query_ == kQueryReturn42) {
|
2019-08-28 19:10:27 +08:00
|
|
|
encoder->MessageRecord(std::vector<Value>{Value(42)});
|
2018-07-18 16:40:06 +08:00
|
|
|
return {};
|
|
|
|
} else if (query_ == kQueryEmpty) {
|
|
|
|
return {};
|
2020-10-16 18:49:33 +08:00
|
|
|
} else if (query_ == kQueryReturnMultiple) {
|
|
|
|
static const std::array elements{1, 2, 3};
|
|
|
|
static size_t global_counter = 0;
|
|
|
|
|
|
|
|
int local_counter = 0;
|
2021-02-18 22:32:43 +08:00
|
|
|
for (; global_counter < elements.size() && (!n || local_counter < *n); ++global_counter) {
|
|
|
|
encoder->MessageRecord(std::vector<Value>{Value(elements[global_counter])});
|
2020-10-16 18:49:33 +08:00
|
|
|
++local_counter;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (global_counter == elements.size()) {
|
|
|
|
global_counter = 0;
|
|
|
|
return {std::pair("has_more", false)};
|
|
|
|
}
|
|
|
|
|
|
|
|
return {std::pair("has_more", true)};
|
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");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
std::map<std::string, Value> Discard(std::optional<int>, std::optional<int>) override { return {}; }
|
2020-10-16 18:49:33 +08:00
|
|
|
|
|
|
|
void BeginTransaction() override {}
|
|
|
|
void CommitTransaction() override {}
|
|
|
|
void RollbackTransaction() override {}
|
|
|
|
|
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 Abort() override {}
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
bool Authenticate(const std::string &username, const std::string &password) override { return true; }
|
2018-07-27 16:54:20 +08:00
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
std::optional<std::string> GetServerNameForInit() override { return std::nullopt; }
|
2020-01-16 16:33:31 +08:00
|
|
|
|
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
|
2018-09-03 21:29:06 +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.
|
2022-03-31 19:52:43 +08:00
|
|
|
inline constexpr uint8_t handshake_req[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
|
|
|
inline constexpr uint8_t handshake_resp[] = {0x00, 0x00, 0x00, 0x01};
|
|
|
|
inline constexpr uint8_t init_req[] = {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};
|
|
|
|
inline constexpr uint8_t init_resp[] = {0x00, 0x18, 0xb1, 0x70, 0xa1, 0x8d, 0x63, 0x6f, 0x6e, 0x6e,
|
|
|
|
0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x69, 0x64, 0x86,
|
|
|
|
0x62, 0x6f, 0x6c, 0x74, 0x2d, 0x31, 0x00, 0x00};
|
|
|
|
inline constexpr uint8_t run_req_header[] = {0xb2, 0x10, 0xd1};
|
|
|
|
inline constexpr uint8_t pullall_req[] = {0xb0, 0x3f};
|
|
|
|
inline constexpr uint8_t discardall_req[] = {0xb0, 0x2f};
|
|
|
|
inline constexpr uint8_t reset_req[] = {0xb0, 0x0f};
|
|
|
|
inline constexpr uint8_t ackfailure_req[] = {0xb0, 0x0e};
|
|
|
|
inline constexpr uint8_t success_resp[] = {0x00, 0x03, 0xb1, 0x70, 0xa0, 0x00, 0x00};
|
|
|
|
inline constexpr uint8_t ignored_resp[] = {0x00, 0x02, 0xb0, 0x7e, 0x00, 0x00};
|
2020-10-16 18:49:33 +08:00
|
|
|
|
|
|
|
namespace v4 {
|
2022-03-31 19:52:43 +08:00
|
|
|
inline constexpr uint8_t handshake_req[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
|
|
|
inline constexpr uint8_t handshake_resp[] = {0x00, 0x00, 0x00, 0x04};
|
|
|
|
inline constexpr uint8_t init_req[] = {
|
2021-02-18 22:32:43 +08:00
|
|
|
0xb1, 0x01, 0xa5, 0x8a, 0x75, 0x73, 0x65, 0x72, 0x5f, 0x61, 0x67, 0x65, 0x6e, 0x74, 0xd0, 0x2f, 0x6e, 0x65, 0x6f,
|
|
|
|
0x34, 0x6a, 0x2d, 0x70, 0x79, 0x74, 0x68, 0x6f, 0x6e, 0x2f, 0x34, 0x2e, 0x31, 0x2e, 0x31, 0x20, 0x50, 0x79, 0x74,
|
|
|
|
0x68, 0x6f, 0x6e, 0x2f, 0x33, 0x2e, 0x37, 0x2e, 0x33, 0x2d, 0x66, 0x69, 0x6e, 0x61, 0x6c, 0x2d, 0x30, 0x20, 0x28,
|
|
|
|
0x6c, 0x69, 0x6e, 0x75, 0x78, 0x29, 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, 0x87, 0x72, 0x6f, 0x75, 0x74, 0x69, 0x6e, 0x67, 0xa1, 0x87, 0x61, 0x64, 0x64, 0x72,
|
|
|
|
0x65, 0x73, 0x73, 0x8e, 0x6c, 0x6f, 0x63, 0x61, 0x6c, 0x68, 0x6f, 0x73, 0x74, 0x3a, 0x37, 0x36, 0x38, 0x37};
|
|
|
|
|
2022-03-31 19:52:43 +08:00
|
|
|
inline constexpr uint8_t init_resp[] = {0x00, 0x18, 0xb1, 0x70, 0xa1, 0x8d, 0x63, 0x6f, 0x6e, 0x6e,
|
|
|
|
0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x69, 0x64, 0x86,
|
|
|
|
0x62, 0x6f, 0x6c, 0x74, 0x2d, 0x31, 0x00, 0x00};
|
|
|
|
inline constexpr uint8_t run_req_header[] = {0xb3, 0x10, 0xd1};
|
|
|
|
inline constexpr uint8_t pullall_req[] = {0xb1, 0x3f, 0xa0};
|
|
|
|
inline constexpr uint8_t pull_one_req[] = {0xb1, 0x3f, 0xa1, 0x81, 0x6e, 0x01};
|
|
|
|
inline constexpr uint8_t reset_req[] = {0xb0, 0x0f};
|
|
|
|
inline constexpr uint8_t goodbye[] = {0xb0, 0x02};
|
|
|
|
inline constexpr uint8_t rollback[] = {0xb0, 0x13};
|
2020-10-16 18:49:33 +08:00
|
|
|
} // namespace v4
|
|
|
|
|
|
|
|
namespace v4_1 {
|
2022-03-31 19:52:43 +08:00
|
|
|
inline constexpr uint8_t handshake_req[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x00, 0x01, 0x04, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
|
|
|
inline constexpr uint8_t handshake_resp[] = {0x00, 0x00, 0x01, 0x04};
|
|
|
|
inline constexpr uint8_t noop[] = {0x00, 0x00};
|
2020-10-16 18:49:33 +08:00
|
|
|
} // namespace v4_1
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2021-10-28 19:05:09 +08:00
|
|
|
namespace v4_3 {
|
2022-08-26 19:19:27 +08:00
|
|
|
inline constexpr uint8_t handshake_req[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x00, 0x03, 0x04, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
|
|
|
inline constexpr uint8_t handshake_resp[] = {0x00, 0x00, 0x03, 0x04};
|
|
|
|
inline constexpr uint8_t route[]{0xb3, 0x66, 0xa0, 0x90, 0xc0};
|
2021-10-28 19:05:09 +08:00
|
|
|
} // namespace v4_3
|
|
|
|
|
2017-04-15 21:14:12 +08:00
|
|
|
// Write bolt chunk header (length)
|
2018-03-23 23:32:17 +08:00
|
|
|
void WriteChunkHeader(TestInputStream &input_stream, uint16_t len) {
|
2022-02-22 20:33:45 +08:00
|
|
|
len = memgraph::utils::HostToBigEndian(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)
|
2021-02-18 22:32:43 +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
|
2021-02-18 22:32:43 +08:00
|
|
|
void ExecuteHandshake(TestInputStream &input_stream, TestSession &session, std::vector<uint8_t> &output,
|
|
|
|
const uint8_t *request = handshake_req, const uint8_t *expected_resp = handshake_resp) {
|
2020-10-16 18:49:33 +08:00
|
|
|
input_stream.Write(request, 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);
|
2020-10-16 18:49:33 +08:00
|
|
|
CheckOutput(output, expected_resp, 4);
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write bolt chunk and execute command
|
2021-02-18 22:32:43 +08:00
|
|
|
void ExecuteCommand(TestInputStream &input_stream, TestSession &session, const uint8_t *data, size_t len,
|
|
|
|
bool chunk = true) {
|
2018-03-23 23:32:17 +08:00
|
|
|
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
|
2021-02-18 22:32:43 +08:00
|
|
|
void ExecuteInit(TestInputStream &input_stream, TestSession &session, std::vector<uint8_t> &output,
|
|
|
|
const bool is_v4 = false) {
|
2020-10-16 18:49:33 +08:00
|
|
|
const auto *request = is_v4 ? v4::init_req : init_req;
|
|
|
|
const auto request_size = is_v4 ? sizeof(v4::init_req) : sizeof(init_req);
|
|
|
|
ExecuteCommand(input_stream, session, request, request_size);
|
2017-10-17 20:05:08 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Idle);
|
2017-03-28 18:42:04 +08:00
|
|
|
PrintOutput(output);
|
2020-10-16 18:49:33 +08:00
|
|
|
const auto *response = is_v4 ? v4::init_resp : init_resp;
|
|
|
|
CheckOutput(output, response, 28);
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write bolt encoded run request
|
2021-02-18 22:32:43 +08:00
|
|
|
void WriteRunRequest(TestInputStream &input_stream, const char *str, const bool is_v4 = false) {
|
2017-04-15 21:14:12 +08:00
|
|
|
// write chunk header
|
|
|
|
auto len = strlen(str);
|
2020-10-16 18:49:33 +08:00
|
|
|
WriteChunkHeader(input_stream, (3 + is_v4) + 2 + len + 1);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2020-10-16 18:49:33 +08:00
|
|
|
const auto *run_header = is_v4 ? v4::run_req_header : run_req_header;
|
2021-02-18 22:32:43 +08:00
|
|
|
const auto run_header_size = is_v4 ? sizeof(v4::run_req_header) : sizeof(run_req_header);
|
2017-04-15 21:14:12 +08:00
|
|
|
// write string header
|
2020-10-16 18:49:33 +08:00
|
|
|
input_stream.Write(run_header, run_header_size);
|
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
|
|
|
|
2020-10-16 18:49:33 +08:00
|
|
|
if (is_v4) {
|
|
|
|
// write empty map for extra field
|
|
|
|
input_stream.Write("\xA0", 1); // TinyMap
|
|
|
|
}
|
|
|
|
|
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);
|
2021-02-18 22:32:43 +08:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2020-10-16 18:49:33 +08:00
|
|
|
TEST(BoltSession, HandshakeMultiVersionRequest) {
|
|
|
|
// Should pick the first version, 4.0, even though a higher version is present
|
|
|
|
// but with a lower priority
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
2021-02-18 22:32:43 +08:00
|
|
|
const uint8_t priority_request[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00,
|
|
|
|
0x01, 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
2020-10-16 18:49:33 +08:00
|
|
|
const uint8_t priority_response[] = {0x00, 0x00, 0x00, 0x04};
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output, priority_request, priority_response);
|
2020-10-16 18:49:33 +08:00
|
|
|
ASSERT_EQ(session.version_.minor, 0);
|
|
|
|
ASSERT_EQ(session.version_.major, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should pick the second version, 4.1, because first, 3.0, is not supported
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
2021-02-18 22:32:43 +08:00
|
|
|
const uint8_t unsupported_first_request[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x00, 0x00, 0x03, 0x00, 0x00,
|
|
|
|
0x01, 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
2020-10-16 18:49:33 +08:00
|
|
|
const uint8_t unsupported_first_response[] = {0x00, 0x00, 0x01, 0x04};
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output, unsupported_first_request, unsupported_first_response);
|
2020-10-16 18:49:33 +08:00
|
|
|
ASSERT_EQ(session.version_.minor, 1);
|
|
|
|
ASSERT_EQ(session.version_.major, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
// No supported version present in the request
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
2021-02-18 22:32:43 +08:00
|
|
|
const uint8_t no_supported_versions_request[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x00, 0x00, 0x03, 0x00, 0x00,
|
|
|
|
0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
|
|
|
ASSERT_THROW(ExecuteHandshake(input_stream, session, output, no_supported_versions_request), SessionException);
|
2020-10-16 18:49:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-28 19:05:09 +08:00
|
|
|
TEST(BoltSession, HandshakeWithVersionOffset) {
|
|
|
|
// It pick the versions depending on the offset given by the second byte
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
const uint8_t priority_request[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x03, 0x03, 0x04, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
|
|
|
const uint8_t priority_response[] = {0x00, 0x00, 0x03, 0x04};
|
|
|
|
ExecuteHandshake(input_stream, session, output, priority_request, priority_response);
|
|
|
|
ASSERT_EQ(session.version_.minor, 3);
|
|
|
|
ASSERT_EQ(session.version_.major, 4);
|
|
|
|
}
|
|
|
|
// This should pick 4.3 version since 4.4 and 4.5 are not existant
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
const uint8_t priority_request[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x03, 0x05, 0x04, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
|
|
|
const uint8_t priority_response[] = {0x00, 0x00, 0x03, 0x04};
|
|
|
|
ExecuteHandshake(input_stream, session, output, priority_request, priority_response);
|
|
|
|
ASSERT_EQ(session.version_.minor, 3);
|
|
|
|
ASSERT_EQ(session.version_.major, 4);
|
|
|
|
}
|
|
|
|
// With multiple offsets
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
const uint8_t priority_request[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x03, 0x03, 0x07, 0x00, 0x03,
|
|
|
|
0x03, 0x06, 0x00, 0x03, 0x03, 0x05, 0x00, 0x03, 0x03, 0x04};
|
|
|
|
const uint8_t priority_response[] = {0x00, 0x00, 0x03, 0x04};
|
|
|
|
ExecuteHandshake(input_stream, session, output, priority_request, priority_response);
|
|
|
|
ASSERT_EQ(session.version_.minor, 3);
|
|
|
|
ASSERT_EQ(session.version_.major, 4);
|
|
|
|
}
|
|
|
|
// Offset overflows
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
const uint8_t priority_request[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x07, 0x06, 0x04, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
|
|
|
const uint8_t priority_response[] = {0x00, 0x00, 0x03, 0x04};
|
|
|
|
ExecuteHandshake(input_stream, session, output, priority_request, priority_response);
|
|
|
|
ASSERT_EQ(session.version_.minor, 3);
|
|
|
|
ASSERT_EQ(session.version_.major, 4);
|
|
|
|
}
|
|
|
|
// Using offset but no version supported
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
const uint8_t no_supported_versions_request[] = {0x60, 0x60, 0xb0, 0x17, 0x00, 0x03, 0x10, 0x04, 0x00, 0x00,
|
|
|
|
0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
|
|
|
ASSERT_THROW(ExecuteHandshake(input_stream, session, output, no_supported_versions_request), SessionException);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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);
|
2021-02-18 22:32:43 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, run_req_header, sizeof(run_req_header)), 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]};
|
2021-02-18 22:32:43 +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);
|
2021-02-18 22:32:43 +08:00
|
|
|
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);
|
2021-02-18 22:32:43 +08:00
|
|
|
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) {
|
2020-10-16 18:49:33 +08:00
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output, v4::handshake_req, v4::handshake_resp);
|
2020-10-16 18:49:33 +08:00
|
|
|
ExecuteInit(input_stream, session, output, true);
|
|
|
|
}
|
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]};
|
2021-02-18 22:32:43 +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) {
|
2021-04-01 23:08:40 +08:00
|
|
|
std::array<uint8_t, 6> run_req_without_parameters{
|
|
|
|
run_req_header[0], run_req_header[1], run_req_header[2], 0x00, 0x00, 0x00};
|
2017-04-15 21:14:12 +08:00
|
|
|
// test lengths, they test the following situations:
|
|
|
|
// missing header data, missing query data, missing parameters
|
2021-04-01 23:08:40 +08:00
|
|
|
int len[] = {1, 2, run_req_without_parameters.size()};
|
2017-04-15 21:14:12 +08:00
|
|
|
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);
|
2021-04-01 23:08:40 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, run_req_without_parameters.data(), 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, 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) {
|
2020-10-16 18:49:33 +08:00
|
|
|
// v1
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
2017-07-18 23:14:43 +08:00
|
|
|
|
2020-10-16 18:49:33 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2017-07-18 23:14:43 +08:00
|
|
|
|
2020-10-16 18:49:33 +08:00
|
|
|
WriteRunRequest(input_stream, kQueryReturn42);
|
|
|
|
session.Execute();
|
2017-07-18 23:14:43 +08:00
|
|
|
|
2020-10-16 18:49:33 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Result);
|
|
|
|
}
|
|
|
|
|
|
|
|
// v4+
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output, v4::handshake_req, v4::handshake_resp);
|
2020-10-16 18:49:33 +08:00
|
|
|
ExecuteInit(input_stream, session, output, true);
|
|
|
|
|
|
|
|
WriteRunRequest(input_stream, kQueryReturn42, true);
|
|
|
|
session.Execute();
|
|
|
|
|
|
|
|
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]};
|
2021-02-18 22:32:43 +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);
|
2021-02-18 22:32:43 +08:00
|
|
|
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
|
2020-10-16 18:49:33 +08:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
|
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
|
|
|
WriteRunRequest(input_stream, kQueryReturn42);
|
|
|
|
session.Execute();
|
|
|
|
|
|
|
|
if (j == 1) output.clear();
|
|
|
|
|
|
|
|
output_stream.SetWriteSuccess(j == 0);
|
|
|
|
if (j == 0) {
|
|
|
|
ExecuteCommand(input_stream, session, dataset[i], 2);
|
|
|
|
} else {
|
2021-02-18 22:32:43 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, dataset[i], 2), SessionException);
|
2020-10-16 18:49:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (j == 0) {
|
|
|
|
ASSERT_EQ(session.state_, State::Idle);
|
|
|
|
ASSERT_FALSE(session.encoder_buffer_.HasData());
|
|
|
|
PrintOutput(output);
|
|
|
|
} else {
|
|
|
|
ASSERT_EQ(session.state_, State::Close);
|
|
|
|
ASSERT_EQ(output.size(), 0);
|
|
|
|
}
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, ExecuteInvalidMessage) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2021-02-18 22:32:43 +08:00
|
|
|
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 {
|
2021-02-18 22:32:43 +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
|
2021-02-18 22:32:43 +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]};
|
2021-02-18 22:32:43 +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) {
|
2020-10-16 18:49:33 +08:00
|
|
|
{
|
2022-08-26 19:19:27 +08:00
|
|
|
SCOPED_TRACE("v1");
|
2020-10-16 18:49:33 +08:00
|
|
|
// test ACK_FAILURE and RESET
|
|
|
|
const uint8_t *dataset[] = {ackfailure_req, reset_req};
|
|
|
|
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
2022-08-26 19:19:27 +08:00
|
|
|
SCOPED_TRACE("i: " + std::to_string(i));
|
2020-10-16 18:49:33 +08:00
|
|
|
// first test with socket write success, then with socket write fail
|
|
|
|
for (int j = 0; j < 2; ++j) {
|
2022-08-26 19:19:27 +08:00
|
|
|
SCOPED_TRACE("j: " + std::to_string(j));
|
|
|
|
const auto write_success = j == 0;
|
2020-10-16 18:49:33 +08:00
|
|
|
INIT_VARS;
|
|
|
|
|
|
|
|
ExecuteHandshake(input_stream, session, output);
|
2022-08-26 19:19:27 +08:00
|
|
|
ASSERT_EQ(session.version_.major, 1U);
|
2020-10-16 18:49:33 +08:00
|
|
|
|
2022-08-26 19:19:27 +08:00
|
|
|
ExecuteInit(input_stream, session, output);
|
2020-10-16 18:49:33 +08:00
|
|
|
WriteRunRequest(input_stream, kInvalidQuery);
|
|
|
|
session.Execute();
|
|
|
|
|
|
|
|
output.clear();
|
|
|
|
|
2022-08-26 19:19:27 +08:00
|
|
|
output_stream.SetWriteSuccess(write_success);
|
|
|
|
if (write_success) {
|
2020-10-16 18:49:33 +08:00
|
|
|
ExecuteCommand(input_stream, session, dataset[i], 2);
|
|
|
|
} else {
|
2021-02-18 22:32:43 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, dataset[i], 2), SessionException);
|
2020-10-16 18:49:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// assert that all data from the init message was cleaned up
|
2022-08-26 19:19:27 +08:00
|
|
|
EXPECT_EQ(session.decoder_buffer_.Size(), 0);
|
2020-10-16 18:49:33 +08:00
|
|
|
|
2022-08-26 19:19:27 +08:00
|
|
|
if (write_success) {
|
|
|
|
EXPECT_EQ(session.state_, State::Idle);
|
2020-10-16 18:49:33 +08:00
|
|
|
CheckOutput(output, success_resp, sizeof(success_resp));
|
|
|
|
} else {
|
2022-08-26 19:19:27 +08:00
|
|
|
EXPECT_EQ(session.state_, State::Close);
|
|
|
|
EXPECT_EQ(output.size(), 0);
|
2020-10-16 18:49:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2020-10-16 18:49:33 +08:00
|
|
|
{
|
2022-08-26 19:19:27 +08:00
|
|
|
SCOPED_TRACE("v4");
|
2020-10-16 18:49:33 +08:00
|
|
|
const uint8_t *dataset[] = {ackfailure_req, v4::reset_req};
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
2022-08-26 19:19:27 +08:00
|
|
|
SCOPED_TRACE("i: " + std::to_string(i));
|
|
|
|
// first test with socket write success, then with socket write fail
|
|
|
|
for (int j = 0; j < 2; ++j) {
|
|
|
|
SCOPED_TRACE("j: " + std::to_string(j));
|
|
|
|
const auto write_success = j == 0;
|
|
|
|
const auto is_reset = i == 1;
|
|
|
|
INIT_VARS;
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2022-08-26 19:19:27 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output, v4::handshake_req, v4::handshake_resp);
|
|
|
|
ASSERT_EQ(session.version_.major, 4U);
|
|
|
|
ExecuteInit(input_stream, session, output, true);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2022-08-26 19:19:27 +08:00
|
|
|
WriteRunRequest(input_stream, kInvalidQuery, true);
|
|
|
|
session.Execute();
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2022-08-26 19:19:27 +08:00
|
|
|
output.clear();
|
|
|
|
output_stream.SetWriteSuccess(write_success);
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2022-08-26 19:19:27 +08:00
|
|
|
// ACK_FAILURE does not exist in v3+, ingored message is sent
|
|
|
|
if (write_success) {
|
|
|
|
ExecuteCommand(input_stream, session, dataset[i], 2);
|
|
|
|
} else {
|
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, dataset[i], 2), SessionException);
|
|
|
|
}
|
2017-04-15 21:14:12 +08:00
|
|
|
|
2022-08-26 19:19:27 +08:00
|
|
|
if (write_success) {
|
|
|
|
if (is_reset) {
|
|
|
|
EXPECT_EQ(session.state_, State::Idle);
|
|
|
|
CheckOutput(output, success_resp, sizeof(success_resp));
|
|
|
|
} else {
|
|
|
|
ASSERT_EQ(session.state_, State::Error);
|
|
|
|
CheckOutput(output, ignored_resp, sizeof(ignored_resp));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
EXPECT_EQ(session.state_, State::Close);
|
|
|
|
}
|
2017-04-15 21:14:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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};
|
2021-02-18 22:32:43 +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);
|
|
|
|
}
|
|
|
|
|
2020-10-16 18:49:33 +08:00
|
|
|
TEST(BoltSession, PartialPull) {
|
|
|
|
INIT_VARS;
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output, v4::handshake_req, v4::handshake_resp);
|
2020-10-16 18:49:33 +08:00
|
|
|
ExecuteInit(input_stream, session, output, true);
|
|
|
|
|
|
|
|
WriteRunRequest(input_stream, kQueryReturnMultiple, true);
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteCommand(input_stream, session, v4::pull_one_req, sizeof(v4::pull_one_req));
|
2020-10-16 18:49:33 +08:00
|
|
|
|
|
|
|
// Not all results were pulled
|
|
|
|
ASSERT_EQ(session.state_, State::Result);
|
|
|
|
PrintOutput(output);
|
|
|
|
|
2021-10-28 19:05:09 +08:00
|
|
|
int len{0}, num{0};
|
2020-10-16 18:49:33 +08:00
|
|
|
while (output.size() > 0) {
|
|
|
|
len = (output[0] << 8) + output[1];
|
|
|
|
output.erase(output.begin(), output.begin() + len + 4);
|
|
|
|
++num;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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);
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteCommand(input_stream, session, v4::pullall_req, sizeof(v4::pullall_req));
|
2020-10-16 18:49:33 +08:00
|
|
|
ASSERT_EQ(session.state_, State::Idle);
|
|
|
|
PrintOutput(output);
|
|
|
|
|
|
|
|
len = 0;
|
|
|
|
num = 0;
|
|
|
|
while (output.size() > 0) {
|
|
|
|
len = (output[0] << 8) + output[1];
|
|
|
|
output.erase(output.begin(), output.begin() + len + 4);
|
|
|
|
++num;
|
|
|
|
}
|
|
|
|
|
|
|
|
// First two are the record messages
|
|
|
|
// and the last is a success message with query run metadata
|
|
|
|
ASSERT_EQ(num, 3);
|
|
|
|
}
|
|
|
|
|
2017-04-15 21:14:12 +08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2020-10-16 18:49:33 +08:00
|
|
|
TEST(BoltSession, Goodbye) {
|
|
|
|
// v4 supports goodbye message
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output, v4::handshake_req, v4::handshake_resp);
|
2020-10-16 18:49:33 +08:00
|
|
|
ExecuteInit(input_stream, session, output, true);
|
2021-02-18 22:32:43 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, v4::goodbye, sizeof(v4::goodbye)),
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::communication::SessionClosedException);
|
2020-10-16 18:49:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// v1 does not support goodbye message
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
2021-02-18 22:32:43 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, v4::goodbye, sizeof(v4::goodbye)), SessionException);
|
2020-10-16 18:49:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, Noop) {
|
|
|
|
// v4.1 supports NOOP chunk
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output, v4_1::handshake_req, v4_1::handshake_resp);
|
2020-10-16 18:49:33 +08:00
|
|
|
ExecuteCommand(input_stream, session, v4_1::noop, sizeof(v4_1::noop));
|
|
|
|
ExecuteInit(input_stream, session, output, true);
|
|
|
|
ExecuteCommand(input_stream, session, v4_1::noop, sizeof(v4_1::noop));
|
|
|
|
WriteRunRequest(input_stream, kQueryReturn42, true);
|
|
|
|
ExecuteCommand(input_stream, session, v4_1::noop, sizeof(v4_1::noop));
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteCommand(input_stream, session, v4::pullall_req, sizeof(v4::pullall_req));
|
2020-10-16 18:49:33 +08:00
|
|
|
ExecuteCommand(input_stream, session, v4_1::noop, sizeof(v4_1::noop));
|
|
|
|
}
|
|
|
|
|
|
|
|
// v1 does not support NOOP chunk
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output, handshake_req, handshake_resp);
|
2020-10-16 18:49:33 +08:00
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, v4_1::noop, sizeof(v4_1::noop)), SessionException);
|
2020-10-16 18:49:33 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
|
|
|
|
session.state_ = State::Init;
|
|
|
|
ExecuteInit(input_stream, session, output);
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, v4_1::noop, sizeof(v4_1::noop)), SessionException);
|
2020-10-16 18:49:33 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
|
|
|
|
session.state_ = State::Idle;
|
|
|
|
WriteRunRequest(input_stream, kQueryEmpty);
|
|
|
|
session.Execute();
|
|
|
|
CheckSuccessMessage(output);
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, v4_1::noop, sizeof(v4_1::noop)), SessionException);
|
2020-10-16 18:49:33 +08:00
|
|
|
CheckFailureMessage(output);
|
|
|
|
|
|
|
|
session.state_ = State::Result;
|
2021-04-01 23:08:40 +08:00
|
|
|
ExecuteCommand(input_stream, session, pullall_req, sizeof(pullall_req));
|
2020-10-16 18:49:33 +08:00
|
|
|
CheckSuccessMessage(output);
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, v4_1::noop, sizeof(v4_1::noop)), SessionException);
|
2020-10-16 18:49:33 +08:00
|
|
|
}
|
|
|
|
}
|
2021-10-28 19:05:09 +08:00
|
|
|
|
|
|
|
TEST(BoltSession, Route) {
|
|
|
|
// Memgraph does not support route message, but it handles it
|
|
|
|
{
|
2022-08-26 19:19:27 +08:00
|
|
|
SCOPED_TRACE("v1");
|
2021-10-28 19:05:09 +08:00
|
|
|
INIT_VARS;
|
|
|
|
|
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, v4_3::route, sizeof(v4_3::route)), SessionException);
|
2022-08-26 19:19:27 +08:00
|
|
|
EXPECT_EQ(session.state_, State::Close);
|
2021-10-28 19:05:09 +08:00
|
|
|
}
|
|
|
|
{
|
2022-08-26 19:19:27 +08:00
|
|
|
SCOPED_TRACE("v4");
|
2021-10-28 19:05:09 +08:00
|
|
|
INIT_VARS;
|
|
|
|
|
2022-08-26 19:19:27 +08:00
|
|
|
ExecuteHandshake(input_stream, session, output, v4_3::handshake_req, v4_3::handshake_resp);
|
2021-10-28 19:05:09 +08:00
|
|
|
ExecuteInit(input_stream, session, output, true);
|
2022-08-26 19:19:27 +08:00
|
|
|
ASSERT_NO_THROW(ExecuteCommand(input_stream, session, v4_3::route, sizeof(v4_3::route)));
|
|
|
|
static constexpr uint8_t expected_resp[] = {
|
|
|
|
0x00 /*two bytes of chunk header, chunk contains 64 bytes of data*/,
|
|
|
|
0x40,
|
|
|
|
0xb1 /*TinyStruct1*/,
|
|
|
|
0x7f /*Failure*/,
|
|
|
|
0xa2 /*TinyMap with 2 items*/,
|
|
|
|
0x84 /*TinyString with 4 chars*/,
|
|
|
|
'c',
|
|
|
|
'o',
|
|
|
|
'd',
|
|
|
|
'e',
|
|
|
|
0x82 /*TinyString with 2 chars*/,
|
|
|
|
'6',
|
|
|
|
'6',
|
|
|
|
0x87 /*TinyString with 7 chars*/,
|
|
|
|
'm',
|
|
|
|
'e',
|
|
|
|
's',
|
|
|
|
's',
|
|
|
|
'a',
|
|
|
|
'g',
|
|
|
|
'e',
|
|
|
|
0xd0 /*String*/,
|
|
|
|
0x2b /*With 43 chars*/,
|
|
|
|
'R',
|
|
|
|
'o',
|
|
|
|
'u',
|
|
|
|
't',
|
|
|
|
'e',
|
|
|
|
' ',
|
|
|
|
'm',
|
|
|
|
'e',
|
|
|
|
's',
|
|
|
|
's',
|
|
|
|
'a',
|
|
|
|
'g',
|
|
|
|
'e',
|
|
|
|
' ',
|
|
|
|
'i',
|
|
|
|
's',
|
|
|
|
' ',
|
|
|
|
'n',
|
|
|
|
'o',
|
|
|
|
't',
|
|
|
|
' ',
|
|
|
|
's',
|
|
|
|
'u',
|
|
|
|
'p',
|
|
|
|
'p',
|
|
|
|
'o',
|
|
|
|
'r',
|
|
|
|
't',
|
|
|
|
'e',
|
|
|
|
'd',
|
|
|
|
' ',
|
|
|
|
'i',
|
|
|
|
'n',
|
|
|
|
' ',
|
|
|
|
'M',
|
|
|
|
'e',
|
|
|
|
'm',
|
|
|
|
'g',
|
|
|
|
'r',
|
|
|
|
'a',
|
|
|
|
'p',
|
|
|
|
'h',
|
|
|
|
'!',
|
|
|
|
0x00 /*Terminating zeros*/,
|
|
|
|
0x00,
|
|
|
|
};
|
|
|
|
EXPECT_EQ(input_stream.size(), 0U);
|
|
|
|
CheckOutput(output, expected_resp, sizeof(expected_resp));
|
|
|
|
EXPECT_EQ(session.state_, State::Error);
|
|
|
|
|
|
|
|
SCOPED_TRACE("Try to reset connection after ROUTE failed");
|
|
|
|
ASSERT_NO_THROW(ExecuteCommand(input_stream, session, v4::reset_req, sizeof(v4::reset_req)));
|
|
|
|
EXPECT_EQ(input_stream.size(), 0U);
|
|
|
|
CheckOutput(output, success_resp, sizeof(success_resp));
|
|
|
|
EXPECT_EQ(session.state_, State::Idle);
|
2021-10-28 19:05:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BoltSession, Rollback) {
|
|
|
|
// v1 does not support ROLLBACK message
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
|
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, v4::rollback, sizeof(v4::rollback)), SessionException);
|
|
|
|
}
|
|
|
|
// v4 supports ROLLBACK message
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
|
|
|
|
ExecuteHandshake(input_stream, session, output, v4::handshake_req, v4::handshake_resp);
|
|
|
|
ExecuteInit(input_stream, session, output, true);
|
|
|
|
ExecuteCommand(input_stream, session, v4::rollback, sizeof(v4::rollback));
|
|
|
|
|
|
|
|
ASSERT_EQ(session.state_, State::Idle);
|
|
|
|
CheckSuccessMessage(output);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
INIT_VARS;
|
|
|
|
|
|
|
|
ExecuteHandshake(input_stream, session, output, v4::handshake_req, v4::handshake_resp);
|
|
|
|
ASSERT_THROW(ExecuteCommand(input_stream, session, v4::rollback, sizeof(v4::rollback)), SessionException);
|
|
|
|
}
|
|
|
|
}
|
2022-08-26 19:19:27 +08:00
|
|
|
|
|
|
|
TEST(BoltSession, ResetInIdle) {
|
|
|
|
{
|
|
|
|
SCOPED_TRACE("v1");
|
|
|
|
INIT_VARS;
|
|
|
|
|
|
|
|
ExecuteHandshake(input_stream, session, output);
|
|
|
|
ExecuteInit(input_stream, session, output);
|
|
|
|
ASSERT_NO_THROW(ExecuteCommand(input_stream, session, reset_req, sizeof(reset_req)));
|
|
|
|
EXPECT_EQ(session.state_, State::Idle);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
SCOPED_TRACE("v4");
|
|
|
|
INIT_VARS;
|
|
|
|
|
|
|
|
ExecuteHandshake(input_stream, session, output, v4_3::handshake_req, v4_3::handshake_resp);
|
|
|
|
ExecuteInit(input_stream, session, output, true);
|
|
|
|
ASSERT_NO_THROW(ExecuteCommand(input_stream, session, v4::reset_req, sizeof(v4::reset_req)));
|
|
|
|
EXPECT_EQ(session.state_, State::Idle);
|
|
|
|
}
|
|
|
|
}
|