Skip to content
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

feat(proto)!: separate client and server feature #118

Open
wants to merge 8 commits into
base: develop
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 7 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
32 changes: 20 additions & 12 deletions abci/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -16,30 +16,38 @@ rust-version.workspace = true
version.workspace = true

[features]
default = [
"server",
"docker-tests",
"crypto",
"tcp",
"unix",
"grpc",
"tracing-span",
]
# By default, include the server feature together with both tcp and unix transports.
default = ["server", "docker-tests", "crypto", "tcp", "unix", "tracing-span"]
# docker-tests includes integration tests that require docker to be available
docker-tests = ["server"]
# Enable server support; this is most likely what you want (plus `tcp` and/or `unix`)
server = [
"tracing-subscriber/fmt",
"tenderdash-proto/server",
"dep:tokio",
"dep:tokio-util",
"dep:futures",
"std",
]
# std is deprecated, use "grpc" instead
std = ["grpc"]
grpc = ["tenderdash-proto/grpc"]
# DEPRECATED; use `server` instead
grpc = ["server"]

# Disable no_std support
std = ["tenderdash-proto/std"]

# Include crypto features, like signature verification
crypto = ["dep:lhash"]

# Include server TCP support
tcp = ["server"]

# Include server unix socket support
unix = ["server"]

# Include `tracing` crate spans
tracing-span = ["dep:uuid"]

# Include `serde` support
serde = ["tenderdash-proto/serde", "dep:serde_json"]

[[example]]
Expand Down
18 changes: 16 additions & 2 deletions abci/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -39,13 +39,27 @@ pub enum Error {
#[error("connection error")]
Connection(#[from] io::Error),
#[error("cannot decode protobuf message")]
Decode(#[from] DecodeError),
Decode(DecodeError),
#[error("cannot encode protobuf message")]
Encode(#[from] EncodeError),
Encode(EncodeError),
#[error("cannot create canonical message: {0}")]
Canonical(String),
#[error("server terminated")]
Cancelled(),
#[error("async runtime error")]
Async(String),
}

// manually implemented due to no_std compatibility
impl From<EncodeError> for Error {
fn from(error: EncodeError) -> Error {
Error::Encode(error)
}
}

// manually implemented due to no_std compatibility
impl From<DecodeError> for Error {
fn from(error: DecodeError) -> Error {
Error::Decode(error)
}
}
2 changes: 1 addition & 1 deletion abci/src/server/generic.rs
Original file line number Diff line number Diff line change
Expand Up @@ -96,7 +96,7 @@ impl<App: RequestDispatcher> GenericServer<App, UnixListener> {
}
}

impl<'a, App: RequestDispatcher + 'a, L: Listener> Server for GenericServer<App, L>
impl<App: RequestDispatcher, L: Listener> Server for GenericServer<App, L>
where
L: Listener + Send + Sync + 'static,
L::Addr: Send + Debug,
Expand Down
5 changes: 0 additions & 5 deletions proto-compiler/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -26,9 +26,4 @@ tonic-build = { version = "0.12.3", optional = true }
[features]
default = []
# Enable gRPC support; needed by server and client features.
# Conflicts with no_std
grpc = ["dep:tonic-build"]
# Build the gRPC server. Requires tenderdash-proto/grpc feature.
server = ["grpc"]
# Build the gRPC client. Requires tenderdash-proto/grpc feature.
client = ["grpc"]
23 changes: 16 additions & 7 deletions proto-compiler/src/constants.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
//! Tenderdash protobuf implementation

use std::fmt::Display;

// Requirements
pub const DEP_PROTOC_VERSION: f32 = 25.0;

Expand All @@ -11,26 +13,33 @@ pub const TENDERDASH_REPO: &str = "https://github.com/dashpay/tenderdash";
pub enum GenerationMode {
/// Generate the files using `tonic` and put them into `tenderdash_grpc`
/// module.
Grpc,
GrpcServer,
/// Generate minimal gRPC client using tonic, without transport
/// implementation. Put them into `tenderdash_grpc_client` module.
GrpcClient,
/// Generate the files without `std` and put them into `tenderdash_nostd`
/// module.
NoStd,
}

impl GenerationMode {
pub fn module_name(&self) -> String {
match self {
GenerationMode::Grpc => "tenderdash_grpc".to_string(),
GenerationMode::GrpcServer => "tenderdash_grpc".to_string(),
GenerationMode::GrpcClient => "tenderdash_grpc_client".to_string(),
GenerationMode::NoStd => "tenderdash_nostd".to_string(),
}
}
}

impl ToString for GenerationMode {
fn to_string(&self) -> String {
match self {
GenerationMode::Grpc => "tonic".to_string(),
impl Display for GenerationMode {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let mode = match self {
GenerationMode::GrpcServer => "grpc-client-server".to_string(),
GenerationMode::GrpcClient => "grpc-client".to_string(),
GenerationMode::NoStd => "nostd".to_string(),
}
};
write!(f, "{}", mode)
}
}

Expand Down
19 changes: 17 additions & 2 deletions proto-compiler/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -124,14 +124,29 @@ pub fn proto_compile(mode: GenerationMode) {
println!("[info] => Creating structs.");

match mode {
GenerationMode::Grpc => {
GenerationMode::GrpcServer => {
#[cfg(feature = "grpc")]
tonic_build::configure()
.build_client(true)
.build_server(true)
.build_transport(true)
.generate_default_stubs(true)
.compile_protos_with_config(pb, &protos, &proto_includes_paths)
.unwrap();
#[cfg(not(feature = "grpc"))]
panic!("grpc feature is required to compile {}", mode.to_string());
panic!("grpc feature is required to compile {}", mode);
},
GenerationMode::GrpcClient => {
#[cfg(feature = "grpc")]
tonic_build::configure()
.build_client(true)
.build_server(false)
.build_transport(false)
.generate_default_stubs(true)
.compile_protos_with_config(pb, &protos, &proto_includes_paths)
.unwrap();
#[cfg(not(feature = "grpc"))]
panic!("grpc feature is required to compile {}", mode);
},
GenerationMode::NoStd => {
pb.compile_protos(&protos, &proto_includes_paths).unwrap();
Expand Down
22 changes: 14 additions & 8 deletions proto/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -34,16 +34,22 @@ all-features = true
#
# Sometimes cleaning the build cache with `cargo clean` might be necessary to solve
# issues related to outdated generated files.
default = ["grpc"]
default = ["server"]

# Enable standard library support; DEPRECATED - use `grpc` instead
std = ["grpc"]
# Build gRPC server using tonic
grpc = [
# Enable standard library support; required by `client` and `server`
std = ["client"]
# Enable gRPC support using tonic; DEPRECATED, consider using `server` or `client` instead
grpc = ["client"]
# Build gRPC server using tonic. Includes `client` feature.
server = ["client", "tonic/transport"]
# Build minimal gRPC client using tonic, without transport
client = [
"tenderdash-proto-compiler/grpc",
"prost/std",
"tenderdash-proto-compiler/server",
"tenderdash-proto-compiler/client",
"dep:tonic",
"tonic/codegen",
"tonic/prost",
"std",
]

serde = ["dep:serde", "bytes/serde"]
Expand All @@ -53,7 +59,7 @@ bytes = { version = "1.7", default-features = false }
prost = { version = "0.13", default-features = false, features = [
"prost-derive",
] }
tonic = { version = "0.12.3", optional = true }
tonic = { version = "0.12.3", optional = true, default-features = false }
serde = { version = "1.0.208", default-features = false, features = [
"derive",
], optional = true }
Expand Down
12 changes: 10 additions & 2 deletions proto/build.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,16 @@ fn main() {
env::set_var("TENDERDASH_COMMITISH", DEFAULT_VERSION);
}

#[cfg(feature = "grpc")]
tenderdash_proto_compiler::proto_compile(GenerationMode::Grpc);
// build gRPC server and client
// note it should be safe to build both server and client; we will just not use
// them in the lib.rs

#[cfg(feature = "server")]
// build gRPC server (includes client)
tenderdash_proto_compiler::proto_compile(GenerationMode::GrpcServer);
#[cfg(feature = "client")]
// build gRPC client only
tenderdash_proto_compiler::proto_compile(GenerationMode::GrpcClient);
// we always build nostd version
tenderdash_proto_compiler::proto_compile(GenerationMode::NoStd);

Expand Down
1 change: 1 addition & 0 deletions proto/src/.gitignore
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
tenderdash_nostd/
tenderdash_grpc/
tenderdash_grpc_client/

# prost/ and tenderdash.rs are deprecated and can be removed in the future
prost/
Expand Down
4 changes: 2 additions & 2 deletions proto/src/error.rs
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
//! This module defines the various errors that be raised during Protobuf
//! conversions.
#[cfg(not(feature = "grpc"))]
#[cfg(not(feature = "std"))]
use core::{convert::TryFrom, fmt::Display, num::TryFromIntError};
#[cfg(feature = "grpc")]
#[cfg(feature = "std")]
use std::{fmt::Display, num::TryFromIntError};

use flex_error::{define_error, DisplayOnly};
Expand Down
31 changes: 22 additions & 9 deletions proto/src/lib.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
//! tenderdash-proto library gives the developer access to the Tenderdash
//! proto-defined structs.

#![cfg_attr(not(feature = "grpc"), no_std)]
#![cfg_attr(not(feature = "std"), no_std)]
#![deny(warnings, trivial_casts, trivial_numeric_casts, unused_import_braces)]
#![allow(clippy::large_enum_variant)]
#![allow(clippy::doc_lazy_continuation)]
#![forbid(unsafe_code)]

extern crate alloc;
Expand All @@ -21,29 +21,42 @@ pub mod google {

mod error;

#[cfg(not(feature = "grpc"))]
#[cfg(not(feature = "std"))]
use core::{
convert::{TryFrom, TryInto},
fmt::Display,
};
#[cfg(feature = "grpc")]
#[cfg(feature = "std")]
use std::fmt::Display;

use bytes::{Buf, BufMut};
pub use error::Error;
pub use prost;
use prost::{encoding::encoded_len_varint, Message};

#[cfg(not(any(feature = "server", feature = "client")))]
#[rustfmt::skip]
#[allow(clippy::empty_docs)]
pub mod tenderdash_nostd;
#[cfg(not(feature = "grpc"))]
// Re-export the nostd module only if the std one is not available
pub use tenderdash_nostd::*;

#[cfg(feature = "grpc")]
#[cfg(feature = "server")]
#[rustfmt::skip]
#[allow(clippy::empty_docs)]
pub mod tenderdash_grpc;
#[cfg(feature = "grpc")]
#[cfg(feature = "client")]
#[rustfmt::skip]
#[allow(clippy::empty_docs)]
pub mod tenderdash_grpc_client;

// Now, re-export correct module

#[cfg(feature = "server")]
pub use tenderdash_grpc::*;
#[cfg(all(not(feature = "server"), feature = "client"))]
pub use tenderdash_grpc_client::*;
#[cfg(all(not(feature = "server"), not(feature = "client")))]
pub use tenderdash_nostd::*;

#[cfg(feature = "serde")]
pub mod serializers;
mod time;
Expand Down
2 changes: 1 addition & 1 deletion proto/src/prelude.rs
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ pub use alloc::{
};
pub use core::prelude::v1::*;

#[cfg(feature = "grpc")]
#[cfg(feature = "std")]
pub use tonic;

pub use crate::time::{FromMillis, ToMillis};
4 changes: 2 additions & 2 deletions proto/src/protobuf.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,9 @@
// Prost does not seem to have a way yet to remove documentations defined in
// protobuf files. These structs are defined in gogoproto v1.3.1 at https://github.com/gogo/protobuf/tree/v1.3.1/protobuf/google/protobuf

#[cfg(not(feature = "grpc"))]
#[cfg(not(feature = "std"))]
use core::fmt;
#[cfg(feature = "grpc")]
#[cfg(feature = "std")]
use std::fmt;

/// A Timestamp represents a point in time independent of any time zone or local
Expand Down
4 changes: 2 additions & 2 deletions proto/src/serializers/from_str.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
//! Serialize and deserialize any `T` that implements [[core::str::FromStr]]
//! and [[core::fmt::Display]] from or into string. Note this can be used for
//! all primitive data types.
#[cfg(not(feature = "grpc"))]
#[cfg(not(feature = "std"))]
use core::{fmt::Display, str::FromStr};
#[cfg(feature = "grpc")]
#[cfg(feature = "std")]
use std::{fmt::Display, str::FromStr};

use serde::{de::Error as _, Deserialize, Deserializer, Serialize, Serializer};
Expand Down
4 changes: 2 additions & 2 deletions proto/src/serializers/part_set_header_total.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,9 @@
//! Tendermint Core v0.34.0. This deserializer allows backwards-compatibility by
//! deserializing both ways. See also: <https://github.com/informalsystems/tendermint-rs/issues/679>

#[cfg(not(feature = "grpc"))]
#[cfg(not(feature = "std"))]
use core::{convert::TryFrom, fmt::Formatter};
#[cfg(feature = "grpc")]
#[cfg(feature = "std")]
use std::fmt::Formatter;

use serde::{
Expand Down
4 changes: 2 additions & 2 deletions proto/src/serializers/time_duration.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
//! Serialize/deserialize core::time::Duration type from and into string:
#[cfg(not(feature = "grpc"))]
#[cfg(not(feature = "std"))]
use core::time::Duration;
#[cfg(feature = "grpc")]
#[cfg(feature = "std")]
use std::time::Duration;

use serde::{de::Error as _, Deserialize, Deserializer, Serialize, Serializer};
Expand Down
4 changes: 2 additions & 2 deletions proto/src/serializers/timestamp.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
//! Serialize/deserialize Timestamp type from and into string:
#[cfg(not(feature = "grpc"))]
#[cfg(not(feature = "std"))]
use core::fmt::{self, Debug};
#[cfg(feature = "grpc")]
#[cfg(feature = "std")]
use std::fmt::{self, Debug};

use serde::{de::Error as _, ser::Error, Deserialize, Deserializer, Serialize, Serializer};
Expand Down
5 changes: 4 additions & 1 deletion proto/src/time.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,12 @@
//! Time conversion traits and functions

// Most likely we build nostd
#[cfg(not(any(feature = "server", feature = "client")))]
use crate::format;
use crate::{google::protobuf::Timestamp, Error};
pub trait ToMillis {
/// Convert protobuf timestamp into milliseconds since epoch

///
/// Note there is a resolution difference, as timestamp uses nanoseconds
///
/// # Arguments
Expand Down
Loading
Loading