161 lines
4.8 KiB
C++
161 lines
4.8 KiB
C++
// Copyright 2022 The Pigweed Authors
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
|
|
// use this file except in compliance with the License. You may obtain a copy of
|
|
// the License at
|
|
//
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
// License for the specific language governing permissions and limitations under
|
|
// the License.
|
|
|
|
#include "gtest/gtest.h"
|
|
#include "pw_assert/check.h"
|
|
#include "pw_rpc/benchmark.rpc.pwpb.h"
|
|
#include "pw_rpc/integration_testing.h"
|
|
#include "pw_sync/binary_semaphore.h"
|
|
|
|
namespace pwpb_rpc_test {
|
|
namespace {
|
|
|
|
using namespace std::chrono_literals;
|
|
using pw::ByteSpan;
|
|
using pw::ConstByteSpan;
|
|
using pw::Function;
|
|
using pw::OkStatus;
|
|
using pw::Status;
|
|
|
|
using pw::rpc::pw_rpc::pwpb::Benchmark;
|
|
|
|
constexpr int kIterations = 10;
|
|
|
|
class PayloadReceiver {
|
|
public:
|
|
const char* Wait() {
|
|
PW_CHECK(sem_.try_acquire_for(1500ms));
|
|
return reinterpret_cast<const char*>(payload_.payload.data());
|
|
}
|
|
|
|
Function<void(const pw::rpc::Payload::Message&, Status)> UnaryOnCompleted() {
|
|
return [this](const pw::rpc::Payload::Message& data, Status) {
|
|
CopyPayload(data);
|
|
};
|
|
}
|
|
|
|
Function<void(const pw::rpc::Payload::Message&)> OnNext() {
|
|
return [this](const pw::rpc::Payload::Message& data) { CopyPayload(data); };
|
|
}
|
|
|
|
private:
|
|
void CopyPayload(const pw::rpc::Payload::Message& data) {
|
|
payload_ = data;
|
|
sem_.release();
|
|
}
|
|
|
|
pw::sync::BinarySemaphore sem_;
|
|
pw::rpc::Payload::Message payload_ = {};
|
|
};
|
|
|
|
template <size_t kSize>
|
|
pw::rpc::Payload::Message Payload(const char (&string)[kSize]) {
|
|
static_assert(kSize <= sizeof(pw::rpc::Payload::Message::payload));
|
|
pw::rpc::Payload::Message payload{};
|
|
payload.payload.resize(kSize);
|
|
std::memcpy(payload.payload.data(), string, kSize);
|
|
return payload;
|
|
}
|
|
|
|
const Benchmark::Client kClient(pw::rpc::integration_test::client(),
|
|
pw::rpc::integration_test::kChannelId);
|
|
|
|
TEST(PwpbRpcIntegrationTest, Unary) {
|
|
char value[] = {"hello, world!"};
|
|
|
|
for (int i = 0; i < kIterations; ++i) {
|
|
PayloadReceiver receiver;
|
|
|
|
value[0] = static_cast<char>(i);
|
|
pw::rpc::PwpbUnaryReceiver call =
|
|
kClient.UnaryEcho(Payload(value), receiver.UnaryOnCompleted());
|
|
ASSERT_STREQ(receiver.Wait(), value);
|
|
}
|
|
}
|
|
|
|
TEST(PwpbRpcIntegrationTest, Unary_ReuseCall) {
|
|
pw::rpc::PwpbUnaryReceiver<pw::rpc::Payload::Message> call;
|
|
char value[] = {"O_o "};
|
|
|
|
for (int i = 0; i < kIterations; ++i) {
|
|
PayloadReceiver receiver;
|
|
|
|
value[sizeof(value) - 2] = static_cast<char>(i);
|
|
call = kClient.UnaryEcho(Payload(value), receiver.UnaryOnCompleted());
|
|
ASSERT_STREQ(receiver.Wait(), value);
|
|
}
|
|
}
|
|
|
|
TEST(PwpbRpcIntegrationTest, Unary_DiscardCalls) {
|
|
constexpr int iterations = PW_RPC_USE_GLOBAL_MUTEX ? 10000 : 1;
|
|
for (int i = 0; i < iterations; ++i) {
|
|
kClient.UnaryEcho(Payload("O_o"));
|
|
}
|
|
}
|
|
|
|
TEST(PwpbRpcIntegrationTest, BidirectionalStreaming_MoveCalls) {
|
|
for (int i = 0; i < kIterations; ++i) {
|
|
PayloadReceiver receiver;
|
|
pw::rpc::PwpbClientReaderWriter call =
|
|
kClient.BidirectionalEcho(receiver.OnNext());
|
|
|
|
ASSERT_EQ(OkStatus(), call.Write(Payload("Yello")));
|
|
ASSERT_STREQ(receiver.Wait(), "Yello");
|
|
|
|
pw::rpc::PwpbClientReaderWriter<pw::rpc::Payload::Message,
|
|
pw::rpc::Payload::Message>
|
|
new_call = std::move(call);
|
|
|
|
// NOLINTNEXTLINE(bugprone-use-after-move)
|
|
EXPECT_EQ(Status::FailedPrecondition(), call.Write(Payload("Dello")));
|
|
|
|
ASSERT_EQ(OkStatus(), new_call.Write(Payload("Dello")));
|
|
ASSERT_STREQ(receiver.Wait(), "Dello");
|
|
|
|
call = std::move(new_call);
|
|
|
|
// NOLINTNEXTLINE(bugprone-use-after-move)
|
|
EXPECT_EQ(Status::FailedPrecondition(), new_call.Write(Payload("Dello")));
|
|
|
|
ASSERT_EQ(OkStatus(), call.Write(Payload("???")));
|
|
ASSERT_STREQ(receiver.Wait(), "???");
|
|
|
|
EXPECT_EQ(OkStatus(), call.Cancel());
|
|
EXPECT_EQ(Status::FailedPrecondition(), new_call.Cancel());
|
|
}
|
|
}
|
|
|
|
TEST(PwpbRpcIntegrationTest, BidirectionalStreaming_ReuseCall) {
|
|
pw::rpc::PwpbClientReaderWriter<pw::rpc::Payload::Message,
|
|
pw::rpc::Payload::Message>
|
|
call;
|
|
|
|
for (int i = 0; i < kIterations; ++i) {
|
|
PayloadReceiver receiver;
|
|
call = kClient.BidirectionalEcho(receiver.OnNext());
|
|
|
|
ASSERT_EQ(OkStatus(), call.Write(Payload("Yello")));
|
|
ASSERT_STREQ(receiver.Wait(), "Yello");
|
|
|
|
ASSERT_EQ(OkStatus(), call.Write(Payload("Dello")));
|
|
ASSERT_STREQ(receiver.Wait(), "Dello");
|
|
|
|
ASSERT_EQ(OkStatus(), call.Write(Payload("???")));
|
|
ASSERT_STREQ(receiver.Wait(), "???");
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace pwpb_rpc_test
|