// Copyright 2021 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_transfer/internal/chunk.h"

#include "pw_protobuf/decoder.h"
#include "pw_status/try.h"
#include "pw_transfer/transfer.pwpb.h"

namespace pw::transfer::internal {

namespace ProtoChunk = transfer::Chunk;

Status DecodeChunk(ConstByteSpan message, Chunk& chunk) {
  protobuf::Decoder decoder(message);
  Status status;
  uint32_t value;

  chunk = {};

  while ((status = decoder.Next()).ok()) {
    ProtoChunk::Fields field =
        static_cast<ProtoChunk::Fields>(decoder.FieldNumber());

    switch (field) {
      case ProtoChunk::Fields::TRANSFER_ID:
        PW_TRY(decoder.ReadUint32(&chunk.transfer_id));
        break;

      case ProtoChunk::Fields::PENDING_BYTES:
        PW_TRY(decoder.ReadUint32(&value));
        chunk.pending_bytes = value;
        break;

      case ProtoChunk::Fields::MAX_CHUNK_SIZE_BYTES:
        PW_TRY(decoder.ReadUint32(&value));
        chunk.max_chunk_size_bytes = value;
        break;

      case ProtoChunk::Fields::MIN_DELAY_MICROSECONDS:
        PW_TRY(decoder.ReadUint32(&value));
        chunk.min_delay_microseconds = value;
        break;

      case ProtoChunk::Fields::OFFSET:
        PW_TRY(decoder.ReadUint32(&chunk.offset));
        break;

      case ProtoChunk::Fields::DATA:
        PW_TRY(decoder.ReadBytes(&chunk.data));
        break;

      case ProtoChunk::Fields::REMAINING_BYTES: {
        uint64_t remaining;
        PW_TRY(decoder.ReadUint64(&remaining));
        chunk.remaining_bytes = remaining;
        break;
      }

      case ProtoChunk::Fields::STATUS:
        PW_TRY(decoder.ReadUint32(&value));
        chunk.status = static_cast<Status::Code>(value);
        break;

      case ProtoChunk::Fields::WINDOW_END_OFFSET:
        PW_TRY(decoder.ReadUint32(&chunk.window_end_offset));
        break;

      case ProtoChunk::Fields::TYPE: {
        uint32_t type;
        PW_TRY(decoder.ReadUint32(&type));
        chunk.type = static_cast<Chunk::Type>(type);
        break;
      }
    }
  }

  return status.IsOutOfRange() ? OkStatus() : status;
}

Result<ConstByteSpan> EncodeChunk(const Chunk& chunk, ByteSpan buffer) {
  ProtoChunk::MemoryEncoder encoder(buffer);

  encoder.WriteTransferId(chunk.transfer_id).IgnoreError();

  if (chunk.window_end_offset != 0) {
    encoder.WriteWindowEndOffset(chunk.window_end_offset).IgnoreError();
  }

  if (chunk.pending_bytes.has_value()) {
    encoder.WritePendingBytes(chunk.pending_bytes.value()).IgnoreError();
  }
  if (chunk.max_chunk_size_bytes.has_value()) {
    encoder.WriteMaxChunkSizeBytes(chunk.max_chunk_size_bytes.value())
        .IgnoreError();
  }
  if (chunk.min_delay_microseconds.has_value()) {
    encoder.WriteMinDelayMicroseconds(chunk.min_delay_microseconds.value())
        .IgnoreError();
  }
  if (chunk.offset != 0) {
    encoder.WriteOffset(chunk.offset).IgnoreError();
  }
  if (!chunk.data.empty()) {
    encoder.WriteData(chunk.data).IgnoreError();
  }
  if (chunk.remaining_bytes.has_value()) {
    encoder.WriteRemainingBytes(chunk.remaining_bytes.value()).IgnoreError();
  }
  if (chunk.status.has_value()) {
    encoder.WriteStatus(chunk.status.value().code()).IgnoreError();
  }

  if (chunk.type.has_value()) {
    encoder.WriteType(static_cast<ProtoChunk::Type>(chunk.type.value()))
        .IgnoreError();
  }

  PW_TRY(encoder.status());
  return ConstByteSpan(encoder);
}

}  // namespace pw::transfer::internal
