DDNS for cloudflare
Code
py
import time
import socket
import requests
import subprocess
from pathlib import Path
import logging
import logging.handlers
def setup_logger(log_file, max_bytes, backup_count):
logger = logging.getLogger('my_logger')
logger.setLevel(logging.INFO)
handler = logging.handlers.RotatingFileHandler(
log_file, maxBytes=max_bytes, backupCount=backup_count)
formatter = logging.Formatter('%(asctime)s - %(message)s')
handler.setFormatter(formatter)
logger.addHandler(handler)
return logger
logger = setup_logger(Path(__file__).parent.absolute() / 'ddns-cf.log', 1024*1024, 2)
config = {
"api_token": "xxx",
"api_key": "xxx",
"account_email": "[email protected]",
"zone_id": "xxx",
"lan_ipv4": "192.168.1.123", # LAN ipv4 for vpn networking
"cname_target": "i", # set all domain as a CNAME of i-d.yourdomain.com for ipv6 and i-v.yourdomain.com for LAN ipv4
"domain_name": [
{
# direct will be: i-d.yourdomain.net
# vpn will be: i-v.yourdomain.net
"name": "i",
},{
"name": "file",
},{
"name": "video",
}],
}
def get_dynamic_ipv6_address(interface="ens18"):
try:
# Run the 'ip -6 addr show' command to get the IPv6 address for the interface
result = subprocess.run(['ip', '-6', 'addr', 'show', interface], capture_output=True, text=True, check=True)
# Extract lines containing 'inet6' and scope global
lines = result.stdout.split('\n')
for line in lines:
if 'inet6' in line and 'scope global' in line:
# Extract the IPv6 address part
address = line.split()[1].split('/')[0]
return address
except subprocess.CalledProcessError as e:
logger.error(f"Failed to get IPv6 address for interface {interface}. Error: {e}")
return None
def get_ip(ip_type):
if ip_type == 'LOCAL':
return config['lan_ipv4']
elif ip_type == 'A':
return requests.get('https://4.ipw.cn').text
return get_dynamic_ipv6_address()
def cf_api(endpoint, method, headers={}, data=False):
api_token = config['api_token']
if api_token != '' and api_token != 'api_token_here':
headers = {
"Authorization": "Bearer " + api_token, **headers
}
else:
headers = {
"X-Auth-Email": config['account_email'],
"X-Auth-Key": config['api_key'],
}
try:
if(data == False):
response = requests.request(method, "https://api.cloudflare.com/client/v4/" + endpoint, headers=headers)
else:
response = requests.request(
method, "https://api.cloudflare.com/client/v4/" + endpoint,
headers=headers, json=data)
if response.ok:
return response.json()
else:
logger.error("Error sending '" + method +
"' request to '" + response.url + "':" + response.text)
return None
except Exception as e:
logger.error("An exception occurred while sending '" +
method + "' request to '" + endpoint + "': " + str(e))
return None
def call_api(domain_suffix, c_domain, domain_type):
is_cname_target = c_domain["name"] == config["cname_target"]
real_type = ('A' if domain_type == 'LOCAL' else domain_type) if is_cname_target else "CNAME"
domain_name = c_domain["name"] + domain_suffix
domain_content = get_ip(domain_type) if is_cname_target else (config["cname_target"] + domain_suffix)
record = {
"type": real_type,
"name": domain_name,
"content": domain_content,
"proxied": False,
"ttl": 1
}
dns_records = cf_api("zones/" + config['zone_id'] + f"/dns_records?per_page=100&type={real_type}&name={domain_name}", "GET")
response = {}
if dns_records['result']:
# updating
logger.info(f'updating {domain_name} with {domain_content}')
dns_record = dns_records['result'][0]
if dns_record['content'] == record['content']:
logger.info(f'local ip: {record["content"]} same to server, no need to update.')
else:
response = cf_api("zones/" + config['zone_id'] + "/dns_records/" + dns_record['id'], "PUT", {}, record)
else:
# creating
logger.info(f'creating {domain_name} with {domain_content}')
response = cf_api("zones/" + config['zone_id'] + "/dns_records", "POST", {}, record)
if response and response.get('success', None):
logger.info(f'succeeded set ip to: {record["content"]}.')
def ddns():
response = cf_api("zones/" + config['zone_id'], "GET")
if response is None or response["result"]["name"] is None:
logger.error(f'get domain info error with zones api')
return
zone_result_name = response["result"]["name"]
for c_domain in config["domain_name"]:
call_api('-d.' + zone_result_name, c_domain, 'AAAA')
if config['lan_ipv4']:
call_api('-v.' + zone_result_name, c_domain, 'LOCAL')
if __name__ == '__main__':
try:
ddns()
except Exception as e:
logger.error(f'error: {str(e)}')
Make a systemd service
sh
sudo vim /etc/systemd/system/pyddns.service
ini
[Unit]
Description=CloudFlare DDNS
After=network-online.target syslog.target
[Service]
User=yunyuyuan
ExecStart=/usr/bin/python3 /home/yunyuyuan/ddns/ddns-cf.py
RestartSec=300
Restart=always
[Install]
WantedBy=multi-user.target
sh
sudo systemctl enable --now pyddns.service