splinter 0.6.14

Splinter is a privacy-focused platform for distributed applications that provides a blockchain-inspired networking environment for communication and transactions between organizations.
Documentation
// Copyright 2018-2022 Cargill Incorporated
//
// Licensed 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.

//! Provides the "count nodes" operation for the `DieselRegistry`.
use std::convert::TryFrom;

use diesel::{dsl::count_star, prelude::*};

use crate::error::InternalError;
use crate::registry::{diesel::schema::splinter_nodes, MetadataPredicate, RegistryError};

use super::{apply_predicate_filters, RegistryOperations};

pub(in crate::registry::diesel) trait RegistryCountNodesOperation {
    fn count_nodes(&self, predicates: &[MetadataPredicate]) -> Result<u32, RegistryError>;
}

impl<'a, C> RegistryCountNodesOperation for RegistryOperations<'a, C>
where
    C: diesel::Connection,
    i64: diesel::deserialize::FromSql<diesel::sql_types::BigInt, C::Backend>,
{
    fn count_nodes(&self, predicates: &[MetadataPredicate]) -> Result<u32, RegistryError> {
        if predicates.is_empty() {
            // No predicates were specified, just count all nodes
            let count = splinter_nodes::table
                .count()
                // Parse as an i64 here because Diesel knows how to convert a `BigInt` into an i64
                .get_result::<i64>(self.conn)?;

            Ok(u32::try_from(count).map_err(|_| {
                RegistryError::InternalError(InternalError::with_message(
                    "The number of nodes is larger than the max u32".to_string(),
                ))
            })?)
        } else {
            let mut query = splinter_nodes::table
                .into_boxed()
                .select(splinter_nodes::all_columns);
            query = apply_predicate_filters(query, predicates);
            let count = query.select(count_star()).first::<i64>(self.conn)?;

            Ok(u32::try_from(count).map_err(|_| {
                RegistryError::InternalError(InternalError::with_message(
                    "The number of nodes is larger than the max u32".to_string(),
                ))
            })?)
        }
    }
}