13 releases (2 stable)

1.0.1 Dec 8, 2021
1.0.0 Apr 26, 2021
1.0.0-rc.2 Feb 15, 2020
0.5.1 Dec 7, 2019
0.0.0 Jun 16, 2018

#399 in Network programming

Download history 617/week @ 2024-09-09 517/week @ 2024-09-16 604/week @ 2024-09-23 704/week @ 2024-09-30 945/week @ 2024-10-07 691/week @ 2024-10-14 1003/week @ 2024-10-21 719/week @ 2024-10-28 822/week @ 2024-11-04 441/week @ 2024-11-11 418/week @ 2024-11-18 532/week @ 2024-11-25 509/week @ 2024-12-02 602/week @ 2024-12-09 678/week @ 2024-12-16 394/week @ 2024-12-23

2,214 downloads per month
Used in 12 crates (10 directly)

MIT license

180KB
2.5K SLoC

A safe Rust wrapper for NNG

docs.rs crates.io MIT License Rustc 1.36+ Pipeline

What Is NNG

From the NNG Github Repository:

NNG, like its predecessors nanomsg (and to some extent ZeroMQ), is a lightweight, broker-less library, offering a simple API to solve common recurring messaging problems, such as publish/subscribe, RPC-style request/reply, or service discovery. The API frees the programmer from worrying about details like connection management, retries, and other common considerations, so that they can focus on the application instead of the plumbing.

Nng-rs

This crate provides a safe wrapper around the NNG library, seeking to maintain an API that is similar to the original library. As such, the majority of examples available online should be easy to apply to this crate.

Rust Version Requirements

The current version requires Rustc v1.36 or greater. In general, this crate should always be able to compile with the Rustc version available on the oldest currently-supported Ubuntu LTS release. Changes to the minimum required Rustc version will only be considered a breaking change if the newly required version is not available on the oldest currently-supported Ubuntu LTS release.

NOTE: This does not necessarily mean that this crate will build without installing packages on Ubuntu LTS, as NNG currently requires a version of CMake (v3.13) that is newer than the one available in the LTS repositories.

Features

  • build-nng (default): Build NNG from source and statically link to the library.
  • ffi-module: Expose the raw FFI bindings via the nng::ffi module. This is useful for utilizing NNG features that are implemented in the base library but not this wrapper. Note that this exposes some internal items of this library and it directly exposes the NNG library, so anything enabled by this can change without bumping versions.

Building NNG

Enabling the build-nng feature will cause the NNG library to be built using the default settings and CMake generator. Most of the time, this should just work. However, in the case that the default are not the desired settings, there are three ways to change the build:

  1. Patch the nng-sys dependency and enable the desired build features.
  2. Disable the build-nng feature and directly depend on nng-sys.
  3. Disable the build-nng feature and manually compile NNG.

The build features are not exposed in this crate because Cargo features are currently strictly additive and there is no way to specify mutually exclusive features (i.e., build settings). Additionally, it does not seem very ergonomic to have this crate expose all of the same build features as the binding crate, which could lead to feature pollution in any dependent crates.

Merge requests for a better solution to this are more than welcome.

Examples

The following example uses the intra-process transport to set up a request/reply socket pair. The "client" sends a string to the "server" which responds with a nice phrase.

use nng::*;

const ADDRESS: &'static str = "inproc://nng/example";

fn request() -> Result<()> {
    // Set up the client and connect to the specified address
    let client = Socket::new(Protocol::Req0)?;
    client.dial(ADDRESS)?;

    // Send the request from the client to the server. In general, it will be
    // better to directly use a `Message` to enable zero-copy, but that doesn't
    // matter here.
    client.send("Ferris".as_bytes())?;

    // Wait for the response from the server.
    let msg = client.recv()?;
    assert_eq!(&msg[..], b"Hello, Ferris");
    Ok(())
}

fn reply() -> Result<()> {
    // Set up the server and listen for connections on the specified address.
    let server = Socket::new(Protocol::Rep0)?;
    server.listen(ADDRESS)?;

    // Receive the message from the client.
    let mut msg = server.recv()?;
    assert_eq!(&msg[..], b"Ferris");

    // Reuse the message to be more efficient.
    msg.push_front(b"Hello, ");

    server.send(msg)?;
    Ok(())
}

Additional examples are in the examples directory.

Dependencies

~3MB
~87K SLoC