如果你有一个经常变化的公网IP,你可能会需要用DDNS将这个IP绑定在同一个域名上,这样就可以不必每次输入一串IP地址了。我用的DNS服务商Cloudflare并没有提供名为DDNS的服务,但是其API可以实现类似的效果。

之前我用的是这里的方法,不过最近cloudflare更新了第四代API,那个方法已经失效。于是水博文的机会来了。

原料:

  1. 一个域名,并用Cloudflare管理
  2. 一台经常开着的,有计划任务或者cron,可以运行curl的电脑
  3. (可选)如果API地址被屏蔽的话,可能需要可用的代理

步骤:

  • https://dash.cloudflare.com/找到你域名的zone ID并新建一个~~API_KEY~~用于单个域名的API tokenAPI_KEY因为权限过高,不建议使用。
  • 新建一条子域名的A记录,IP写什么都行。
  • 查询这条记录的RECORD_ID,在终端运行:
curl -X GET "https://api.cloudflare.com/client/v4/zones/<ZONE_ID>/dns_records" \
     -H "Authorization: Bearer <API token>" \
     -H "Content-Type: application/json" \
     -x http://localhost:8087

其中 -x是本地代理地址,可选。大概能从返回的一坨json中找到id那一项。

  • 保存下面的脚本,修改<ZONE_ID><RECORD_ID>邮箱地址API_KEY子域名什么的。如果不需要代理的话删除-x那一行。

IPv4的情况

#!/bin/sh
NEW_IP=$(curl http://members.3322.org/dyndns/getip)
CURRENT_IP=$(cat /var/tmp/current_ip.txt)
if [ "$NEW_IP" = "$CURRENT_IP" ]
then
     echo "No Change in IP Adddress"
else
     OUTPUT=$(curl -X PUT "https://api.cloudflare.com/client/v4/zones/<ZONE_ID>/dns_records/<RECORD_ID>" \
     -x http://<代理地址,不要的话删去这一行>:8087 \
     -H "Authorization: Bearer <API token>" \
     -H "Content-Type: application/json" \
     --data '{"type":"A","name":"<子域名地址>","content":"'$NEW_IP'","ttl":1}')
     if echo "$OUTPUT" | grep -q "$NEW_IP"; then
          echo "Update IP Success!"
          echo $NEW_IP > /var/tmp/current_ip.txt
     else
          echo "Update Error!"
     fi
fi

IPv6的情况,放在两个不同的脚本里面可以共存

#!/bin/sh
NEW_IP=$(ip -6 addr | grep inet6 | awk -F '[ \t]+|/' '{print $3}' | grep -v ^::1 | grep -v ^fe80 | head -n 1)
CURRENT_IP=$(cat /var/tmp/current_ip_6.txt)
if [ "$NEW_IP" = "$CURRENT_IP" ]
then
     echo "No Change in IP Adddress!"
else
     OUTPUT=$(curl -X PUT "https://api.cloudflare.com/client/v4/zones/<ZONE_ID>/dns_records/<RECORD_ID>" \
     -x socks5h://<代理地址,不要的话删去这一行>:1085 \
     -H "Authorization: Bearer <API token>" \
     -H "Content-Type: application/json" \
     --data '{"type":"AAAA","name":"<子域名>.<域名>.xyz","content":"'$NEW_IP'","ttl":1}')
     if echo "$OUTPUT" | grep -q "$NEW_IP"; then
          echo "Update IP Success!"
          echo $NEW_IP > /var/tmp/current_ip_6.txt
     else
          echo "Update Error!"
     fi
fi
  • 接下来让它每两个小时运行一次就好了。
# crontab -e
*/120 * * * * /path/of/your/script.sh

写字好累(躺


来自19-10-25的更新:DNSpod的公网IP查询服务响应有些奇怪,所以换成了3322。

来自20-06-23:增加了判断,现在不会在一次网络问题后直接写入current_ip.txt了。