mirror of
https://github.com/fluencelabs/rust-libp2p
synced 2025-06-29 09:41:34 +00:00
Clean up directory structure (#426)
* Remove unused circular-buffer crate * Move transports into subdirectory * Move misc into subdirectory * Move stores into subdirectory * Move multiplexers * Move protocols * Move libp2p top layer * Fix Test: skip doctest if secio isn't enabled
This commit is contained in:
committed by
GitHub
parent
f5ce93c730
commit
2ea49718f3
@ -1,280 +0,0 @@
|
||||
// Copyright 2018 Parity Technologies (UK) Ltd.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining a
|
||||
// copy of this software and associated documentation files (the "Software"),
|
||||
// to deal in the Software without restriction, including without limitation
|
||||
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
// and/or sell copies of the Software, and to permit persons to whom the
|
||||
// Software is furnished to do so, subject to the following conditions:
|
||||
//
|
||||
// The above copyright notice and this permission notice shall be included in
|
||||
// all copies or substantial portions of the Software.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
||||
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
// DEALINGS IN THE SOFTWARE.
|
||||
|
||||
//! Libp2p is a peer-to-peer framework.
|
||||
//!
|
||||
//! # Major libp2p concepts
|
||||
//!
|
||||
//! Here is a list of all the major concepts of libp2p.
|
||||
//!
|
||||
//! ## Multiaddr
|
||||
//!
|
||||
//! A `Multiaddr` is a way to reach a node. Examples:
|
||||
//!
|
||||
//! * `/ip4/80.123.90.4/tcp/5432`
|
||||
//! * `/ip6/[::1]/udp/10560`
|
||||
//! * `/unix//path/to/socket`
|
||||
//!
|
||||
//! ## Transport
|
||||
//!
|
||||
//! `Transport` is a trait that represents an object capable of dialing multiaddresses or
|
||||
//! listening on multiaddresses. The `Transport` produces an output which varies depending on the
|
||||
//! object that implements the trait.
|
||||
//!
|
||||
//! Each implementation of `Transport` typically supports only some multiaddresses. For example
|
||||
//! the `TcpConfig` type (which implements `Transport`) only supports multiaddresses of the format
|
||||
//! `/ip4/.../tcp/...`.
|
||||
//!
|
||||
//! Example:
|
||||
//!
|
||||
//! ```rust
|
||||
//! use libp2p::{Multiaddr, Transport, tcp::TcpConfig};
|
||||
//! let tcp_transport = TcpConfig::new();
|
||||
//! let addr: Multiaddr = "/ip4/98.97.96.95/tcp/20500".parse().expect("invalid multiaddr");
|
||||
//! let _outgoing_connec = tcp_transport.dial(addr);
|
||||
//! // Note that `_outgoing_connec` is a `Future`, and therefore doesn't do anything by itself
|
||||
//! // unless it is run through a tokio runtime.
|
||||
//! ```
|
||||
//!
|
||||
//! The easiest way to create a transport is to use the `CommonTransport` struct. This struct
|
||||
//! provides support for the most common protocols.
|
||||
//!
|
||||
//! Example:
|
||||
//!
|
||||
//! ```rust
|
||||
//! use libp2p::CommonTransport;
|
||||
//! let _transport = CommonTransport::new();
|
||||
//! // _transport.dial(...);
|
||||
//! ```
|
||||
//!
|
||||
//! See the documentation of the `libp2p-core` crate for more details about transports.
|
||||
//!
|
||||
//! # Connection upgrades
|
||||
//!
|
||||
//! Once a connection has been opened with a remote through a `Transport`, it can be *upgraded*.
|
||||
//! This consists in negotiating a protocol with the remote (through the `multistream-select`
|
||||
//! protocol), and applying that protocol on the socket.
|
||||
//!
|
||||
//! Example upgrades:
|
||||
//!
|
||||
//! - Adding a security layer on top of the connection.
|
||||
//! - Applying multiplexing, so that a connection can be split into multiple substreams.
|
||||
//! - Negotiating a specific protocol, such as *ping* or *kademlia*.
|
||||
//!
|
||||
//! A potential connection upgrade is represented with the `ConnectionUpgrade` trait. The trait
|
||||
//! consists in a protocol name plus a method that turns the socket into an `Output` object whose
|
||||
//! nature and type is specific to each upgrade. For example, if you upgrade a connection with a
|
||||
//! security layer, the output might contain an encrypted stream and the public key of the remote.
|
||||
//!
|
||||
//! You can combine a `Transport` with a compatible `ConnectionUpgrade` in order to obtain another
|
||||
//! `Transport` that yields the output of the upgrade.
|
||||
//!
|
||||
//! Example:
|
||||
//!
|
||||
//! ```rust
|
||||
//! use libp2p::{Transport, tcp::TcpConfig, secio::{SecioConfig, SecioKeyPair}};
|
||||
//! let tcp_transport = TcpConfig::new();
|
||||
//! let secio_upgrade = SecioConfig {
|
||||
//! key: SecioKeyPair::ed25519_generated().unwrap(),
|
||||
//! };
|
||||
//! let with_security = tcp_transport.with_upgrade(secio_upgrade);
|
||||
//! // let _ = with_security.dial(...);
|
||||
//! // `with_security` also implements the `Transport` trait, and all the connections opened
|
||||
//! // through it will automatically negotiate the `secio` protocol.
|
||||
//! ```
|
||||
//!
|
||||
//! See the documentation of the `libp2p-core` crate for more details about upgrades.
|
||||
//!
|
||||
//! ## Swarm
|
||||
//!
|
||||
//! Once you have created an object that implements the `Transport` trait, you can put it in a
|
||||
//! *swarm*. This is done by calling the `swarm()` freestanding function with the transport
|
||||
//! alongside with a function or a closure that will turn the output of the upgrade (usually an
|
||||
//! actual protocol, as explained above) into a `Future` producing `()`.
|
||||
//!
|
||||
//! See the documentation of the `libp2p-core` crate for more details about creating a swarm.
|
||||
//!
|
||||
//! # Using libp2p
|
||||
//!
|
||||
//! This section contains details about how to use libp2p in practice.
|
||||
//!
|
||||
//! The most simple way to use libp2p consists in the following steps:
|
||||
//!
|
||||
//! - Create a *base* implementation of `Transport` that combines all the protocols you want and
|
||||
//! the upgrades you want, such as the security layer and multiplexing.
|
||||
//! - Create structs that implement the `ConnectionUpgrade` trait for the protocols you want to
|
||||
//! create, or use the protocols provided by the `libp2p` crate.
|
||||
//! - Create a swarm that combines your base transport and all the upgrades and that handles the
|
||||
//! behaviour that happens.
|
||||
//! - Use this swarm to dial and listen.
|
||||
//!
|
||||
//! You probably also want to have some sort of nodes discovery mechanism, so that you
|
||||
//! automatically connect to nodes of the network. The details of this haven't been fleshed out
|
||||
//! in libp2p and will be written later.
|
||||
//!
|
||||
|
||||
pub extern crate bytes;
|
||||
pub extern crate futures;
|
||||
#[cfg(not(target_os = "emscripten"))]
|
||||
pub extern crate tokio_current_thread;
|
||||
pub extern crate multiaddr;
|
||||
pub extern crate tokio_io;
|
||||
pub extern crate tokio_codec;
|
||||
|
||||
pub extern crate libp2p_core as core;
|
||||
#[cfg(not(target_os = "emscripten"))]
|
||||
pub extern crate libp2p_dns as dns;
|
||||
pub extern crate libp2p_identify as identify;
|
||||
pub extern crate libp2p_kad as kad;
|
||||
pub extern crate libp2p_floodsub as floodsub;
|
||||
pub extern crate libp2p_mplex as mplex;
|
||||
pub extern crate libp2p_peerstore as peerstore;
|
||||
pub extern crate libp2p_ping as ping;
|
||||
pub extern crate libp2p_ratelimit as ratelimit;
|
||||
pub extern crate libp2p_relay as relay;
|
||||
#[cfg(all(not(target_os = "emscripten"), feature = "libp2p-secio"))]
|
||||
pub extern crate libp2p_secio as secio;
|
||||
#[cfg(not(target_os = "emscripten"))]
|
||||
pub extern crate libp2p_tcp_transport as tcp;
|
||||
pub extern crate libp2p_transport_timeout as transport_timeout;
|
||||
pub extern crate libp2p_uds as uds;
|
||||
pub extern crate libp2p_websocket as websocket;
|
||||
pub extern crate libp2p_yamux as yamux;
|
||||
|
||||
pub mod simple;
|
||||
|
||||
pub use self::core::{Transport, ConnectionUpgrade, PeerId, swarm};
|
||||
pub use self::multiaddr::Multiaddr;
|
||||
pub use self::simple::SimpleProtocol;
|
||||
pub use self::transport_timeout::TransportTimeout;
|
||||
|
||||
/// Implementation of `Transport` that supports the most common protocols.
|
||||
///
|
||||
/// The list currently is TCP/IP, DNS, and WebSockets. However this list could change in the
|
||||
/// future to get new transports.
|
||||
#[derive(Debug, Clone)]
|
||||
pub struct CommonTransport {
|
||||
// The actual implementation of everything.
|
||||
inner: CommonTransportInner
|
||||
}
|
||||
|
||||
#[cfg(not(target_os = "emscripten"))]
|
||||
pub type InnerImplementation = core::transport::OrTransport<dns::DnsConfig<tcp::TcpConfig>, websocket::WsConfig<dns::DnsConfig<tcp::TcpConfig>>>;
|
||||
#[cfg(target_os = "emscripten")]
|
||||
pub type InnerImplementation = websocket::BrowserWsConfig;
|
||||
|
||||
#[derive(Debug, Clone)]
|
||||
struct CommonTransportInner {
|
||||
inner: InnerImplementation,
|
||||
}
|
||||
|
||||
impl CommonTransport {
|
||||
/// Initializes the `CommonTransport`.
|
||||
#[inline]
|
||||
#[cfg(not(target_os = "emscripten"))]
|
||||
pub fn new() -> CommonTransport {
|
||||
let tcp = tcp::TcpConfig::new();
|
||||
let with_dns = dns::DnsConfig::new(tcp);
|
||||
let with_ws = websocket::WsConfig::new(with_dns.clone());
|
||||
let inner = with_dns.or_transport(with_ws);
|
||||
|
||||
CommonTransport {
|
||||
inner: CommonTransportInner { inner }
|
||||
}
|
||||
}
|
||||
|
||||
/// Initializes the `CommonTransport`.
|
||||
#[inline]
|
||||
#[cfg(target_os = "emscripten")]
|
||||
pub fn new() -> CommonTransport {
|
||||
let inner = websocket::BrowserWsConfig::new();
|
||||
CommonTransport {
|
||||
inner: CommonTransportInner { inner: inner }
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Transport for CommonTransport {
|
||||
type Output = <InnerImplementation as Transport>::Output;
|
||||
type MultiaddrFuture = <InnerImplementation as Transport>::MultiaddrFuture;
|
||||
type Listener = <InnerImplementation as Transport>::Listener;
|
||||
type ListenerUpgrade = <InnerImplementation as Transport>::ListenerUpgrade;
|
||||
type Dial = <InnerImplementation as Transport>::Dial;
|
||||
|
||||
#[inline]
|
||||
fn listen_on(self, addr: Multiaddr) -> Result<(Self::Listener, Multiaddr), (Self, Multiaddr)> {
|
||||
match self.inner.inner.listen_on(addr) {
|
||||
Ok(res) => Ok(res),
|
||||
Err((inner, addr)) => {
|
||||
let trans = CommonTransport { inner: CommonTransportInner { inner: inner } };
|
||||
Err((trans, addr))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn dial(self, addr: Multiaddr) -> Result<Self::Dial, (Self, Multiaddr)> {
|
||||
match self.inner.inner.dial(addr) {
|
||||
Ok(res) => Ok(res),
|
||||
Err((inner, addr)) => {
|
||||
let trans = CommonTransport { inner: CommonTransportInner { inner: inner } };
|
||||
Err((trans, addr))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn nat_traversal(&self, server: &Multiaddr, observed: &Multiaddr) -> Option<Multiaddr> {
|
||||
self.inner.inner.nat_traversal(server, observed)
|
||||
}
|
||||
}
|
||||
|
||||
/// The `multiaddr!` macro is an easy way for a user to create a `Multiaddr`.
|
||||
///
|
||||
/// Example:
|
||||
///
|
||||
/// ```rust
|
||||
/// # #[macro_use]
|
||||
/// # extern crate libp2p;
|
||||
/// # fn main() {
|
||||
/// let _addr = multiaddr![IP4([127, 0, 0, 1]), TCP(10500u16)];
|
||||
/// # }
|
||||
/// ```
|
||||
///
|
||||
/// Each element passed to `multiaddr![]` should be a variant of the `AddrComponent` enum. The
|
||||
/// optional parameter is casted into the proper type with the `Into` trait.
|
||||
///
|
||||
/// For example, `IP4([127, 0, 0, 1])` works because `Ipv4Addr` implements `From<[u8; 4]>`.
|
||||
#[macro_export]
|
||||
macro_rules! multiaddr {
|
||||
($($comp:ident $(($param:expr))*),+) => {
|
||||
{
|
||||
use std::iter;
|
||||
let elem = iter::empty::<$crate::multiaddr::AddrComponent>();
|
||||
$(
|
||||
let elem = {
|
||||
let cmp = $crate::multiaddr::AddrComponent::$comp $(( $param.into() ))*;
|
||||
elem.chain(iter::once(cmp))
|
||||
};
|
||||
)+
|
||||
elem.collect::<$crate::Multiaddr>()
|
||||
}
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user