Skip to content

Commit 339d366

Browse files
authored
RUST-1983 Cherry-picks for 3.0.1 (#1164)
1 parent 7bc5b39 commit 339d366

File tree

4 files changed

+58
-23
lines changed

4 files changed

+58
-23
lines changed

src/action/aggregate.rs

+8-7
Original file line numberDiff line numberDiff line change
@@ -128,7 +128,7 @@ impl<'a, Session, T> Aggregate<'a, Session, T> {
128128
);
129129
}
130130

131-
impl<'a> Aggregate<'a, ImplicitSession> {
131+
impl<'a, T> Aggregate<'a, ImplicitSession, T> {
132132
/// Use the provided session when running the operation.
133133
pub fn session(
134134
self,
@@ -144,7 +144,7 @@ impl<'a> Aggregate<'a, ImplicitSession> {
144144
}
145145
}
146146

147-
impl<'a, Session> Aggregate<'a, Session, Document> {
147+
impl<'a, Session, T> Aggregate<'a, Session, T> {
148148
/// Use the provided type for the returned cursor.
149149
///
150150
/// ```rust
@@ -167,7 +167,7 @@ impl<'a, Session> Aggregate<'a, Session, Document> {
167167
/// # Ok(())
168168
/// # }
169169
/// ```
170-
pub fn with_type<T>(self) -> Aggregate<'a, Session, T> {
170+
pub fn with_type<U>(self) -> Aggregate<'a, Session, U> {
171171
Aggregate {
172172
target: self.target,
173173
pipeline: self.pipeline,
@@ -199,11 +199,11 @@ impl<'a, T> Action for Aggregate<'a, ImplicitSession, T> {
199199
}
200200
}
201201

202-
#[action_impl(sync = crate::sync::SessionCursor<Document>)]
203-
impl<'a> Action for Aggregate<'a, ExplicitSession<'a>> {
202+
#[action_impl(sync = crate::sync::SessionCursor<T>)]
203+
impl<'a, T> Action for Aggregate<'a, ExplicitSession<'a>, T> {
204204
type Future = AggregateSessionFuture;
205205

206-
async fn execute(mut self) -> Result<SessionCursor<Document>> {
206+
async fn execute(mut self) -> Result<SessionCursor<T>> {
207207
resolve_read_concern_with_session!(self.target, self.options, Some(&mut *self.session.0))?;
208208
resolve_write_concern_with_session!(self.target, self.options, Some(&mut *self.session.0))?;
209209
resolve_selection_criteria_with_session!(
@@ -218,8 +218,9 @@ impl<'a> Action for Aggregate<'a, ExplicitSession<'a>> {
218218
self.options,
219219
);
220220
let client = self.target.client();
221+
let session = self.session;
221222
client
222-
.execute_session_cursor_operation(aggregate, self.session.0)
223+
.execute_session_cursor_operation(aggregate, session.0)
223224
.await
224225
}
225226
}

src/action/watch.rs

+20-15
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
1-
use std::time::Duration;
1+
use std::{marker::PhantomData, time::Duration};
22

33
use bson::{Bson, Document, Timestamp};
4+
use serde::de::DeserializeOwned;
45

56
use super::{action_impl, deeplink, option_setters, ExplicitSession, ImplicitSession};
67
use crate::{
@@ -96,11 +97,11 @@ where
9697
/// Change streams require either a "majority" read concern or no read concern. Anything else
9798
/// will cause a server error.
9899
///
99-
/// `await` will return d[`Result<ChangeStream<ChangeStreamEvent<Document>>>`] or
100-
/// d[`Result<SessionChangeStream<ChangeStreamEvent<Document>>>`] if a
100+
/// `await` will return d[`Result<ChangeStream<ChangeStreamEvent<T>>>`] or
101+
/// d[`Result<SessionChangeStream<ChangeStreamEvent<T>>>`] if a
101102
/// [`ClientSession`] has been provided.
102103
#[deeplink]
103-
pub fn watch(&self) -> Watch {
104+
pub fn watch(&self) -> Watch<T> {
104105
Watch::new(self.client(), self.namespace().into())
105106
}
106107
}
@@ -153,24 +154,25 @@ where
153154
///
154155
/// Change streams require either a "majority" read concern or no read concern. Anything else
155156
/// will cause a server error.
156-
pub fn watch(&self) -> Watch {
157+
pub fn watch(&self) -> Watch<T> {
157158
self.async_collection.watch()
158159
}
159160
}
160161

161162
/// Starts a new [`ChangeStream`] that receives events for all changes in a given scope. Create by
162163
/// calling [`Client::watch`], [`Database::watch`], or [`Collection::watch`].
163164
#[must_use]
164-
pub struct Watch<'a, S = ImplicitSession> {
165+
pub struct Watch<'a, T = Document, S = ImplicitSession> {
165166
client: &'a Client,
166167
target: AggregateTarget,
167168
pipeline: Vec<Document>,
168169
options: Option<ChangeStreamOptions>,
169170
session: S,
170171
cluster: bool,
172+
phantom: PhantomData<fn() -> T>,
171173
}
172174

173-
impl<'a> Watch<'a, ImplicitSession> {
175+
impl<'a, T> Watch<'a, T, ImplicitSession> {
174176
fn new(client: &'a Client, target: AggregateTarget) -> Self {
175177
Self {
176178
client,
@@ -179,6 +181,7 @@ impl<'a> Watch<'a, ImplicitSession> {
179181
options: None,
180182
session: ImplicitSession,
181183
cluster: false,
184+
phantom: PhantomData,
182185
}
183186
}
184187

@@ -190,6 +193,7 @@ impl<'a> Watch<'a, ImplicitSession> {
190193
options: None,
191194
session: ImplicitSession,
192195
cluster: true,
196+
phantom: PhantomData,
193197
}
194198
}
195199
}
@@ -235,28 +239,29 @@ impl<'a, S> Watch<'a, S> {
235239
);
236240
}
237241

238-
impl<'a> Watch<'a, ImplicitSession> {
242+
impl<'a, T> Watch<'a, T, ImplicitSession> {
239243
/// Use the provided ['ClientSession'].
240244
pub fn session<'s>(
241245
self,
242246
session: impl Into<&'s mut ClientSession>,
243-
) -> Watch<'a, ExplicitSession<'s>> {
247+
) -> Watch<'a, T, ExplicitSession<'s>> {
244248
Watch {
245249
client: self.client,
246250
target: self.target,
247251
pipeline: self.pipeline,
248252
options: self.options,
249253
session: ExplicitSession(session.into()),
250254
cluster: self.cluster,
255+
phantom: PhantomData,
251256
}
252257
}
253258
}
254259

255-
#[action_impl(sync = crate::sync::ChangeStream<ChangeStreamEvent<Document>>)]
256-
impl<'a> Action for Watch<'a, ImplicitSession> {
260+
#[action_impl(sync = crate::sync::ChangeStream<ChangeStreamEvent<T>>)]
261+
impl<'a, T: DeserializeOwned + Unpin + Send + Sync> Action for Watch<'a, T, ImplicitSession> {
257262
type Future = WatchFuture;
258263

259-
async fn execute(mut self) -> Result<ChangeStream<ChangeStreamEvent<Document>>> {
264+
async fn execute(mut self) -> Result<ChangeStream<ChangeStreamEvent<T>>> {
260265
resolve_options!(
261266
self.client,
262267
self.options,
@@ -273,11 +278,11 @@ impl<'a> Action for Watch<'a, ImplicitSession> {
273278
}
274279
}
275280

276-
#[action_impl(sync = crate::sync::SessionChangeStream<ChangeStreamEvent<Document>>)]
277-
impl<'a> Action for Watch<'a, ExplicitSession<'a>> {
281+
#[action_impl(sync = crate::sync::SessionChangeStream<ChangeStreamEvent<T>>)]
282+
impl<'a, T: DeserializeOwned + Unpin + Send + Sync> Action for Watch<'a, T, ExplicitSession<'a>> {
278283
type Future = WatchSessionFuture;
279284

280-
async fn execute(mut self) -> Result<SessionChangeStream<ChangeStreamEvent<Document>>> {
285+
async fn execute(mut self) -> Result<SessionChangeStream<ChangeStreamEvent<T>>> {
281286
resolve_read_concern_with_session!(self.client, self.options, Some(&mut *self.session.0))?;
282287
resolve_selection_criteria_with_session!(
283288
self.client,

src/test/change_stream.rs

+10
Original file line numberDiff line numberDiff line change
@@ -662,3 +662,13 @@ async fn split_large_event() -> Result<()> {
662662

663663
Ok(())
664664
}
665+
666+
// Regression test: `Collection::watch` uses the type parameter. This is not flagged as a test to
667+
// run because it's just asserting that this compiles.
668+
#[allow(unreachable_code, unused_variables, clippy::diverging_sub_expression)]
669+
async fn _collection_watch_typed() {
670+
let coll: Collection<bson::RawDocumentBuf> = unimplemented!();
671+
let mut stream = coll.watch().await.unwrap();
672+
let _: Option<crate::error::Result<ChangeStreamEvent<bson::RawDocumentBuf>>> =
673+
stream.next().await;
674+
}

src/test/db.rs

+20-1
Original file line numberDiff line numberDiff line change
@@ -405,7 +405,26 @@ async fn aggregate_with_generics() {
405405
let _: Cursor<Document> = database.aggregate(pipeline.clone()).await.unwrap();
406406

407407
// Assert that data is properly deserialized when using with_type
408-
let mut cursor = database.aggregate(pipeline).with_type::<A>().await.unwrap();
408+
let mut cursor = database
409+
.aggregate(pipeline.clone())
410+
.with_type::<A>()
411+
.await
412+
.unwrap();
409413
assert!(cursor.advance().await.unwrap());
410414
assert_eq!(&cursor.deserialize_current().unwrap().str, "hi");
415+
416+
// Assert that `with_type` can be used with an explicit session.
417+
let mut session = client.start_session().await.unwrap();
418+
let _ = database
419+
.aggregate(pipeline.clone())
420+
.session(&mut session)
421+
.with_type::<A>()
422+
.await
423+
.unwrap();
424+
let _ = database
425+
.aggregate(pipeline.clone())
426+
.with_type::<A>()
427+
.session(&mut session)
428+
.await
429+
.unwrap();
411430
}

0 commit comments

Comments
 (0)