pub struct PendingCursorOperation<'order_by> { /* private fields */ }
Expand description

Use .find_all(), .find(...), or .find_with_args(...) to query documents in a collection.

This object lets you take various actions on the selected documents, such as:

  • .exec(): Immediately return the current state of the documents from the local store.

  • .observe_local(...): Register a callback to receive notifications whenever the documents change in the local store. Note that this does not automatically begin syncing these documents from remote peers, use .subscribe() for that.

  • .subscribe(): Create a Subscription that syncs these documents from remote peers.

  • .limit(...): Limit the number of documents that get returned by this query.

  • .offset(...): Offsets the resulting set of matching documents. Useful for skipping the first N matching documents, such as with paging.

  • .sort(...): Sort the matching documents according to chosen fields and ordering.

  • .remove(): Remove the selected documents from the collection, replacing them with tombstones. If another peer concurrently updates these documents, the update will win (“add-wins”). This removal may be synced to other peers.

  • .evict(): Evicts the selected documents from this peer’s local store. This can be used to reduce resource consumption on peers that have finished using documents. This will not cause remote peers to evict or remove these document, it only impacts the local peer’s store.

  • .update(...): Update the documents via a callback that receives a mutable reference to them.

Typically, an app would call .subscribe() in some part of the application which is long-lived to ensure the device receives updates from the mesh. These updates will be automatically received and written into the local store. Elsewhere, where you need to use this data, .observe_local(...) can be used to notify you of the data, and all subsequent changes to the data.

§Example

use dittolive_ditto::prelude::*;

let collection = ditto.store().collection("cars")?;
let pending_op = collection.find("color == 'blue'");

// Hold a `Subscription` to continuously sync from remote peers
let _subscription = pending_op.subscribe();

// Hold a `LiveQuery` to receive callbacks when selected documents are changed
let _live_query = pending_op.observe_local(|docs, _event| {
    let typed: Vec<serde_json::Value> = docs
        .into_iter()
        .flat_map(|doc| doc.typed::<serde_json::Value>().ok())
        .collect();
    println!("Observed doc changes: {typed:?}");
})?;

Implementations§

source§

impl<'order_by> PendingCursorOperation<'order_by>

source

pub fn exec(&self) -> Result<Vec<BoxedDocument>, DittoError>

Execute the find operation and return the matching documents.

The documents returned represent the current state of the local store, this function does not cause any syncing with remote peers.

§Example
use dittolive_ditto::prelude::*;

let collection = ditto.store().collection("cars")?;
let pending_op = collection.find_all();

// Execute the query to receive Ditto documents
let documents: Vec<_> = pending_op.exec()?;

// Use `doc.typed::<T>` with a `Deserialize` type for typed access
let typed_docs: Vec<serde_json::Value> = documents
    .into_iter()
    .flat_map(|doc| doc.typed::<serde_json::Value>().ok())
    .collect();
source

pub fn observe_local<Handler>( &self, handler: Handler, ) -> Result<LiveQuery, DittoError>
where Handler: EventHandler,

Register a callback that delivers updates when selected documents change.

The callback will be called any time the selected documents are changed in this peer’s local store. It will NOT cause the documents to be synced with remote peers, for that use .subscribe().

Hold the returned LiveQuery for as long as you wish to continue receiving updates. Dropping the LiveQuery will cancel the callbacks.

use dittolive_ditto::prelude::*;

let collection = ditto.store().collection("cars")?;
let pending_op = collection.find("color == 'blue'");

// Hold the LiveQuery to continue receiving callbacks
let _live_query = pending_op.observe_local(|docs, _event| {
    let typed: Vec<serde_json::Value> = docs
        .into_iter()
        .flat_map(|doc| doc.typed::<serde_json::Value>().ok())
        .collect();
    println!("Received doc updates: {typed:?}");
})?;

// Drop the LiveQuery to cancel the callback
drop(_live_query);
source

pub fn subscribe(&self) -> Subscription

Creates a Subscription that syncs the selected documents from remote peers.

Holding the Subscription will cause Ditto to continue syncing the selected documents from remote peers. Dropping the Subscription will cancel the sync with remote peers.

§Example
use dittolive_ditto::prelude::*;

let collection = ditto.store().collection("cars")?;
let pending_op = collection.find_all();

// Hold the Subscription to continue syncing with remote peers
let _subscription = pending_op.subscribe();

// Drop the Subscription to cancel sync with remote peers
drop(_subscription);
§Panics

Panics if the Ditto object has been closed.

source

pub fn limit(&mut self, limit: i32) -> &mut Self

Limit the number of documents that get returned when querying a collection for matching documents.

§Example
use dittolive_ditto::prelude::*;

let collection = ditto.store().collection("cars")?;
let mut pending_op = collection.find_all();

// The number of documents returned is limited to at most 20
let documents: Vec<_> = pending_op.limit(20).exec()?;
assert!(documents.len() <= 20);
source

pub fn offset(&mut self, offset: u32) -> &mut Self

Offset the resulting set of matching documents.

This is useful if you aren’t interested in the first N matching documents for one reason or another. For example, you might already have queried the collection and obtained the first 20 matching documents and so you might want to run the same query as you did previously but ignore the first 20 matching documents, and that is where you would use offset.

source

pub fn sort(&mut self, sort_param: Vec<COrderByParam<'order_by>>) -> &mut Self

Sort the documents that match the query provided in the preceding find-like function call.

Documents that are missing the field to sort by will appear at the beginning of the results when sorting in ascending order.

source

pub fn remove(&self) -> Result<Vec<DocumentId>, DittoError>

Remove the queried documents from this Collection, returning the DocumentIds of all documents removed.

A document removal may sync to other peers, causing them to remove the document as well. However, a concurrent update from a remote peer will win when merging the update with the removal (“add-wins”).

§Example
use dittolive_ditto::prelude::*;

let collection = ditto.store().collection("todos")?;
let pending_op = collection.find("done == true");

// Removing documents returns their IDs
let removed_ids: Vec<DocumentId> = pending_op.remove()?;
source

pub fn evict(&self) -> Result<Vec<DocumentId>, DittoError>

Evict the selected documents from the local peer’s store.

Unlike a removal, an eviction does not sync with other peers. When a document is evicted, it is simply removed from the local store to save disk space.

This returns the DocumentIds of the documents that were evicted.

§Example
use dittolive_ditto::prelude::*;

let collection = ditto.store().collection("cars")?;
let pending_op = collection.find("sold == true");

// Evicting documents returns their DocumentIds
let evicted_ids: Vec<DocumentId> = pending_op.evict()?;
source

pub fn update<Updater>( &self, updater: Updater, ) -> Result<BTreeMap<DocumentId, Vec<UpdateResult>>, DittoError>
where Updater: Fn(&mut [BoxedDocument]),

Update the selected documents by mutating them directly.

This function takes a callback and provides it with a mutable slice of the documents matching the query.

After performing the update, this function returns a map of each DocumentId affected and a list of UpdateResults describing the updates that happened.

§Example
use dittolive_ditto::prelude::*;

let collection = ditto.store().collection("cars")?;
let pending_op = collection.find("model == 'pinto'");

let _update_results = pending_op.update(|docs| {
    for doc in docs {
        doc.set("recalled", true).unwrap();
    }
})?;

Auto Trait Implementations§

§

impl<'order_by> Freeze for PendingCursorOperation<'order_by>

§

impl<'order_by> RefUnwindSafe for PendingCursorOperation<'order_by>

§

impl<'order_by> Send for PendingCursorOperation<'order_by>

§

impl<'order_by> Sync for PendingCursorOperation<'order_by>

§

impl<'order_by> Unpin for PendingCursorOperation<'order_by>

§

impl<'order_by> UnwindSafe for PendingCursorOperation<'order_by>

Blanket Implementations§

source§

impl<T> Any for T
where T: 'static + ?Sized,

source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
source§

impl<T> Borrow<T> for T
where T: ?Sized,

source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
§

impl<T> CompatExt for T

§

fn compat(self) -> Compat<T>

Applies the [Compat] adapter by value. Read more
§

fn compat_ref(&self) -> Compat<&T>

Applies the [Compat] adapter by shared reference. Read more
§

fn compat_mut(&mut self) -> Compat<&mut T>

Applies the [Compat] adapter by mutable reference. Read more
§

impl<T> FitForCBox for T

§

type CBoxWrapped = Box_<T>

source§

impl<T> From<T> for T

source§

fn from(t: T) -> T

Returns the argument unchanged.

§

impl<T> Instrument for T

§

fn instrument(self, span: Span) -> Instrumented<Self>

Instruments this type with the provided [Span], returning an Instrumented wrapper. Read more
§

fn in_current_span(self) -> Instrumented<Self>

Instruments this type with the current Span, returning an Instrumented wrapper. Read more
source§

impl<T, U> Into<U> for T
where U: From<T>,

source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

§

impl<T> ManuallyDropMut for T

§

type Ret = ManuallyDrop<T>

§

fn manually_drop_mut<'__>(&'__ mut self) -> &'__ mut ManuallyDrop<T>

§

impl<T> To for T
where T: ?Sized,

§

fn to<T>(self) -> T
where Self: Into<T>,

Converts to T by calling Into<T>::into.
§

fn try_to<T>(self) -> Result<T, Self::Error>
where Self: TryInto<T>,

Tries to convert to T by calling TryInto<T>::try_into.
source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

§

type Error = Infallible

The type returned in the event of a conversion error.
source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.
§

impl<V, T> VZip<V> for T
where V: MultiLane<T>,

§

fn vzip(self) -> V

§

impl<T> WithSubscriber for T

§

fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>
where S: Into<Dispatch>,

Attaches the provided Subscriber to this type, returning a [WithDispatch] wrapper. Read more
§

fn with_current_subscriber(self) -> WithDispatch<Self>

Attaches the current default Subscriber to this type, returning a [WithDispatch] wrapper. Read more
§

impl<F> ZeroSizedElseWrathOfTheGඞds for F