aboutsummaryrefslogtreecommitdiff
path: root/src/main.rs
blob: feb4d3e261d27ca63b982e8626bec45296a96394 (plain) (blame)
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
108
109
110
111
use rsdsl_dslite::{Error, Result};

use std::fs::File;
use std::net::{Ipv6Addr, SocketAddr};
use std::thread;
use std::time::Duration;

use ipnet::Ipv6Net;
use rsdsl_netlinkd::tunnel::IpIp6;
use rsdsl_pd_config::PdConfig;
use signal_hook::{consts::SIGUSR1, iterator::Signals};
use sysinfo::{ProcessExt, Signal, System, SystemExt};
use trust_dns_resolver::config::{NameServerConfig, Protocol, ResolverConfig, ResolverOpts};
use trust_dns_resolver::Resolver;

const MAX_ATTEMPTS: usize = 3;
const BACKOFF: u64 = 900;

fn main() -> Result<()> {
    println!("[info] init");

    let mut tnl = None;

    let mut signals = Signals::new([SIGUSR1])?;
    for _ in signals.forever() {
        logic(&mut tnl)?;
    }

    unreachable!()
}

fn logic(tnl: &mut Option<IpIp6>) -> Result<()> {
    *tnl = None; // Delete old tunnel.

    let mut file = File::open(rsdsl_pd_config::LOCATION)?;
    let pdconfig: PdConfig = serde_json::from_reader(&mut file)?;

    if let Some(ref aftr) = pdconfig.aftr {
        let local = local_address(&pdconfig)?;
        let remote = multitry_resolve6(&pdconfig, aftr)?;
        *tnl = Some(IpIp6::new(
            "dslite0".to_string(),
            "ppp0".to_string(),
            local,
            remote,
        )?);

        for netlinkd in System::default().processes_by_exact_name("/bin/rsdsl_netlinkd") {
            netlinkd.kill_with(Signal::User1);
        }

        println!("[info] init ds-lite tunnel {} <=> {}", local, remote);
    } else {
        println!("[info] no aftr");
    }

    Ok(())
}

fn local_address(pdconfig: &PdConfig) -> Result<Ipv6Addr> {
    let prefix = Ipv6Net::new(pdconfig.prefix, pdconfig.len)?.trunc();
    let mut subnets = prefix.subnets(64)?;

    let addr = next_ifid1(&mut subnets)?;
    Ok(addr)
}

fn next_ifid1<T: Iterator<Item = Ipv6Net>>(subnets: &mut T) -> Result<Ipv6Addr> {
    Ok((u128::from(subnets.next().ok_or(Error::NotEnoughIpv6Subnets)?.addr()) + 1).into())
}

fn resolve6(pdconfig: &PdConfig, fqdn: &str) -> Result<Ipv6Addr> {
    let mut cfg = ResolverConfig::new();

    cfg.add_name_server(NameServerConfig::new(
        SocketAddr::new(pdconfig.dns1.into(), 53),
        Protocol::Udp,
    ));
    cfg.add_name_server(NameServerConfig::new(
        SocketAddr::new(pdconfig.dns2.into(), 53),
        Protocol::Udp,
    ));

    let resolver = Resolver::new(cfg, ResolverOpts::default())?;
    let response = resolver.ipv6_lookup(fqdn)?;

    let addr = response.iter().next().ok_or(Error::NoDnsRecord)?;
    Ok(*addr)
}

fn multitry_resolve6(pdconfig: &PdConfig, fqdn: &str) -> Result<Ipv6Addr> {
    for i in 0..MAX_ATTEMPTS {
        match resolve6(pdconfig, fqdn) {
            Ok(v) => return Ok(v),
            Err(e) => {
                if i >= MAX_ATTEMPTS - 1 {
                    return Err(e);
                } else {
                    println!(
                        "[warn] resolve aftr {}: {} (attempt {}/{})",
                        fqdn, e, i, MAX_ATTEMPTS
                    )
                }
            }
        }

        thread::sleep(Duration::from_secs(BACKOFF));
    }

    unreachable!()
}