blob: 5a8a1f7798c8434f21277a61b9ac0b3fbe777e29 [file] [log] [blame]
// Copyright 2023 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// General-purpose abstractions for read/write streams.
#ifndef QUICHE_COMMON_QUICHE_STREAM_H_
#define QUICHE_COMMON_QUICHE_STREAM_H_
#include <cstddef>
#include <string>
#include "absl/status/status.h"
#include "absl/strings/string_view.h"
#include "absl/types/span.h"
#include "quiche/common/platform/api/quiche_export.h"
#include "quiche/common/quiche_callbacks.h"
namespace quiche {
// A shared base class for read and write stream to support abrupt termination.
class QUICHE_EXPORT TerminableStream {
public:
virtual ~TerminableStream() = default;
// Abruptly terminate the stream due to an error. If `error` is not OK, it may
// carry the error information that could be potentially communicated to the
// peer in case the stream is remote. If the stream is a duplex stream, both
// ends of the stream are terminated.
virtual void AbruptlyTerminate(absl::Status error) = 0;
};
// A general-purpose visitor API that gets notifications for ReadStream-related
// events.
class QUICHE_EXPORT ReadStreamVisitor {
public:
virtual ~ReadStreamVisitor() = default;
// Called whenever the stream has new data available to read. Unless otherwise
// specified, QUICHE stream reads are level-triggered, which means that the
// callback will be called repeatedly as long as there is still data in the
// buffer.
virtual void OnCanRead() = 0;
};
// General purpose abstraction for a stream of data that can be read from the
// network. The class is designed around the idea that a network stream stores
// all of the received data in a sequence of contiguous buffers. Because of
// that, there are two ways to read from a stream:
// - Read() will copy data into a user-provided buffer, reassembling it if it
// is split across multiple buffers internally.
// - PeekNextReadableRegion()/SkipBytes() let the caller access the underlying
// buffers directly, potentially avoiding the copying at the cost of the
// caller having to deal with discontinuities.
class QUICHE_EXPORT ReadStream {
public:
struct QUICHE_EXPORT ReadResult {
// Number of bytes actually read.
size_t bytes_read = 0;
// Whether the FIN has been received; if true, no further data will arrive
// on the stream, and the stream object can be soon potentially garbage
// collected.
bool fin = false;
};
struct PeekResult {
// The next available chunk in the sequencer buffer.
absl::string_view peeked_data;
// True if all of the data up to the FIN has been read.
bool fin_next = false;
// True if all of the data up to the FIN has been received (but not
// necessarily read).
bool all_data_received = false;
// Indicates that `SkipBytes()` will make progress if called.
bool has_data() const { return !peeked_data.empty() || fin_next; }
};
virtual ~ReadStream() = default;
// Reads at most `buffer.size()` bytes into `buffer`.
[[nodiscard]] virtual ReadResult Read(absl::Span<char> buffer) = 0;
// Reads all available data and appends it to the end of `output`.
[[nodiscard]] virtual ReadResult Read(std::string* output) = 0;
// Indicates the total number of bytes that can be read from the stream.
virtual size_t ReadableBytes() const = 0;
// Returns a contiguous buffer to read (or an empty buffer, if there is no
// data to read). See `ProcessAllReadableRegions` below for an example of how
// to use this method while handling FIN correctly.
virtual PeekResult PeekNextReadableRegion() const = 0;
// Equivalent to reading `bytes`, but does not perform any copying. `bytes`
// must be less than or equal to `ReadableBytes()`. The return value indicates
// if the FIN has been reached. `SkipBytes(0)` can be used to consume the FIN
// if it's the only thing remaining on the stream.
[[nodiscard]] virtual bool SkipBytes(size_t bytes) = 0;
};
// Calls `callback` for every contiguous chunk available inside the stream.
// Returns true if the FIN has been reached.
inline bool ProcessAllReadableRegions(
ReadStream& stream, UnretainedCallback<void(absl::string_view)> callback) {
for (;;) {
ReadStream::PeekResult peek_result = stream.PeekNextReadableRegion();
if (!peek_result.has_data()) {
return false;
}
callback(peek_result.peeked_data);
bool fin = stream.SkipBytes(peek_result.peeked_data.size());
if (fin) {
return true;
}
}
}
// A general-purpose visitor API that gets notifications for WriteStream-related
// events.
class QUICHE_EXPORT WriteStreamVisitor {
public:
virtual ~WriteStreamVisitor() {}
// Called whenever the stream is not write-blocked and can accept new data.
virtual void OnCanWrite() = 0;
};
// Options for writing data into a WriteStream.
class QUICHE_EXPORT StreamWriteOptions {
public:
StreamWriteOptions() = default;
// If send_fin() is set to true, the write operation also sends a FIN on the
// stream.
bool send_fin() const { return send_fin_; }
void set_send_fin(bool send_fin) { send_fin_ = send_fin; }
// If buffer_unconditionally() is set to true, the write operation will buffer
// data even if the internal buffer limit is exceeded.
bool buffer_unconditionally() const { return buffer_unconditionally_; }
void set_buffer_unconditionally(bool value) {
buffer_unconditionally_ = value;
}
private:
bool send_fin_ = false;
bool buffer_unconditionally_ = false;
};
inline constexpr StreamWriteOptions kDefaultStreamWriteOptions =
StreamWriteOptions();
// WriteStream is an object that can accept a stream of bytes.
//
// The writes into a WriteStream are all-or-nothing. A WriteStream object has
// to either accept all data written into it by returning absl::OkStatus, or ask
// the caller to try again once via OnCanWrite() by returning
// absl::UnavailableError.
class QUICHE_EXPORT WriteStream {
public:
virtual ~WriteStream() {}
// Writes |data| into the stream.
virtual absl::Status Writev(absl::Span<const absl::string_view> data,
const StreamWriteOptions& options) = 0;
// Indicates whether it is possible to write into stream right now.
virtual bool CanWrite() const = 0;
// Legacy convenience method for writing a single string_view. New users
// should use quiche::WriteIntoStream instead, since this method does not
// return useful failure information.
[[nodiscard]] bool SendFin() {
StreamWriteOptions options;
options.set_send_fin(true);
return Writev(absl::Span<const absl::string_view>(), options).ok();
}
// Legacy convenience method for writing a single string_view. New users
// should use quiche::WriteIntoStream instead, since this method does not
// return useful failure information.
[[nodiscard]] bool Write(absl::string_view data) {
return Writev(absl::MakeSpan(&data, 1), kDefaultStreamWriteOptions).ok();
}
};
// Convenience methods to write a single chunk of data into the stream.
inline absl::Status WriteIntoStream(
WriteStream& stream, absl::string_view data,
const StreamWriteOptions& options = kDefaultStreamWriteOptions) {
return stream.Writev(absl::MakeSpan(&data, 1), options);
}
// Convenience methods to send a FIN on the stream.
inline absl::Status SendFinOnStream(WriteStream& stream) {
StreamWriteOptions options;
options.set_send_fin(true);
return stream.Writev(absl::Span<const absl::string_view>(), options);
}
} // namespace quiche
#endif // QUICHE_COMMON_QUICHE_STREAM_H_