blob: da7e00f7f57878ed7128376a1a766e8fcb14c8c3 [file] [log] [blame]
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you 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
//
// http://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.
#ifndef KUDU_RPC_RPC_SIDECAR_H
#define KUDU_RPC_RPC_SIDECAR_H
#include "kudu/gutil/gscoped_ptr.h"
#include "kudu/util/faststring.h"
#include "kudu/util/slice.h"
namespace kudu {
namespace rpc {
// An RpcSidecar is a mechanism which allows replies to RPCs
// to reference blocks of data without extra copies. In other words,
// whenever a protobuf would have a large field where additional copies
// become expensive, one may opt instead to use an RpcSidecar.
//
// The RpcSidecar saves on an additional copy to/from the protobuf on both the
// server and client side. The InboundCall class accepts RpcSidecars, ignorant
// of the form that the sidecar's data is kept in, requiring only that it can
// be represented as a Slice. Data is then immediately copied from the
// Slice returned from AsSlice() to the socket that is responding to the original
// RPC.
//
// In order to distinguish between separate sidecars, whenever a sidecar is
// added to the RPC response on the server side, an index for that sidecar is
// returned. This index must then in some way (i.e., via protobuf) be
// communicated to the client side.
//
// After receiving the RPC response on the client side, OutboundCall decodes
// the original message along with the separate sidecars by using a list
// of sidecar byte offsets that was sent in the message header.
//
// After reconstructing the array of sidecars, the OutboundCall (through
// RpcController's interface) is able to offer retrieval of the sidecar data
// through the same indices that were returned by InboundCall (or indirectly
// through the RpcContext wrapper) on the client side.
class RpcSidecar {
public:
// Generates a sidecar with the parameter faststring as its data.
explicit RpcSidecar(gscoped_ptr<faststring> data) : data_(std::move(data)) {}
// Returns a Slice representation of the sidecar's data.
Slice AsSlice() const { return *data_; }
private:
const gscoped_ptr<faststring> data_;
DISALLOW_COPY_AND_ASSIGN(RpcSidecar);
};
} // namespace rpc
} // namespace kudu
#endif /* KUDU_RPC_RPC_SIDECAR_H */