1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
use std::fmt;
use std::sync::Arc;
use agner_actors::ActorID;
use arc_swap::ArcSwapWeak;
use tokio::sync::{oneshot, Mutex};
#[derive(Debug, Clone, Default)]
pub struct Service(Arc<ServiceInner>);
#[derive(Debug)]
pub struct Registered(Arc<ActorID>);
#[derive(Debug, Default)]
struct ServiceInner {
label: Option<Box<str>>,
actor_id: ArcSwapWeak<ActorID>,
waiting: Mutex<Vec<oneshot::Sender<ActorID>>>,
}
impl Service {
pub fn new() -> Self {
Default::default()
}
#[deprecated(since = "0.3.5")]
pub fn new_with_label(_label: impl Into<Box<str>>) -> Self {
Self::new()
}
pub async fn register(&self, actor_id: ActorID) -> Registered {
let arc = Arc::new(actor_id);
let weak = Arc::downgrade(&arc);
let registered = Registered(arc);
self.0.actor_id.swap(weak);
self.0.waiting.lock().await.drain(..).for_each(|tx| {
let _ = tx.send(actor_id);
});
registered
}
pub fn resolve(&self) -> Option<ActorID> {
self.0.actor_id.load_full().upgrade().map(|arc| *arc)
}
pub async fn wait(&self) -> Option<ActorID> {
if let Some(registered) = self.resolve() {
return Some(registered)
}
let (tx, rx) = oneshot::channel();
let mut waiting = self.0.waiting.lock().await;
let maybe_replace = waiting.iter_mut().enumerate().find(|(_idx, tx)| tx.is_closed());
if let Some((_idx, to_replace)) = maybe_replace {
*to_replace = tx;
} else {
waiting.push(tx);
}
std::mem::drop(waiting);
if let Some(registered) = self.resolve() {
Some(registered)
} else {
rx.await.ok()
}
}
}
impl fmt::Display for Service {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let mut s = f.debug_struct("Service");
if let Some(id) = self.resolve() {
s.field("id", &id);
}
if let Some(label) = self.0.label.as_ref() {
s.field("label", label);
}
s.finish()
}
}
impl fmt::Display for Registered {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "Registered[{}]", self.0.as_ref())
}
}
#[tokio::test]
async fn happy_case() {
let id_1: ActorID = "1.0.0".parse().unwrap();
let id_2: ActorID = "1.1.1".parse().unwrap();
let service = Service::new();
assert!(service.resolve().is_none());
{
let _registered = service.register(id_1).await;
assert_eq!(service.resolve(), Some(id_1));
}
assert!(service.resolve().is_none());
{
let _registered = service.register(id_2).await;
assert_eq!(service.resolve(), Some(id_2));
}
assert!(service.resolve().is_none());
}