dyndnsd/lib/dyndnsd.rb

229 lines
6.5 KiB
Ruby
Raw Normal View History

2013-04-27 14:07:14 +02:00
#!/usr/bin/env ruby
2013-04-30 23:19:08 +02:00
require 'etc'
2013-04-27 14:07:14 +02:00
require 'logger'
require 'ipaddr'
require 'json'
require 'yaml'
require 'rack'
require 'metriks'
require 'metriks/reporter/graphite'
2013-04-27 14:07:14 +02:00
require 'dyndnsd/generator/bind'
require 'dyndnsd/updater/command_with_bind_zone'
2013-04-27 14:30:41 +02:00
require 'dyndnsd/responder/dyndns_style'
2013-04-27 14:07:14 +02:00
require 'dyndnsd/responder/rest_style'
require 'dyndnsd/database'
require 'dyndnsd/version'
module Dyndnsd
def self.logger
@logger
end
def self.logger=(logger)
@logger = logger
end
class LogFormatter
def call(lvl, time, progname, msg)
2013-04-27 23:27:55 +02:00
"[%s] %-5s %s\n" % [Time.now.strftime('%Y-%m-%d %H:%M:%S'), lvl, msg.to_s]
2013-04-27 14:07:14 +02:00
end
end
class Daemon
def initialize(config, db, updater)
2013-04-27 14:07:14 +02:00
@users = config['users']
2013-04-27 14:59:25 +02:00
@domain = config['domain']
2013-04-27 14:07:14 +02:00
@db = db
@updater = updater
@db.load
@db['serial'] ||= 1
@db['hosts'] ||= {}
(@db.save; update) if @db.changed?
end
2013-04-27 14:07:14 +02:00
def update
@updater.update(@db)
end
2013-04-27 14:59:25 +02:00
def is_fqdn_valid?(hostname)
return false if hostname.length < @domain.length + 2
return false if not hostname.end_with?(@domain)
name = hostname.chomp(@domain)
return false if not name.match(/^[a-zA-Z0-9_-]+\.$/)
true
end
2013-04-27 14:07:14 +02:00
def call(env)
return [422, {'X-DynDNS-Response' => 'method_forbidden'}, []] if env["REQUEST_METHOD"] != "GET"
return [422, {'X-DynDNS-Response' => 'not_found'}, []] if env["PATH_INFO"] != "/nic/update"
2013-04-27 14:07:14 +02:00
params = Rack::Utils.parse_query(env["QUERY_STRING"])
return [422, {'X-DynDNS-Response' => 'hostname_missing'}, []] if not params["hostname"]
hostnames = params["hostname"].split(',')
2013-04-27 14:59:25 +02:00
# Check if hostname match rules
hostnames.each do |hostname|
return [422, {'X-DynDNS-Response' => 'hostname_malformed'}, []] if not is_fqdn_valid?(hostname)
end
2013-04-27 14:07:14 +02:00
user = env["REMOTE_USER"]
hostnames.each do |hostname|
return [422, {'X-DynDNS-Response' => 'host_forbidden'}, []] if not @users[user]['hosts'].include? hostname
end
myip = nil
if params.has_key?("myip6")
# require presence of myip parameter as valid IPAddr (v4) and valid myip6
return [422, {'X-DynDNS-Response' => 'host_forbidden'}, []] if not params["myip"]
begin
IPAddr.new(params["myip"], Socket::AF_INET)
IPAddr.new(params["myip6"], Socket::AF_INET6)
2018-01-26 13:50:02 +01:00
# myip will be an array
myip = [params["myip"], params["myip6"]]
rescue ArgumentError
return [422, {'X-DynDNS-Response' => 'host_forbidden'}, []]
end
else
# fallback value, always present
myip = env["REMOTE_ADDR"]
# check whether X-Real-IP header has valid IPAddr
if env.has_key?("HTTP_X_REAL_IP")
begin
IPAddr.new(env["HTTP_X_REAL_IP"])
myip = env["HTTP_X_REAL_IP"]
rescue ArgumentError
end
end
# check whether myip parameter has valid IPAddr
if params.has_key?("myip")
begin
IPAddr.new(params["myip"])
myip = params["myip"]
rescue ArgumentError
end
end
2013-04-27 14:07:14 +02:00
end
Metriks.meter('requests.valid').mark
2013-04-27 22:02:54 +02:00
Dyndnsd.logger.info "Request to update #{hostnames} to #{myip} for user #{user}"
changes = []
hostnames.each do |hostname|
if (not @db['hosts'].include? hostname) or (@db['hosts'][hostname] != myip)
changes << :good
@db['hosts'][hostname] = myip
Metriks.meter('requests.good').mark
else
changes << :nochg
Metriks.meter('requests.nochg').mark
end
end
2013-04-27 14:07:14 +02:00
if @db.changed?
@db['serial'] += 1
2013-04-27 22:02:54 +02:00
Dyndnsd.logger.info "Committing update ##{@db['serial']}"
2013-04-27 14:07:14 +02:00
@db.save
update
Metriks.meter('updates.committed').mark
2013-04-27 14:07:14 +02:00
end
[200, {'X-DynDNS-Response' => 'success'}, [changes, myip]]
2013-04-27 14:07:14 +02:00
end
def self.run!
if ARGV.length != 1
puts "Usage: dyndnsd config_file"
exit 1
end
config_file = ARGV[0]
if not File.file?(config_file)
2013-04-27 22:02:54 +02:00
puts "Config file not found!"
2013-04-27 14:07:14 +02:00
exit 1
end
2013-04-27 22:02:54 +02:00
puts "DynDNSd version #{Dyndnsd::VERSION}"
puts "Using config file #{config_file}"
2013-04-27 14:07:14 +02:00
config = YAML::load(File.open(config_file, 'r') { |f| f.read })
2013-04-27 22:02:54 +02:00
if config['logfile']
Dyndnsd.logger = Logger.new(config['logfile'])
else
Dyndnsd.logger = Logger.new(STDOUT)
end
2013-04-27 22:02:54 +02:00
Dyndnsd.logger.progname = "dyndnsd"
Dyndnsd.logger.formatter = LogFormatter.new
Dyndnsd.logger.info "Starting..."
2013-04-30 23:19:08 +02:00
# drop privs (first change group than user)
Process::Sys.setgid(Etc.getgrnam(config['group']).gid) if config['group']
Process::Sys.setuid(Etc.getpwnam(config['user']).uid) if config['user']
2013-04-27 14:07:14 +02:00
# configure metriks
2013-05-29 21:03:25 +02:00
if config['graphite']
host = config['graphite']['host'] || 'localhost'
port = config['graphite']['port'] || 2003
options = {}
options[:prefix] = config['graphite']['prefix'] if config['graphite']['prefix']
reporter = Metriks::Reporter::Graphite.new(host, port, options)
reporter.start
else
reporter = Metriks::Reporter::ProcTitle.new
reporter.add 'good', 'sec' do
Metriks.meter('requests.good').mean_rate
end
reporter.add 'nochg', 'sec' do
Metriks.meter('requests.nochg').mean_rate
end
reporter.start
end
# configure daemon
2013-04-27 14:12:04 +02:00
db = Database.new(config['db'])
2013-04-27 15:08:36 +02:00
updater = Updater::CommandWithBindZone.new(config['domain'], config['updater']['params']) if config['updater']['name'] == 'command_with_bind_zone'
# configure rack
app = Daemon.new(config, db, updater)
2013-04-27 14:07:14 +02:00
app = Rack::Auth::Basic.new(app, "DynDNS") do |user,pass|
allow = ((config['users'].has_key? user) and (config['users'][user]['password'] == pass))
if not allow
Dyndnsd.logger.warn "Login failed for #{user}"
Metriks.meter('requests.auth_failed').mark
end
2013-04-27 22:02:54 +02:00
allow
2013-04-27 14:07:14 +02:00
end
if config['responder'] == 'RestStyle'
app = Responder::RestStyle.new(app)
else
app = Responder::DynDNSStyle.new(app)
end
2013-04-27 14:07:14 +02:00
Signal.trap('INT') do
2013-04-27 22:02:54 +02:00
Dyndnsd.logger.info "Quitting..."
2013-04-27 14:07:14 +02:00
Rack::Handler::WEBrick.shutdown
end
2013-10-08 13:25:35 +02:00
Signal.trap('TERM') do
Dyndnsd.logger.info "Quitting..."
Rack::Handler::WEBrick.shutdown
end
2013-04-27 14:07:14 +02:00
Rack::Handler::WEBrick.run app, :Host => config['host'], :Port => config['port']
end
end
end