Initial Commit
This commit is contained in:
commit
73e6444a00
330
cloudflare_dyndns_update.rb
Executable file
330
cloudflare_dyndns_update.rb
Executable file
@ -0,0 +1,330 @@
|
|||||||
|
#!/usr/bin/env ruby
|
||||||
|
|
||||||
|
# -----
|
||||||
|
#
|
||||||
|
# Copyright (c) 2024 mahesh
|
||||||
|
#
|
||||||
|
# Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
|
# this software and associated documentation files (the "Software"), to deal in
|
||||||
|
# the Software without restriction, including without limitation the rights to
|
||||||
|
# use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
|
||||||
|
# of the Software, and to permit persons to whom the Software is furnished to do
|
||||||
|
# so, subject to the following conditions:
|
||||||
|
#
|
||||||
|
# The above copyright notice and this permission notice shall be included in all
|
||||||
|
# copies or substantial portions of the Software.
|
||||||
|
#
|
||||||
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
# SOFTWARE.
|
||||||
|
#
|
||||||
|
# -----
|
||||||
|
|
||||||
|
require 'pp'
|
||||||
|
require 'optparse'
|
||||||
|
require 'net/http'
|
||||||
|
require 'uri'
|
||||||
|
require 'json'
|
||||||
|
require 'yaml'
|
||||||
|
require 'logger'
|
||||||
|
|
||||||
|
# ------------------
|
||||||
|
# Helper Classes
|
||||||
|
# ------------------
|
||||||
|
class CloudflareDynDNSUpdater
|
||||||
|
attr_accessor :options, :myip, :config, :cache
|
||||||
|
|
||||||
|
def initialize(options)
|
||||||
|
@logger = Logger.new(STDOUT)
|
||||||
|
@options = options
|
||||||
|
@config = read_config_file
|
||||||
|
@logger.level = @options[:debug] ? Logger::DEBUG : Logger::INFO
|
||||||
|
@logger.info($0) { "Starting..." }
|
||||||
|
@query_info = Hash.new
|
||||||
|
@cache_updated = false
|
||||||
|
if @options[:api_key].empty?
|
||||||
|
@logger.error "Need Cloudflare API key to proceed"
|
||||||
|
exit 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def run()
|
||||||
|
#
|
||||||
|
# Load the cache so we know hosts and their IPs were updated
|
||||||
|
# in the past
|
||||||
|
#
|
||||||
|
@cache = read_cache_file
|
||||||
|
@logger.debug @cache.inspect
|
||||||
|
|
||||||
|
#
|
||||||
|
# Get the Public IP (WAN). We will compare this with the IP
|
||||||
|
# in the cache to know if the IP has changed and needs to be
|
||||||
|
# updated in Cloudflare's DNS
|
||||||
|
#
|
||||||
|
@myip = get_public_ip
|
||||||
|
@logger.info "Public IP: #{@myip}"
|
||||||
|
|
||||||
|
#
|
||||||
|
# Query DNS records
|
||||||
|
#
|
||||||
|
process_dns
|
||||||
|
|
||||||
|
# Write updated cache
|
||||||
|
if @cache_updated
|
||||||
|
write_cache_file
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def report()
|
||||||
|
@logger.info($0) { "Done." }
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
#
|
||||||
|
# Read configuration file, which contains the list of hosts and
|
||||||
|
# their credentials. Configuration file uses the YAML format
|
||||||
|
# and has the following template:
|
||||||
|
#
|
||||||
|
# -----------
|
||||||
|
# hosts:
|
||||||
|
# - host: hostone.com
|
||||||
|
# zone_id: <cloudflare hostone zone_id>
|
||||||
|
# - host: hosttwo.com
|
||||||
|
# zone_id: <cloudflare hosttwo zone_id>
|
||||||
|
# - ...
|
||||||
|
# -----------
|
||||||
|
#
|
||||||
|
# Default location of the configuration file is:
|
||||||
|
#
|
||||||
|
# ~/.cfdyndns.yaml
|
||||||
|
#
|
||||||
|
# Alternate location may be provided with [-f|--config_file] option
|
||||||
|
#
|
||||||
|
def read_config_file()
|
||||||
|
if File.exists?(options[:config_file])
|
||||||
|
return YAML.load_file(options[:config_file])
|
||||||
|
else
|
||||||
|
@logger.error "Config file [#{options[:config_file]}] missing!"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
#
|
||||||
|
# To avoid frequent queries to Cloudflare, store the status of
|
||||||
|
# latest update to a cache file. Cloudflare is updated only if
|
||||||
|
# our public IP has changed.
|
||||||
|
#
|
||||||
|
# Default location of cache file is:
|
||||||
|
#
|
||||||
|
# ~/.cfdyndns.cache
|
||||||
|
#
|
||||||
|
# Alternate location may be provided with [-c|--cache_file] option
|
||||||
|
#
|
||||||
|
def read_cache_file()
|
||||||
|
#
|
||||||
|
# Update to Cloudflare may be forced by forgetting cached
|
||||||
|
# values. This is controlled by the [-u|--force_update] option
|
||||||
|
#
|
||||||
|
if options[:force_update]
|
||||||
|
if File.exists?(options[:cache_file])
|
||||||
|
File.delete(options[:cache_file])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
#
|
||||||
|
# Load the cache file
|
||||||
|
#
|
||||||
|
return File.exists?(options[:cache_file]) ? YAML.load_file(options[:cache_file]) : {}
|
||||||
|
end
|
||||||
|
|
||||||
|
#
|
||||||
|
# Store IPs from current update into cache
|
||||||
|
#
|
||||||
|
def write_cache_file()
|
||||||
|
@logger.debug @cache.inspect
|
||||||
|
@logger.info "Writing updated cache file"
|
||||||
|
File.open(options[:cache_file], "w") {|f| f.write(@cache.to_yaml)}
|
||||||
|
end
|
||||||
|
|
||||||
|
#
|
||||||
|
# Acquire the public IP from an external service
|
||||||
|
#
|
||||||
|
def get_public_ip()
|
||||||
|
return Net::HTTP.get(URI('http://ipinfo.io/ip'))
|
||||||
|
end
|
||||||
|
|
||||||
|
#
|
||||||
|
# Query Cloudflare for Domains info
|
||||||
|
#
|
||||||
|
def process_dns()
|
||||||
|
#
|
||||||
|
# Handle each host in the config file at a time
|
||||||
|
#
|
||||||
|
@config['hosts'].each {|h|
|
||||||
|
if @cache[h['host']] && @myip.eql?(@cache[h['host']]['ip'])
|
||||||
|
@logger.info "Skipping #{h['host']} - Already pointing to #{@myip}"
|
||||||
|
else
|
||||||
|
query_sts_good = query_dns(h)
|
||||||
|
|
||||||
|
if query_sts_good
|
||||||
|
update_sts_good = update_dns(h['host'])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
def query_dns(host)
|
||||||
|
sts_good = false
|
||||||
|
url = "https://api.cloudflare.com/client/v4/zones/#{host['zone_id']}/dns_records"
|
||||||
|
|
||||||
|
@logger.debug "Querying Cloudflare for host [#{host['host']}] - #{url}"
|
||||||
|
uri = URI(url)
|
||||||
|
|
||||||
|
Net::HTTP.start(uri.host, uri.port, :use_ssl => true) do |http|
|
||||||
|
request = Net::HTTP::Get.new uri
|
||||||
|
request.add_field 'Content-Type', 'application/json'
|
||||||
|
request.add_field 'Authorization', "Bearer #{@options[:api_key]}"
|
||||||
|
request.add_field 'User-Agent', "#{@options[:user_agent]}"
|
||||||
|
response = http.request request # Net::HTTPResponse object
|
||||||
|
sts_good = handle_query(JSON.parse(response.body))
|
||||||
|
end
|
||||||
|
return sts_good
|
||||||
|
end
|
||||||
|
|
||||||
|
def handle_query(body)
|
||||||
|
if body['success']
|
||||||
|
body['result'].each {|entry|
|
||||||
|
if entry['type'].eql?('A')
|
||||||
|
@query_info[entry['name']] = entry
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
else
|
||||||
|
body['errors'].each {|err|
|
||||||
|
@logger.error "Error status returned #{err['code']}: #{err['message']}"
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
#
|
||||||
|
# Update Cloudflare if IP has changed
|
||||||
|
#
|
||||||
|
def update_dns(host)
|
||||||
|
sts_good = false
|
||||||
|
|
||||||
|
#
|
||||||
|
# Handle each host in the config file at a time
|
||||||
|
#
|
||||||
|
if @query_info.has_key?(host)
|
||||||
|
# Use information from query response when possible
|
||||||
|
q_info = @query_info[host]
|
||||||
|
|
||||||
|
url = "https://api.cloudflare.com/client/v4/zones/#{q_info['zone_id']}/dns_records/#{q_info['id']}"
|
||||||
|
@logger.debug "Updating Cloudflare for host [#{host['host']}] - #{url}"
|
||||||
|
|
||||||
|
uri = URI(url)
|
||||||
|
request_body = {
|
||||||
|
'type' => q_info['type'],
|
||||||
|
'name' => q_info['name'],
|
||||||
|
'content' => @myip
|
||||||
|
}
|
||||||
|
|
||||||
|
@logger.debug request_body.to_json
|
||||||
|
|
||||||
|
Net::HTTP.start(uri.host, uri.port, :use_ssl => true) do |http|
|
||||||
|
request = Net::HTTP::Patch.new uri
|
||||||
|
request.add_field 'Content-Type', 'application/json'
|
||||||
|
request.add_field 'Authorization', "Bearer #{@options[:api_key]}"
|
||||||
|
request.add_field 'User-Agent', "#{@options[:user_agent]}"
|
||||||
|
request.body = request_body.to_json
|
||||||
|
response = http.request request # Net::HTTPResponse object
|
||||||
|
sts_good = handle_update(JSON.parse(response.body))
|
||||||
|
end
|
||||||
|
return sts_good
|
||||||
|
|
||||||
|
else
|
||||||
|
@logger.error "Did not find query information for host #{host}"
|
||||||
|
end
|
||||||
|
return sts_good
|
||||||
|
end
|
||||||
|
|
||||||
|
def handle_update(body)
|
||||||
|
if body['success']
|
||||||
|
entry = body['result']
|
||||||
|
@logger.info "Successfully updated DNS entry type #{entry['type']} - #{entry['name']} - #{entry['content']}"
|
||||||
|
@cache[entry['name']] = {'ip' => entry['content']}
|
||||||
|
@cache_updated = true
|
||||||
|
return true
|
||||||
|
else
|
||||||
|
body['errors'].each {|err|
|
||||||
|
@logger.error "Error status returned #{err['code']}: #{err['message']}"
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# -------------------
|
||||||
|
# Options parsing
|
||||||
|
# -------------------
|
||||||
|
#
|
||||||
|
# Default values of command line options
|
||||||
|
#
|
||||||
|
options = {
|
||||||
|
:config_file => "#{Dir.home}/.cfdyndns.yaml",
|
||||||
|
:cache_file => "#{Dir.home}/.cfdyndns.cache",
|
||||||
|
:user_agent => 'HeshApps Cloudflare Dynamic DNS Updater v1.0',
|
||||||
|
:force_update => false,
|
||||||
|
:debug => false,
|
||||||
|
:api_key => ''
|
||||||
|
}
|
||||||
|
|
||||||
|
#
|
||||||
|
# Get command line options
|
||||||
|
#
|
||||||
|
op = OptionParser.new do |opts|
|
||||||
|
opts.banner = "Usage: cloudflare_dyndns_update.rb [options]"
|
||||||
|
|
||||||
|
opts.on("-d", "--debug", "Enable debug messages") do |debug|
|
||||||
|
options[:debug] = debug
|
||||||
|
end
|
||||||
|
opts.on("-u", "--force_update", "Force DNS update") do |force_update|
|
||||||
|
options[:force_update] = force_update
|
||||||
|
end
|
||||||
|
opts.on("-f", "--config_file FILE", "Location of configuration file") do |file|
|
||||||
|
options[:config_file] = file
|
||||||
|
end
|
||||||
|
opts.on("-c", "--cache_file FILE", "Location of cache file") do |file|
|
||||||
|
options[:cache_file] = file
|
||||||
|
end
|
||||||
|
opts.on("-k", "--api_key KEY", "Cloudflare API authorization key") do |key|
|
||||||
|
options[:api_key] = key
|
||||||
|
end
|
||||||
|
opts.on("-h", "--help", "Display this help") do |help|
|
||||||
|
options[:help] = help
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
op.parse!
|
||||||
|
|
||||||
|
if options[:help]
|
||||||
|
puts op.help
|
||||||
|
exit
|
||||||
|
end
|
||||||
|
|
||||||
|
# ------------------
|
||||||
|
# Execution part
|
||||||
|
# ------------------
|
||||||
|
updater = CloudflareDynDNSUpdater.new(options)
|
||||||
|
updater.run
|
||||||
|
updater.report
|
||||||
|
|
||||||
|
#
|
||||||
|
# Mahesh Asolkar, 2024
|
||||||
|
# vim: ai ts=2 sts=2 et sw=2 filetype=ruby
|
Loading…
Reference in New Issue
Block a user