// 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 "pw_rpc/internal/method.h"

#include "pb_decode.h"
#include "pb_encode.h"

namespace pw::rpc::internal {

using std::byte;

Status Method::DecodeRequest(span<const byte> buffer,
                             void* proto_struct) const {
  auto input = pb_istream_from_buffer(
      reinterpret_cast<const pb_byte_t*>(buffer.data()), buffer.size());
  return pb_decode(&input, request_fields_, proto_struct) ? Status::OK
                                                          : Status::INTERNAL;
}

StatusWithSize Method::EncodeResponse(const void* proto_struct,
                                      span<byte> buffer) const {
  auto output = pb_ostream_from_buffer(
      reinterpret_cast<pb_byte_t*>(buffer.data()), buffer.size());
  if (pb_encode(&output, response_fields_, proto_struct)) {
    return StatusWithSize(output.bytes_written);
  }
  return StatusWithSize::INTERNAL;
}

StatusWithSize Method::CallUnary(ServerContext& context,
                                 span<const byte> request_buffer,
                                 span<byte> response_buffer,
                                 void* request_struct,
                                 void* response_struct) const {
  Status status = DecodeRequest(request_buffer, request_struct);
  if (!status.ok()) {
    return StatusWithSize(status, 0);
  }

  status = function_.unary(context, request_struct, response_struct);

  StatusWithSize encoded = EncodeResponse(response_struct, response_buffer);
  if (encoded.ok()) {
    return StatusWithSize(status, encoded.size());
  }
  return encoded;
}

StatusWithSize Method::CallServerStreaming(ServerContext& context,
                                           span<const byte> request_buffer,
                                           void* request_struct) const {
  Status status = DecodeRequest(request_buffer, request_struct);
  if (!status.ok()) {
    return StatusWithSize(status, 0);
  }

  internal::BaseServerWriter server_writer(context);
  return StatusWithSize(
      function_.server_streaming(context, request_struct, server_writer), 0);
}

}  // namespace pw::rpc::internal
