forked from rs-ipfs/rust-ipfs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdns.rs
106 lines (93 loc) · 3.37 KB
/
dns.rs
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
use crate::error::Error;
use crate::path::IpfsPath;
use bytes::Bytes;
use domain::base::iana::Rtype;
use domain::base::{Dname, Question};
use domain::rdata::rfc1035::Txt;
use domain_resolv::{stub::Answer, StubResolver};
use futures::future::{select_ok, SelectOk};
use futures::pin_mut;
use std::future::Future;
use std::pin::Pin;
use std::str::FromStr;
use std::task::{Context, Poll};
use std::{fmt, io};
#[derive(Debug)]
pub struct DnsLinkError(String);
impl fmt::Display for DnsLinkError {
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(fmt, "DNS error: {}", self.0)
}
}
impl std::error::Error for DnsLinkError {}
type FutureAnswer = Pin<Box<dyn Future<Output = Result<Answer, io::Error>> + Send>>;
pub struct DnsLinkFuture {
query: SelectOk<FutureAnswer>,
}
impl Future for DnsLinkFuture {
type Output = Result<IpfsPath, Error>;
fn poll(self: Pin<&mut Self>, context: &mut Context) -> Poll<Self::Output> {
let _self = self.get_mut();
loop {
let query = &mut _self.query;
pin_mut!(query);
match query.poll(context) {
Poll::Ready(Ok((answer, rest))) => {
for record in answer.answer()?.limit_to::<Txt<_>>() {
let txt = record?;
let bytes: &[u8] = txt.data().as_flat_slice().unwrap_or(b"");
let string = String::from_utf8_lossy(&bytes).to_string();
if string.starts_with("dnslink=") {
let path = IpfsPath::from_str(&string[8..])?;
return Poll::Ready(Ok(path));
}
}
if !rest.is_empty() {
_self.query = select_ok(rest);
} else {
return Poll::Ready(Err(
DnsLinkError("no DNS records found".to_owned()).into()
));
}
}
Poll::Pending => return Poll::Pending,
Poll::Ready(Err(e)) => return Poll::Ready(Err(DnsLinkError(e.to_string()).into())),
}
}
}
}
pub async fn resolve(domain: &str) -> Result<IpfsPath, Error> {
let mut dnslink = "_dnslink.".to_string();
dnslink.push_str(domain);
let qname = Dname::<Bytes>::from_str(&domain)?;
let question = Question::new_in(qname, Rtype::Txt);
let resolver = StubResolver::new();
let query1 = Box::pin(async move { resolver.query(question).await });
let qname = Dname::<Bytes>::from_str(&dnslink)?;
let question = Question::new_in(qname, Rtype::Txt);
let resolver = StubResolver::new();
let query2 = Box::pin(async move { resolver.query(question).await });
Ok(DnsLinkFuture {
query: select_ok(vec![query1 as FutureAnswer, query2]),
}
.await?)
}
#[cfg(test)]
mod tests {
use super::*;
#[tokio::test(max_threads = 1)]
#[ignore]
async fn test_resolve1() {
let res = resolve("ipfs.io").await.unwrap().to_string();
assert_eq!(res, "/ipns/website.ipfs.io");
}
#[tokio::test(max_threads = 1)]
#[ignore]
async fn test_resolve2() {
let res = resolve("website.ipfs.io").await.unwrap().to_string();
assert_eq!(
res,
"/ipfs/bafybeiayvrj27f65vbecspbnuavehcb3znvnt2strop2rfbczupudoizya"
);
}
}