DN42探究日记 - Ep.2 通过OSPF搭建内部网络

DN42探究日记 - Ep.2 通过OSPF搭建内部网络

KaguraiYoRoy
2025-07-22 / 1 评论 / 57 阅读 / 正在检测是否收录...

写在前面

本人是BGP小白,文章中可能会存在不严谨内容/小白理解/低级错误,请诸位大佬们手下留情。若发现存在问题,您愿意的话可以邮件联系我,我会在第一时间更正。如果您不能接受,建议现在就关闭此文章

本文更新日志

为什么需要内部路由

当节点数量增多,我们需要一个合适的方式处理自己AS的内部路由。因为BGP路由只负责将数据包路由到AS,这就导致了一个问题:假如我有A、B两个节点都与别人peer,但是在路由器看来这两台设备同属于一个AS,从A节点发出的请求回包可能被回到B上。这个时候,如果没有做内部路由,则A会无法收到回包。因此,我们需要保证自家内网各个设备之间都能连通。常见的几种方式如下:

  1. 通过ZeroTier等组网工具:这种方式较为简单,只需要在每个节点上配置一个客户端即可实现各个节点之间的P2P连接。
  2. 通过WireGuard等P2P工具手动建立$\frac{n(n+1)}{2}$条隧道,效果同1,但是节点一多工作量呈指数级上升
  3. 通过WireGuard等P2P工具手动建立<$\frac{n(n+1)}{2}$条隧道,再通过OSPF、Babel等内网寻路协议建立内网路由,优点是较为灵活,易于后期添加新节点,缺点就是较为危险,容易配置错误引爆DN42

因此我决定冒险一下

网络结构

我的网络结构如下图,其中A-E分别代表5个不同的节点:

  graph LR
    A[A<br>172.20.234.225<br>fd18:3e15:61d0::1]
    B[B<br>172.20.234.226<br>fd18:3e15:61d0::2]
    C[C<br>172.20.234.227<br>fd18:3e15:61d0::3]
    D[D<br>172.20.234.228<br>fd18:3e15:61d0::4]
    E[E<br>172.20.234.229<br>fd18:3e15:61d0::5]
    
    E <--> A
    E <--> C
    A <--> D
    C <--> D
    A <--> C
    D <--> B

更新Bird2至v2.16及以上

因为我希望使用IPv6 Link-Local地址传递IPv4的OSPF数据,而Bird在2.16及以后才支持这项功能,因此需要更新至v2.16。如果你不希望更新,可以先跳过此步骤。
使用以下指令安装最新版本的Bird2:

sudo apt update && sudo apt -y install apt-transport-https ca-certificates wget lsb-release
sudo wget -O /usr/share/keyrings/cznic-labs-pkg.gpg https://pkg.labs.nic.cz/gpg

echo "deb [signed-by=/usr/share/keyrings/cznic-labs-pkg.gpg] https://pkg.labs.nic.cz/bird2 $(lsb_release -sc) main" | sudo tee /etc/apt/sources.list.d/cznic-labs-bird2.list

sudo apt update && sudo apt install bird2 -y

配置隧道

[Interface]
PrivateKey = <本地WireGuard私钥>
ListenPort = <监听端口>
Table = off
Address = <IPv6 LLA>/64
PostUp = sysctl -w net.ipv6.conf.%i.autoconf=0

[Peer]
PublicKey = <对端公钥>
Endpoint = <对端公网接入点>
AllowedIPs = 10.0.0.0/8, 172.20.0.0/14, 172.31.0.0/16, fd00::/8, fe00::/8, ff02::5
  • ff02::5是OSPFv3路由器专用的链路本地范围组播地址,需要添加进AllowedIPs。
  • 如果你正在使用v2.16以前的Bird,请再为隧道配置一个IPv4地址,不一定非要是DN42 IPv4,其他私有地址也可以。请参考:

启用OSPF

默认你已经配置好了如上一篇文章所写的bird基础配置
/etc/bird下新建一个名为ospf.conf的文件,填入如下内容:

protocol ospf v3 <name> {
    ipv4 {
        import where is_self_net() && source != RTS_BGP;
        export where is_self_net() && source != RTS_BGP;
    };

    include "/etc/bird/ospf/*";
};

protocol ospf v3 <name> {
    ipv6 {
        import where is_self_net_v6() && source != RTS_BGP;
        export where is_self_net_v6() && source != RTS_BGP;
    };

    include "/etc/bird/ospf/*";
};
  • 理论上来说应该使用OSPF v2处理IPv4,但是因为需要通过IPv6 LLA地址通信IPv4,因此此处IPv4也使用OSPF v3。
  • 过滤规则保证只有本网段内的路由能够通过OSPF传递,并且过滤掉外部BGP协议的路由
  • 千万不要随意使用import all;export all;,有可能会导致路由劫持并影响到整个DN42网络。OSPF只应该处理网段内部的路由。

接着,新建/etc/bird/ospf文件夹,在其中创建area配置文件(如:/etc/bird/ospf/0.conf),填写区域信息:

area 0.0.0.0 {
    interface "<DN42 dummy网卡>" { stub; };
    interface "<wg0网卡名称>" {
        cost 80; # 按照你的网络情况修改
        type ptp;
    };
    interface "<wg1网卡名称>" {
        cost 100; # 按照你的网络情况修改
        type ptp;
    };
    # 以此类推
};
  • 0.0.0.0区域代表骨干网
  • 此处dummy网卡指上一篇文章中所写的DN42虚拟网卡
  • cost值本应该是用于开销计算,但在DN42这种对带宽要求不大而对延迟较为敏感的场景下可以直接填写延迟,OSPF会自动走开销值之和最短的路由。

最后,打开/etc/bird/bird.conf,在末尾引入OSPF的配置文件:

include "ospf.conf";

运行birdc configure,然后birdc show protocols应该就能看到OSPF的状态是Running了。如果不是,请检查配置步骤是否出错。

2.png

此时,在非直连的两台机器上互相ping应该就能通了: 3.png


参考文章:

  1. https://soha.moe/post/bird-bgp-kickstart.html
  2. https://milu.ink/519.html
  3. https://blog.xeiu.top/archives/344
  4. https://baimeow.cn/posts/dn11/configureospf/
  5. https://lantian.pub/article/modify-website/dn42-experimental-network-2020.lantian/
  6. https://lantian.pub/article/modify-website/how-to-kill-the-dn42-network.lantian/
  7. https://edu.51cto.com/article/note/3019.html
  8. https://bird.nic.cz/news/2024-12-07-new-release-2.16/
  9. https://bird.xmsl.dev/docs/user-guide/1-2-installing.html
0

评论 (1)

取消
  1. 头像
    @

    [...]DN42探究日记 – Ep.2 通过OSPF搭建内部网络 – 悠笙の开发日记[...]

    回复