Crate plug

Crate plug 

Source
Expand description

A library for type-safe interprocess communication.

§Usage

Define the input and output of a wired connection:

use serde::{Deserialize, Serialize};

#[derive(Serialize, Deserialize)]
pub struct Credentials {
    username: String,
    password: String,
}

#[derive(Serialize, Deserialize)]
pub enum Authentication {
    Success { token: String },
    Error { description: String },
}

Create a Plug for this specific action:

use plug::Plug;

pub const LOG_IN: Plug<Credentials, Authentication> = Plug::new("log_in");

Connect the Plug to the server leveraging type-safety:

use std::io;

async fn log_in(credentials: Credentials) -> io::Result<Authentication> {
    let mut connection = LOG_IN.connect("127.0.0.1:1234").await?;

    connection.write(credentials).await?;
    connection.read().await
}

Implement the server by gathering all the Plug definitions in a Strip:

use plug::{Connection, Strip};

use std::io;
use tokio::net;
use tokio::task;

async fn run_server() -> io::Result<()> {
    let strip = Strip::new().plug(LOG_IN, log_in);
    let server = net::TcpListener::bind("127.0.0.1:1234").await?;

    loop {
        let Ok((client, _address)) = server.accept().await else {
            continue;
        };

        let _ = strip.attach(client).await;
    }
}

async fn log_in(mut connection: Connection<Authentication, Credentials>) -> io::Result<()> {
    let credentials = connection.read().await?;

    let result = if credentials.username == "admin" && credentials.password == "1234" {
        Authentication::Success { token: "verysecure".to_owned() }
    } else {
        Authentication::Error { description: "Invalid credentials!".to_owned() }
    };

    connection.write(result).await
}

Structs§

Connection
A type-safe TCP socket connection.
Plug
A type-safe connection endpoint with a name.
Strip
A set of plugs that can be used to accept incoming connections.

Enums§

Never
The error type for errors that can never happen.