You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
163 lines
4.9 KiB
163 lines
4.9 KiB
4 months ago
|
// Copyright 2020 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_protobuf/decoder.h"
|
||
|
#include "pw_rpc/internal/hash.h"
|
||
|
#include "pw_rpc/raw_test_method_context.h"
|
||
|
#include "pw_rpc_test_protos/test.pwpb.h"
|
||
|
#include "pw_rpc_test_protos/test.raw_rpc.pb.h"
|
||
|
|
||
|
namespace pw::rpc {
|
||
|
namespace test {
|
||
|
|
||
|
class TestService final : public generated::TestService<TestService> {
|
||
|
public:
|
||
|
static StatusWithSize TestRpc(ServerContext&,
|
||
|
ConstByteSpan request,
|
||
|
ByteSpan response) {
|
||
|
int64_t integer;
|
||
|
Status status;
|
||
|
|
||
|
if (!DecodeRequest(request, integer, status)) {
|
||
|
return StatusWithSize::DataLoss();
|
||
|
}
|
||
|
|
||
|
protobuf::NestedEncoder encoder(response);
|
||
|
TestResponse::Encoder test_response(&encoder);
|
||
|
test_response.WriteValue(integer + 1);
|
||
|
|
||
|
return StatusWithSize(status, encoder.Encode().value().size());
|
||
|
}
|
||
|
|
||
|
void TestStreamRpc(ServerContext&,
|
||
|
ConstByteSpan request,
|
||
|
RawServerWriter& writer) {
|
||
|
int64_t integer;
|
||
|
Status status;
|
||
|
|
||
|
ASSERT_TRUE(DecodeRequest(request, integer, status));
|
||
|
for (int i = 0; i < integer; ++i) {
|
||
|
ByteSpan buffer = writer.PayloadBuffer();
|
||
|
protobuf::NestedEncoder encoder(buffer);
|
||
|
TestStreamResponse::Encoder test_stream_response(&encoder);
|
||
|
test_stream_response.WriteNumber(i);
|
||
|
writer.Write(encoder.Encode().value());
|
||
|
}
|
||
|
|
||
|
writer.Finish(status);
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
static bool DecodeRequest(ConstByteSpan request,
|
||
|
int64_t& integer,
|
||
|
Status& status) {
|
||
|
protobuf::Decoder decoder(request);
|
||
|
Status decode_status;
|
||
|
bool has_integer = false;
|
||
|
bool has_status = false;
|
||
|
|
||
|
while (decoder.Next().ok()) {
|
||
|
switch (static_cast<TestRequest::Fields>(decoder.FieldNumber())) {
|
||
|
case TestRequest::Fields::INTEGER:
|
||
|
decode_status = decoder.ReadInt64(&integer);
|
||
|
EXPECT_EQ(OkStatus(), decode_status);
|
||
|
has_integer = decode_status.ok();
|
||
|
break;
|
||
|
case TestRequest::Fields::STATUS_CODE: {
|
||
|
uint32_t status_code;
|
||
|
decode_status = decoder.ReadUint32(&status_code);
|
||
|
EXPECT_EQ(OkStatus(), decode_status);
|
||
|
has_status = decode_status.ok();
|
||
|
status = static_cast<Status::Code>(status_code);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
EXPECT_TRUE(has_integer);
|
||
|
EXPECT_TRUE(has_status);
|
||
|
return has_integer && has_status;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
} // namespace test
|
||
|
|
||
|
namespace {
|
||
|
|
||
|
TEST(RawCodegen, CompilesProperly) {
|
||
|
test::TestService service;
|
||
|
EXPECT_EQ(service.id(), internal::Hash("pw.rpc.test.TestService"));
|
||
|
EXPECT_STREQ(service.name(), "TestService");
|
||
|
}
|
||
|
|
||
|
TEST(RawCodegen, Server_InvokeUnaryRpc) {
|
||
|
PW_RAW_TEST_METHOD_CONTEXT(test::TestService, TestRpc) context;
|
||
|
|
||
|
std::byte buffer[64];
|
||
|
protobuf::NestedEncoder encoder(buffer);
|
||
|
test::TestRequest::Encoder test_request(&encoder);
|
||
|
test_request.WriteInteger(123);
|
||
|
test_request.WriteStatusCode(OkStatus().code());
|
||
|
|
||
|
auto sws = context.call(encoder.Encode().value());
|
||
|
EXPECT_EQ(OkStatus(), sws.status());
|
||
|
|
||
|
protobuf::Decoder decoder(context.response());
|
||
|
|
||
|
while (decoder.Next().ok()) {
|
||
|
switch (static_cast<test::TestResponse::Fields>(decoder.FieldNumber())) {
|
||
|
case test::TestResponse::Fields::VALUE: {
|
||
|
int32_t value;
|
||
|
decoder.ReadInt32(&value);
|
||
|
EXPECT_EQ(value, 124);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
TEST(RawCodegen, Server_InvokeServerStreamingRpc) {
|
||
|
PW_RAW_TEST_METHOD_CONTEXT(test::TestService, TestStreamRpc) context;
|
||
|
|
||
|
std::byte buffer[64];
|
||
|
protobuf::NestedEncoder encoder(buffer);
|
||
|
test::TestRequest::Encoder test_request(&encoder);
|
||
|
test_request.WriteInteger(5);
|
||
|
test_request.WriteStatusCode(Status::Unauthenticated().code());
|
||
|
|
||
|
context.call(encoder.Encode().value());
|
||
|
EXPECT_TRUE(context.done());
|
||
|
EXPECT_EQ(Status::Unauthenticated(), context.status());
|
||
|
EXPECT_EQ(context.total_responses(), 5u);
|
||
|
|
||
|
protobuf::Decoder decoder(context.responses().back());
|
||
|
while (decoder.Next().ok()) {
|
||
|
switch (
|
||
|
static_cast<test::TestStreamResponse::Fields>(decoder.FieldNumber())) {
|
||
|
case test::TestStreamResponse::Fields::NUMBER: {
|
||
|
int32_t value;
|
||
|
decoder.ReadInt32(&value);
|
||
|
EXPECT_EQ(value, 4);
|
||
|
break;
|
||
|
}
|
||
|
case test::TestStreamResponse::Fields::CHUNK:
|
||
|
FAIL();
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
} // namespace
|
||
|
} // namespace pw::rpc
|