22 个版本 (13 个破坏性更新)
0.26.0 | 2023年10月11日 |
---|---|
0.24.2 | 2023年8月26日 |
0.22.0 | 2023年6月25日 |
0.18.0 | 2023年3月30日 |
0.11.0 | 2022年7月6日 |
在 网络编程 中排名 1269
每月下载量 140
用于 2 个包
1MB
24K SLoC
什么是Netidx
Netidx是一种中间件,它允许在一个程序中发布一个值,如42,并在另一个程序中消费它,无论是在同一台机器上还是在网络上。
值在层次命名空间中具有全局唯一的名称。例如,我们发布的42可能被命名为/the-ultimate-answer(通常我们不会在根目录下直接放置值,但在这个例子中是合适的)。网络上的任何其他程序都可以通过该名称引用42,并在/the-ultimate-answer更改的(不太可能)事件中接收更新。
与其他系统的比较
-
类似于LDAP
- Netidx跟踪一个值的层次目录
- Netidx在某种程度上可浏览和可查询
- Netidx支持身份验证、授权和加密
- Netidx的值既可以写入也可以读取。
- 可以通过在较小的系统之间添加引用来构建较大的Netidx系统。解析服务器集群可以有父级和子级。
-
不同于LDAP
- 在Netidx中,解析服务器(如slapd)只保存具有数据的发布者的位置,而不是数据本身。
- 没有'条目'、'属性'、'ldif记录'等。系统中的每个名称要么是结构性的,要么是一个单独的值。使用层次结构创建类似条目的结构。因此,也没有模式检查。
- 可以订阅一个值,并且如果它更改,将会立即通知。
- 没有全局数据过滤器,例如,您不能查询(&(cn=bob)(uid=foo)),因为Netidx不是数据库。是否存在和什么查询机制存在取决于发布者。但是,您可以查询结构,例如,/foo/**/bar将返回任何以bar结束的foo下的路径。
-
类似于MQTT
- Netidx的值是发布/订阅
- 单个Netidx值可能有多名订阅者
- 当订阅的值发生变化时,所有Netidx订阅者都会收到更新。
- Netidx消息传递是可靠且有序的。
-
与MQTT不同
- 在Netidx中,没有集中的消息代理。消息直接通过TCP从发布者流向订阅者。解析器服务器仅存储发布者/发布值的地址。
有关更多详细信息,请参阅netidx书籍
以下是一个示例服务,它发布CPU温度,以及相应的消费者订阅的数据。
发布者
use netidx::{
publisher::{Publisher, Value, BindCfg},
config::Config,
resolver::Auth,
path::Path,
};
use tokio::time;
use std::time::Duration;
use anyhow::Result;
fn get_cpu_temp() -> f32 { 42. }
async fn run() -> Result<()> {
// load the site cluster config. You can also just use a file.
let cfg = Config::load_default()?;
// no authentication (kerberos v5 is the other option)
// listen on any unique address matching 192.168.0.0/16
let publisher = Publisher::new(cfg, Auth::Anonymous, "192.168.0.0/16".parse()?).await?;
let temp = publisher.publish(
Path::from("/hw/washu-chan/cpu-temp"),
Value::F32(get_cpu_temp())
)?;
loop {
time::sleep(Duration::from_millis(500)).await;
let mut batch = publisher.start_batch();
temp.update(&mut batch, Value::F32(get_cpu_temp()));
batch.commit(None).await;
}
Ok(())
}
订阅者
use netidx::{
subscriber::{Subscriber, UpdatesFlags},
config::Config,
resolver::Auth,
path::Path,
};
use futures::{prelude::*, channel::mpsc};
use anyhow::Result;
async fn run() -> Result<()> {
let cfg = Config::load_default()?;
let subscriber = Subscriber::new(cfg, Auth::Anonymous)?;
let path = Path::from("/hw/washu-chan/cpu-temp");
let temp = subscriber.subscribe_one(path, None).await?;
println!("washu-chan cpu temp is: {:?}", temp.last());
let (tx, mut rx) = mpsc::channel(10);
temp.updates(UpdatesFlags::empty(), tx);
while let Some(mut batch) = rx.next().await {
for (_, v) in batch.drain(..) {
println!("washu-chan cpu temp is: {:?}", v);
}
}
Ok(())
}
发布的内容始终有一个值,新订阅者最初接收该值。之后,订阅是一个无损耗的有序流,就像Tcp连接一样,只是传输单位是publisher::Value
,而不是字节。由于订阅者可以将值写回发布者,因此连接是双向的,就像Tcp流一样。
值包括许多有用的原语,包括零拷贝字节数据缓冲区(使用出色的bytes crate),因此您可以轻松使用netidx高效地发送任何您喜欢的消息。然而,建议坚持使用原语,并通过层次结构中的多个发布值来表达结构,因为这使您的系统更容易被发现,并且也相当高效。
netidx包括对kerberos v5(包括活动目录)的可选支持。如果启用,所有组件将在解析器、订阅者和发布者之间进行相互身份验证,并在线路上加密所有数据。
在krb5模式下,解析器服务器维护并强制执行整个命名空间的一组授权权限。系统管理员可以集中执行谁可以在哪里发布,以及谁可以订阅什么。
依赖项
~28–60MB
~1M SLoC