-
Notifications
You must be signed in to change notification settings - Fork 196
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Remove futures_core::stream::Stream
from public API
#2910
Conversation
A new generated diff is ready to view.
A new doc preview is ready to view. |
A new generated diff is ready to view.
A new doc preview is ready to view. |
This commit appends `release-2023-08-03` to `NOTABLE_SDK_RELEASE_TAGS`. Elements in that `Vec` should be sorted in an ascending order for a function `enabled_feature` to work correctly. The change has been verified by the following (each executed from canary-runner directory) ``` cargo run -- build-bundle \ --sdk-release-tag release-2023-08-03 \ --canary-path ../canary-lambda \ --manifest-only --musl && \ cd ../canary-lambda && \ cargo check ``` ``` cargo run -- build-bundle \ --sdk-release-tag release-2023-05-24 \ --canary-path ../canary-lambda \ --manifest-only --musl && \ cd ../canary-lambda && \ cargo check ``` ``` cargo run -- build-bundle \ --sdk-release-tag release-2023-12-14 \ --canary-path ../canary-lambda \ --manifest-only --musl && \ cd ../canary-lambda && \ cargo check
A new generated diff is ready to view.
A new doc preview is ready to view. |
A new generated diff is ready to view.
A new doc preview is ready to view. |
This commit addresses #2910 (comment)
This commit addresses #2910 (comment)
This commit addresses #2910 (comment)
A new generated diff is ready to view.
A new doc preview is ready to view. |
...ftware/amazon/smithy/rust/codegen/core/smithy/protocols/HttpBoundProtocolPayloadGenerator.kt
Outdated
Show resolved
Hide resolved
...tware/amazon/smithy/rust/codegen/server/smithy/protocols/ServerHttpBoundProtocolGenerator.kt
Outdated
Show resolved
Hide resolved
...lin/software/amazon/smithy/rust/codegen/core/smithy/customizations/SmithyTypesPubUseExtra.kt
Outdated
Show resolved
Hide resolved
This commit addresses #2910 (comment)
This commit addresses #2910 (comment)
This commit addresses #2910 (comment)
A new generated diff is ready to view.
A new doc preview is ready to view. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM
let stream = #{FnStream}::new(|tx| { | ||
Box::pin(async move { | ||
while let #{Some}(item) = stream.next().await { | ||
tx.send(item).await.expect("send should succeed"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is it safe to panic here?
message = """ | ||
The `futures_core::stream::Stream` trait has been removed from public API. The methods that were made available through the `Stream` trait have been removed from `FnStream`, `TryFlatMap`, ByteStream`, `EventStreamSender`, and `MessageStreamAdapter`. However, we have preserved `.next()` and `.collect()` to continue supporting existing call sites in `smithy-rs` and `aws-sdk-rust`, including tests and rustdocs. If we need to support missing stream operations, we are planning to do so in an additive, backward compatible manner. | ||
|
||
If your code uses a `stream!` macro from the `async_stream` crate to generate stream data, it needs to be replaced by `aws_smithy_async::future::fn_steram::FnStream`. See https://github.com/awslabs/smithy-rs/discussions/2952 for more details. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
fn_steram
-> fn_stream
.
@@ -84,3 +84,21 @@ message = "Generate a region setter when a model uses SigV4." | |||
references = ["smithy-rs#2960"] | |||
meta = { "breaking" = false, "tada" = false, "bug" = true, "target" = "client" } | |||
author = "jdisanti" | |||
|
|||
[[aws-sdk-rust]] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
These entries support Markdown. We should link to the relevant Rust items e.g. https://docs.rs/aws-smithy-async/latest/aws_smithy_async/future/fn_stream/struct.FnStream.html
yield Ok(AudioStream::AudioEvent(AudioEvent::builder().audio_chunk(Blob::new(chunk)).build())); | ||
} | ||
}; | ||
let input_stream = FnStream::new(|tx| { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Out of curiosity. From https://docs.rs/aws-smithy-async/latest/aws_smithy_async/future/fn_stream/struct.FnStream.html:
Because the stream is 1-bounded, the function will not proceed until the stream is read.
Why is the channel 1-bounded? My understanding is that the user sends stream events to the channel and the client polls and dequeues them to send them through the network. What would be bad about allowing the channel to be unbounded?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
FnStream is an implementation of generators–the network isn't involved in all use cases. By bounding it, you all the caller to avoid using more memory than is necessary
@@ -6,12 +6,14 @@ | |||
//! Utility to drive a stream with an async function and a channel. | |||
|
|||
use crate::future::rendezvous; | |||
use futures_util::StreamExt; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: I guess we could mention in the changelog that the type now doesn't implement this or TryStream
, TryStreamExt
either, for completion.
// `aws_smithy_http_server_python::types::ByteStream` already implements | ||
// `futures::stream::Stream`, so no need to wrap it in a futures' stream-compatible |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why the asymmetry though? I'd prefer it if we made aws_smithy_http_server_python::types::ByteStream
more similar to aws_smithy_http::byte_stream::ByteStream
and have both not implement Stream
, and wrap it there too in a HyperBodyWrapByteStream
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The generator code would also become simpler; we wouldn't need to draw distinctions based on what we're generating.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The asymmetry comes from the fact that HyperBodyWrapByteStream
is only capable of wrapping aws_smithy_http::byte_stream::ByteStream
; even if we dropped impl Stream for aws_smithy_http_server_python::types::ByteStream
, it could not be wrapped in HyperBodyWrapByteStream
.
For this reason, if something already implements the Stream
trait, it's better to leave it as-is since we don't need to wrap it in a new-type to enable the trait.
// `FnStream` does not have a `Sync` bound but this struct needs to be `Sync` | ||
// as demonstrated by a unit test `event_stream_sender_send_sync`. | ||
// Wrapping `input_stream` with a `Mutex` will make `EventStreamSender` `Sync`. | ||
input_stream: Mutex<FnStream<Result<T, E>>>, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why does EventStreamSender
need to be Sync
? i.e. why does that test exist?
crate::error::CapturePokemonEventsError::InvalidPokeballError( | ||
crate::error::InvalidPokeballError { | ||
pokeball: pokeball.to_owned() | ||
} | ||
) | ||
); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Related to Burak's comment, I'm having trouble understanding when the receiver can disconnect and this can panic. Can we document that somewhere?
- If the receiver can indeed drop, we shouldn't have examples where we panic, and instead teach users in the example how best to handle the disconnection.
- If the receiver can't really drop, we shouldn't expose a fallible interface.
@@ -48,7 +48,8 @@ | |||
//! | |||
//! ### Stream a ByteStream into a file | |||
//! The previous example is recommended in cases where loading the entire file into memory first is desirable. For extremely large | |||
//! files, you may wish to stream the data directly to the file system, chunk by chunk. This is posible using the `futures::Stream` implementation. | |||
//! files, you may wish to stream the data directly to the file system, chunk by chunk. | |||
//! This is possible using the [`.next()`](crate::byte_stream::ByteStream::next). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
//! This is possible using the [`.next()`](crate::byte_stream::ByteStream::next). | |
//! This is possible using the [`.next()`](crate::byte_stream::ByteStream::next) method. |
This commit addresses #2910 (comment). Now that `HyperBodyWrapByteStream` is also used outside the context of `hyper::body::Body::wrap_stream` (passed to `tokio_util::io::StreamReader`), the module has been renamed to `futures_stream_adapter`. Furthermore, `HyperBodyWrapByteStream` and `HyperBodyWrapEventStream` have been renamed to `FuturesStreamCompatByteStream` and `FuturesStreamCompatEventStream` respectively.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
need more discussion here about how we will do this with paginators. I think we should probably split out some smaller PRs—I don't think FnStream
is the interface we want to be generally exposing to people—it's not really use friendly and has a confusing name.
When stream is used as an input type (e.g. transcribe streaming) I think we may want to attempt to retain some amount of compatibility with the Stream
trait, possibly via the smithy conversions / compatibility crate (or our own trait?)
I don't think that making it a concrete struct is really the right option here.
Can we make this PR much smaller to start piecing off individual chunks?
yield Ok(AudioStream::AudioEvent(AudioEvent::builder().audio_chunk(Blob::new(chunk)).build())); | ||
} | ||
}; | ||
let input_stream = FnStream::new(|tx| { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
FnStream is an implementation of generators–the network isn't involved in all use cases. By bounding it, you all the caller to avoid using more memory than is necessary
pub fn send(self) -> impl #{Stream}<Item = #{item_type}> + #{Unpin} { | ||
/// _Note:_ No requests will be dispatched until the stream is used | ||
/// (e.g. with [`.next().await`](aws_smithy_async::future::fn_stream::FnStream::next)). | ||
pub fn send(self) -> #{fn_stream}::FnStream<#{item_type}> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
hmmm...not sure I like this. It's pretty constraining and also exposes a weird name to customers. I think we should wrap this in a newtype so we aren't publicly exposing the FnStream
internals
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can we make this PR much smaller to start piecing off individual chunks?
We can create a smaller PR just to remove impl Stream
from aws-smithy-async
and but keep it for event streaming (that'll take care of Burak & David's questions around FnStream
since the code around event stream will be unchanged).
The above send
methods needs to return a new-type hiding FnStream
.
A new generated diff is ready to view.
A new doc preview is ready to view. |
A new generated diff is ready to view.
A new doc preview is ready to view. |
Closing this PR for the reason #2910 (comment). |
…eam::ByteStream` (#2983) ## Motivation and Context Removes `futures_core::stream::Stream` from `aws_smithy_http::byte_stream::ByteStream` ## Description This PR is part of our ongoing effort, #2413. We remove the `futures_core::stream::Stream` trait from `aws_smithy_http::byte_stream::ByteStream`. To continue support existing places that relied upon `ByteStream` implementing the `Stream` trait, we provide explicit `.next`, `.try_next`, and `.size_hints` methods on that type. As part of it, a doc-hidden type `FuturesStreamCompatByteStream`, which implements `Stream`, has been added to `aws_smithy_http` to cater to those who need a type implementing `Stream` (see [discussion](#2910 (comment)) why doc-hidden). Another place we need to update is codegen responsible for rendering stream payload serializer, and this affects the server. The regular server and the python server have slightly different rendering requirements, since the former uses `aws_smithy_http::byte_stream::ByteStream` (that does _not_ implement the `Stream` trait) and latter uses its own [ByteStream](https://github.com/awslabs/smithy-rs/blob/cb79a68e3c38d1e62d3980d5e7baedc1144bacc7/rust-runtime/aws-smithy-http-server-python/src/types.rs#L343) (that does implement `Stream`). We use `ServerHttpBoundProtocolCustomization` to handle the difference: `StreamPayloadSerializerCustomization` and `PythonServerStreamPayloadSerializerCustomization`. ## Testing No new behavior has been added, relied on the existing tests in CI. ## Checklist - [x] I have updated `CHANGELOG.next.toml` if I made changes to the smithy-rs codegen or runtime crates - [x] I have updated `CHANGELOG.next.toml` if I made changes to the AWS SDK, generated SDK code, or SDK runtime crates ---- _By submitting this pull request, I confirm that you can use, modify, copy, and redistribute this contribution, under the terms of your choice._
Motivation and Context
Addresses boxes 2 and 5 in #2413
Description
This PR removes the
futures_core::stream::Stream
trait from public API, as enforced by our tool check-external-types. In our codebase, we previously relied on theStream
trait for the following use cases:.next().await
to retrieve items from a stream.collect().await
to dump elements into a collectionimpl Stream
Removing the
Stream
trait from the use cases above affected the following types:aws_smithy_async::fn_steram::{FnStream, TryFlatMap}
aws_smithy_http::byte_stream::ByteStream
aws_smithy_http::event_stream::sender::{EventStreamSender, MessageStreamAdapter}
The primary approach taken in this PR involves
async fn next(&mut self) -> XXX
for types that used to implement theimpl Stream
.collect
stream extension method forFnStream
since it was previously used in its exampleFnStream
at its core when types needed to hold/pass/returnStream
trait objectsFnStream
to replace usages of astream!
macro fromasync_stream
HyperBodyWrapEventStream
andHyperBodyWrapByteStream
for those that were previously passed tohyper::body::Body::wrap_stream
, which requiresimpl Stream
, but no longer implement the trait.Testing
Ported existing tests (many of which used a
stream!
macro from theasync-stream
crate) to useFnStream
.Checklist
CHANGELOG.next.toml
if I made changes to the smithy-rs codegen or runtime cratesCHANGELOG.next.toml
if I made changes to the AWS SDK, generated SDK code, or SDK runtime cratesBy submitting this pull request, I confirm that you can use, modify, copy, and redistribute this contribution, under the terms of your choice.