opnsense-plugins/dns/dyndns/src/etc/inc/plugins.inc.d/dyndns/phpDynDNS.inc

1994 lines
95 KiB
PHP

<?php
/*
* phpDynDNS
*
* Public Functions
* - updatedns()
*
* Private Functions
* - _update()
* - _checkStatus()
* - _error()
* - _detectChange()
* - _debug()
* - _checkIP()
* +----------------------------------------------------+
* 3322 - Last Tested: 26 May 2017
* All-Inkl - Last Tested: 02 March 2020
* Amazon Route53 - Last Tested: 01 April 2012
* Amazon Route53 v6 - Last Tested: 19 November 2017
* Azure DNS - Last Tested: 16 October 2019
* City Network - Last Tested: 13 November 2013
* Cloudflare - Last Tested: 16 April 2019
* Cloudflare IPv6 - Last Tested: 16 April 2019
* Cloudflare w/API token - Last Tested: 13 June 2020
* DHS - Last Tested: 12 July 2005
* DNS-O-Matic - Last Tested: 9 September 2010
* DNSexit - Last Tested: 20 July 2008
* DigitalOcean - Last Tested: 25 June 2019
* DigitalOcean v6 - Last Tested: 13 May 2020
* Duck DNS - Last Tested: 04 March 2015
* DynDNS Dynamic - Last Tested: 12 July 2005
* EasyDNS - Last Tested: 20 July 2008
* Eurodns - Last Tested: 25 July 2018
* FreeDNS - Last Tested: 06 March 2021
* Gandi LiveDNS - Last Tested: 24 August 2020
* GoDaddy - Last Tested: 10 July 2020
* GoDaddy v6 - Last Tested: 10 July 2020
* Google Domains - Last Tested: 20 February 2017
* GratisDNS - Last Tested: 26 January 2020
* HE.net - Last Tested: 7 July 2013
* HE.net IPv6 - Last Tested: 7 July 2013
* HE.net Tunnel - Last Tested: 28 June 2011
* Hetzner DNS Console - Last Tested: 06 February 2021
* Hetzner DNS Console v6 - Last Tested: 06 February 2021
* HN.org - Last Tested: 12 July 2005
* Linode - Last Tested: 25 February 2020
* Linode v6 - Last Tested: 25 February 2020
* Namecheap - Last Tested: 31 August 2010
* No-IP - Last Tested: 15 May 2020
* ODS - Last Tested: 02 August 2005
* Oray - Last Tested: 26 May 2017
* STRATO - Last Tested: 09 May 2017
* SelfHost - Last Tested: 26 December 2011
* StaticCling - Last Tested: 27 April 2006
* deSEC - Last Tested: 09 September 2020
* deSEC v6 - Last Tested: 09 September 2020
* deSEC v4 + v6 - Last Tested: 09 September 2020
* dynv6 - Last Tested: 25 June 2019
* dynv6 v6 - Last Tested: 25 June 2019
* regfish - Last Tested: 15 August 2017
* regfish v6 - Last Tested: 15 August 2017
* +====================================================+
*
* @author E.Kristensen
* @link http://www.idylldesigns.com/projects/phpdns/
* @version 0.8
* @updated 13 October 05 at 21:02:42 GMT
*
* DNSexit support and multiwan extension for pfSense by Ermal Luçi
* Custom DNS support by Matt Corallo
*/
class updatedns
{
var $_cacheFile;
var $_cacheFile_v6;
var $_debugFile;
var $_UserAgent = 'User-Agent: phpDynDNS/0.8';
var $_errorVerbosity = 0;
var $_dnsService;
var $_dnsUser;
var $_dnsPass;
var $_dnsHost;
var $_dnsIP;
var $_dnsWildcard;
var $_dnsMX;
var $_dnsBackMX;
var $_dnsServer;
var $_dnsPort;
var $_dnsUpdateURL;
var $_dnsZoneID;
var $_dnsResourceID;
var $_dnsTTL;
var $status;
var $_debugID;
var $_if;
var $_dnsResultMatch;
var $_dnsRequestIf;
var $_dnsRequestIfIP;
var $_dnsVerboseLog;
var $_curlIpresolveV4;
var $_curlSslVerifypeer;
var $_dnsMaxCacheAgeDays;
var $_dnsDummyUpdateDone;
var $_forceUpdateNeeded;
var $_useIPv6;
/*
* Public Constructor Function (added 12 July 05) [beta]
* - Gets the dice rolling for the update.
* - $dnsResultMatch should only be used with $dnsService = 'custom'
* - $dnsResultMatch is parsed for '%IP%', which is the IP the provider was updated to,
* - it is otherwise expected to be exactly identical to what is returned by the Provider.
* - $dnsUser, and $dnsPass indicate HTTP Auth for custom DNS, if they are needed in the URL (GET Variables), include them in $dnsUpdateURL.
* - $For custom requests, $dnsUpdateURL is parsed for '%IP%', which is replaced with the new IP.
*/
public function __construct(
$dnsService = '',
$dnsHost = '',
$dnsUser = '',
$dnsPass = '',
$dnsWildcard = 'OFF',
$dnsMX = '',
$dnsIf = '',
$dnsBackMX = '',
$dnsServer = '',
$dnsPort = '',
$dnsUpdateURL = '',
$forceUpdate = false,
$dnsZoneID = '',
$dnsResourceID = '',
$dnsTTL = '',
$dnsResultMatch = '',
$dnsRequestIf = '',
$dnsID = '',
$dnsVerboseLog = false,
$curlIpresolveV4 = false,
$curlSslVerifypeer = true
) {
/* XXX because the call stack is upside down we need to reassemble config parts here... */
$conf = array('host' => $dnsHost, 'id' => $dnsID, 'interface' => $dnsIf);
$this->_cacheFile = dyndns_cache_file($conf, 4);
$this->_cacheFile_v6 = dyndns_cache_file($conf, 6);
$this->_debugFile = dyndns_cache_file($conf, 4) . '.debug';
$this->_dnsServiceList = dyndns_list();
$this->_curlIpresolveV4 = $curlIpresolveV4;
$this->_curlSslVerifypeer = $curlSslVerifypeer;
$this->_dnsVerboseLog = $dnsVerboseLog;
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS: updatedns() starting");
}
$dyndnslck = lock("DDNS" . $dnsID, LOCK_EX);
if (!$dnsService) {
$this->_error(2);
}
switch ($dnsService) {
case 'freedns':
if (!$dnsHost) {
$this->_error(5);
}
break;
case 'linode':
case 'linode-v6':
case 'namecheap':
if (!$dnsPass) {
$this->_error(4);
} elseif (!$dnsHost) {
$this->_error(5);
}
break;
case 'route53':
case 'route53-v6':
if (!$dnsZoneID) {
$this->_error(8);
} elseif (!$dnsTTL) {
$this->_error(9);
}
break;
case 'custom':
case 'custom-v6':
if (!$dnsUpdateURL) {
$this->_error(7);
}
break;
case 'duckdns':
case 'dynv6':
case 'dynv6-v6':
case 'regfish':
case 'regfish-v6':
if (!$dnsUser) {
$this->_error(3);
} elseif (!$dnsHost) {
$this->_error(5);
}
break;
case 'azure':
case 'azurev6':
if (!$dnsUser) {
$this->_error(3);
} elseif (!$dnsPass) {
$this->_error(4);
} elseif (!$dnsHost) {
$this->_error(5);
} elseif (!$dnsResourceID) {
$this->_error(8);
} elseif (!$dnsTTL) {
$this->_error(9);
}
break;
case 'cloudflare-token':
case 'cloudflare-token-v6':
case 'hetzner':
case 'hetzner-v6':
if (!$dnsPass) {
$this->_error(4);
} elseif (!$dnsHost) {
$this->_error(5);
} elseif (!$dnsTTL) {
$this->_error(9);
}
break;
case 'desec':
case 'desec-v4-v6':
case 'desec-v6':
if (!$dnsPass) {
$this->_error(4);
} elseif (!$dnsHost) {
$this->_error(5);
}
break;
default:
if (!$dnsUser) {
$this->_error(3);
} elseif (!$dnsPass) {
$this->_error(4);
} elseif (!$dnsHost) {
$this->_error(5);
}
break;
}
switch ($dnsService) {
case 'all-inkl-v6':
case 'azurev6':
case 'cloudflare-v6':
case 'custom-v6':
case 'digitalocean-v6':
case 'dynv6-v6':
case 'he-net-v6':
case 'godaddy-v6':
case 'linode-v6':
case 'regfish-v6':
case 'route53-v6':
case 'cloudflare-token-v6':
case 'desec-v4-v6':
case 'desec-v6':
case 'hetzner-v6':
$this->_useIPv6 = true;
break;
default:
$this->_useIPv6 = false;
}
$this->_dnsService = strtolower($dnsService);
$this->_dnsUser = $dnsUser;
$this->_dnsPass = $dnsPass;
$this->_dnsHost = $dnsHost;
$this->_dnsServer = $dnsServer;
$this->_dnsPort = $dnsPort;
$this->_dnsWildcard = $dnsWildcard;
$this->_dnsMX = $dnsMX;
$this->_dnsZoneID = $dnsZoneID;
$this->_dnsResourceID = $dnsResourceID;
$this->_dnsTTL = $dnsTTL;
$this->_if = dyndns_failover_interface($dnsIf, $this->_useIPv6 ? 'inet6' : 'all');
$this->_checkIP();
$this->_dnsUpdateURL = $dnsUpdateURL;
$this->_dnsResultMatch = $dnsResultMatch;
$this->_dnsRequestIf = dyndns_failover_interface($dnsRequestIf, $this->_useIPv6 ? 'inet6' : 'all');
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS ({$this->_dnsHost}): running dyndns_failover_interface for {$dnsRequestIf}. found {$this->_dnsRequestIf}");
}
$this->_dnsRequestIfIP = $this->_useIPv6 ? get_interface_ipv6($this->_dnsRequestIf) : get_interface_ip($this->_dnsRequestIf);
$this->_dnsMaxCacheAgeDays = 25;
$this->_dnsDummyUpdateDone = false;
$this->_forceUpdateNeeded = $forceUpdate;
// Ensure that we were able to lookup the IP
if (!is_ipaddr($this->_dnsIP)) {
log_error("Dynamic DNS ({$this->_dnsHost}) There was an error trying to determine the public IP for interface - {$dnsIf}({$this->_if}). Probably interface is not a WAN interface.");
unlock($dyndnslck);
return;
}
$this->_debugID = rand(1000000, 9999999);
if ($forceUpdate == false && $this->_detectChange() == false) {
$this->_error(10);
} else {
switch ($this->_dnsService) {
case '3322':
case 'all-inkl':
case 'all-inkl-v6':
case 'azure':
case 'azurev6':
case 'citynetwork':
case 'cloudflare':
case 'cloudflare-v6':
case 'cloudflare-token':
case 'cloudflare-token-v6':
case 'custom':
case 'custom-v6':
case 'desec':
case 'desec-v4-v6':
case 'desec-v6':
case 'dhs':
case 'digitalocean':
case 'gandi-livedns':
case 'dnsexit':
case 'dnsomatic':
case 'duckdns':
case 'dyndns':
case 'dyndns-custom':
case 'dyndns-static':
case 'dyns':
case 'dynv6':
case 'dynv6-v6':
case 'easydns':
case 'eurodns':
case 'freedns':
case 'godaddy':
case 'godaddy-v6':
case 'googledomains':
case 'gratisdns':
case 'he-net':
case 'he-net-tunnelbroker':
case 'he-net-v6':
case 'hetzner':
case 'hetzner-v6':
case 'hn':
case 'linode':
case 'linode-v6':
case 'loopia':
case 'namecheap':
case 'noip':
case 'noip-free':
case 'ods':
case 'oray':
case 'ovh-dynhost':
case 'regfish':
case 'regfish-v6':
case 'route53':
case 'route53-v6':
case 'selfhost':
case 'staticcling':
case 'strato':
case 'zoneedit':
$this->_update();
if ($this->_dnsDummyUpdateDone == true) {
// If a dummy update was needed, then sleep a while and do the update again to put the proper address back.
// Some providers (e.g. No-IP free accounts) need to have at least 1 address change every month.
// If the address has not changed recently, or the user did "Force Update", then the code does
// a dummy address change for providers like this.
sleep(10);
$this->_update();
}
break;
default:
$this->_error(6);
break;
}
}
unlock($dyndnslck);
}
/*
* Private Function (added 12 July 05) [beta]
* Send Update To Selected Service.
*/
function _update()
{
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS ({$this->_dnsHost} via {$this->_dnsServiceList[$this->_dnsService]}): _update() starting.");
}
if ($this->_dnsService != 'ods' and $this->_dnsService != 'route53' and $this->_dnsService != 'route53-v6') {
$ch = curl_init();
curl_setopt($ch, CURLOPT_HEADER, 0);
curl_setopt($ch, CURLOPT_USERAGENT, $this->_UserAgent);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_INTERFACE, $this->_dnsRequestIfIP);
curl_setopt($ch, CURLOPT_TIMEOUT, 15);
}
switch ($this->_dnsService) {
case 'dyndns':
case 'dyndns-static':
case 'dyndns-custom':
if (isset($this->_dnsWildcard) && $this->_dnsWildcard != "OFF") {
$this->_dnsWildcard = "ON";
}
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
$server = "https://members.dyndns.org/nic/update";
$port = "";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, $server . $port . '?system=dyndns&hostname=' . $this->_dnsHost . '&myip=' . $this->_dnsIP . '&wildcard=' . $this->_dnsWildcard . '&mx=' . $this->_dnsMX . '&backmx=NO');
break;
case 'dhs':
$post_data['hostscmd'] = 'edit';
$post_data['hostscmdstage'] = '2';
$post_data['type'] = '4';
$post_data['updatetype'] = 'Online';
$post_data['mx'] = $this->_dnsMX;
$post_data['mx2'] = '';
$post_data['txt'] = '';
$post_data['offline_url'] = '';
$post_data['cloak'] = 'Y';
$post_data['cloak_title'] = '';
$post_data['ip'] = $this->_dnsIP;
$post_data['domain'] = 'dyn.dhs.org';
$post_data['hostname'] = $this->_dnsHost;
$post_data['submit'] = 'Update';
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
$server = "https://members.dhs.org/nic/hosts";
$port = "";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, '{$server}{$port}');
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
curl_setopt($ch, CURLOPT_POSTFIELDS, $post_data);
break;
case 'noip':
case 'noip-free':
curl_setopt_array($ch, [
CURLOPT_SSL_VERIFYPEER => true,
CURLOPT_USERPWD => $this->_dnsUser . ':' . $this->_dnsPass
]);
$server = "https://dynupdate.no-ip.com/nic/update";
$port = "";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
if (
($this->_dnsService == "noip-free") &&
($this->_forceUpdateNeeded == true) &&
($this->_dnsDummyUpdateDone == false)
) {
// Update the IP to a dummy value to force No-IP free accounts to see a change.
$iptoset = "192.168.1.1";
$this->_dnsDummyUpdateDone = true;
log_error("Dynamic DNS ({$this->_dnsHost}): Processing dummy update on No-IP free account. IP temporarily set to " . $iptoset);
} else {
$iptoset = $this->_dnsIP;
}
curl_setopt($ch, CURLOPT_URL, $server . $port . '?hostname=' . $this->_dnsHost . '&myip=' . $iptoset);
break;
case 'easydns':
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
$server = "https://members.easydns.com/dyn/dyndns.php";
$port = "";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, $server . $port . '?hostname=' . $this->_dnsHost . '&myip=' . $this->_dnsIP . '&wildcard=' . $this->_dnsWildcard . '&mx=' . $this->_dnsMX . '&backmx=' . $this->_dnsBackMX);
break;
case 'hn':
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
$server = "http://dup.hn.org/vanity/update";
$port = "";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, $server . $port . '?ver=1&IP=' . $this->_dnsIP);
break;
case 'zoneedit':
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
$server = "https://dynamic.zoneedit.com/auth/dynamic.html";
$port = "";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, "{$server}{$port}?host=" . $this->_dnsHost);
break;
case 'dyns':
/* XXX HTTPS is currently broken for them */
$server = 'http://www.dyns.cx/postscript011.php';
$port = '';
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, $server . $port . '?username=' . urlencode($this->_dnsUser) . '&password=' . $this->_dnsPass . '&host=' . $this->_dnsHost);
break;
case 'ods':
$misc_errno = 0;
$misc_error = "";
$server = "ods.org";
$port = "";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
$this->con['socket'] = fsockopen("{$server}{$port}", "7070", $misc_errno, $misc_error, 30);
/* Check that we have connected */
if (!$this->con['socket']) {
print "error! could not connect.";
break;
}
/* Here is the loop. Read the incoming data (from the socket connection) */
while (!feof($this->con['socket'])) {
$this->con['buffer']['all'] = trim(fgets($this->con['socket'], 4096));
$code = substr($this->con['buffer']['all'], 0, 3);
sleep(1);
switch ($code) {
case 100:
fputs($this->con['socket'], "LOGIN " . $this->_dnsUser . " " . $this->_dnsPass . "\n");
break;
case 225:
fputs($this->con['socket'], "DELRR " . $this->_dnsHost . " A\n");
break;
case 901:
fputs($this->con['socket'], "ADDRR " . $this->_dnsHost . " A " . $this->_dnsIP . "\n");
break;
case 795:
fputs($this->con['socket'], "QUIT\n");
break;
}
}
$this->_checkStatus(0, $code);
break;
case 'freedns':
curl_setopt($ch, CURLOPT_URL, 'https://sync.afraid.org/u/' . $this->_dnsPass . '/');
break;
case 'dnsexit':
curl_setopt($ch, CURLOPT_URL, 'https://update.dnsexit.com/RemoteUpdate.sv?login=' . urlencode($this->_dnsUser) . '&password=' . $this->_dnsPass . '&host=' . $this->_dnsHost . '&myip=' . $this->_dnsIP);
break;
case 'loopia':
$this->_dnsWildcard = (isset($this->_dnsWildcard) && $this->_dnsWildcard == true) ? 'ON' : 'OFF';
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
curl_setopt($ch, CURLOPT_URL, 'https://dns.loopia.se/XDynDNSServer/XDynDNS.php?hostname=' . $this->_dnsHost . '&myip=' . $this->_dnsIP . '&wildcard=' . $this->_dnsWildcard);
break;
case 'staticcling':
curl_setopt($ch, CURLOPT_URL, 'https://www.staticcling.org/update.html?login=' . urlencode($this->_dnsUser) . '&pass=' . $this->_dnsPass);
break;
case 'dnsomatic':
/* Example syntax
https://username:password@updates.dnsomatic.com/nic/update?hostname=yourhostname&myip=ipaddress&wildcard=NOCHG&mx=NOCHG&backmx=NOCHG
*/
if (isset($this->_dnsWildcard) && $this->_dnsWildcard != "OFF") {
$this->_dnsWildcard = "ON";
}
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
/*
Reference: https://www.dnsomatic.com/wiki/api
DNS-O-Matic usernames are 3-25 characters.
DNS-O-Matic passwords are 6-20 characters.
All ASCII letters and numbers accepted.
Dots, dashes, and underscores allowed, but not at the beginning or end of the string.
Required: "rawurlencode" http://www.php.net/manual/en/function.rawurlencode.php
Encodes the given string according to RFC 3986.
*/
$server = "https://" . rawurlencode($this->_dnsUser) . ":" . rawurlencode($this->_dnsPass) . "@updates.dnsomatic.com/nic/update?hostname=";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, $server . $this->_dnsHost . '&myip=' . $this->_dnsIP . '&wildcard=' . $this->_dnsWildcard . '&mx=' . $this->_dnsMX . '&backmx=NOCHG');
break;
case 'namecheap':
/* Example:
https://dynamicdns.park-your-domain.com/update?host=[host_name]&domain=[domain.com]&password=[domain_password]&ip=[your_ip]
*/
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
$dparts = explode(".", trim($this->_dnsHost));
$domain_part_count = ($dparts[count($dparts) - 1] == "uk") ? 3 : 2;
$domain_offset = count($dparts) - $domain_part_count;
$hostname = implode(".", array_slice($dparts, 0, $domain_offset));
$domain = implode(".", array_slice($dparts, $domain_offset));
$dnspass = trim($this->_dnsPass);
$server = "https://dynamicdns.park-your-domain.com/update?host={$hostname}&domain={$domain}&password={$dnspass}&ip={$this->_dnsIP}";
curl_setopt($ch, CURLOPT_URL, $server);
break;
case 'he-net':
case 'he-net-v6':
$server = "https://dyn.dns.he.net/nic/update?";
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_URL, $server . 'hostname=' . $this->_dnsHost . '&password=' . $this->_dnsPass . '&myip=' . $this->_dnsIP);
break;
case 'he-net-tunnelbroker':
$server = "https://ipv4.tunnelbroker.net/nic/update?";
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
curl_setopt($ch, CURLOPT_URL, $server . 'hostname=' . $this->_dnsHost);
break;
case 'digitalocean':
case 'digitalocean-v6':
/*
* dnsHost should be the root domain
* dnsUser should be the record ID
* dnsPass should be the API key
*/
$server = "https://api.digitalocean.com/v2/domains/" . $this->_dnsHost . "/records/" . $this->_dnsUser;
$hostData = array("data" => "{$this->_dnsIP}");
/*
* DigitalOcean does not offer the API via IPv6, so we need
* to force sending the request using IPv4 when updating for IPv6
*/
curl_setopt($ch, CURLOPT_INTERFACE, $this->_dnsRequestIf);
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'PUT');
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_HTTPHEADER, array(
"Authorization: Bearer {$this->_dnsPass}",
'Content-Type: application/json'
));
curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($hostData));
curl_setopt($ch, CURLOPT_URL, $server);
break;
case 'gandi-livedns':
/*
* https://github.com/vizion8-dan
* Tested on OPNsense 20.1.8_1-amd64 - IPv4 (A)
* dnsHost ("Hostname" field in OPNsense) should be the 2nd-level domain ("example.org")
* dnsUser ("Username" field in OPNsense) should be the subdomain / A-record ("myrecord" in 2nd-level domain)
* dnsPass should be the Gandi-API key
*/
$server = "https://dns.api.gandi.net/api/v5/domains/" . $this->_dnsHost . "/records/" . $this->_dnsUser . "/A";
$body = '{"rrset_ttl":"' . "300" . '", "rrset_values":["' . $this->_dnsIP . '"]}';
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'PUT');
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_HTTPHEADER, array(
"X-Api-Key: {$this->_dnsPass}",
'Content-Type: application/json'
));
curl_setopt($ch, CURLOPT_POSTFIELDS, $body);
curl_setopt($ch, CURLOPT_URL, $server);
break;
case 'selfhost':
if (isset($this->_dnsWildcard) && $this->_dnsWildcard != "OFF") {
$this->_dnsWildcard = "ON";
}
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
$server = "https://carol.selfhost.de/nic/update";
$port = "";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, $server . $port . '?system=dyndns&hostname=' . $this->_dnsHost . '&myip=' . $this->_dnsIP . '&wildcard=' . $this->_dnsWildcard . '&mx=' . $this->_dnsMX . '&backmx=NO');
break;
case 'route53':
case 'route53-v6':
/* Setting Variables */
$hostname = "{$this->_dnsHost}.";
$ZoneID = $this->_dnsZoneID;
$AccessKeyId = $this->_dnsUser;
$SecretAccessKey = $this->_dnsPass;
$NewIP = $this->_dnsIP;
$NewTTL = $this->_dnsTTL;
$RecordType = ($this->_useIPv6) ? "AAAA" : "A";
/* Set Amazon AWS Credentials for this record */
$r53 = new Route53($AccessKeyId, $SecretAccessKey);
/* Function to find old values of records in Route 53 */
if (!function_exists('Searchrecords')) {
function SearchRecords($records, $name)
{
if (!is_array($records)) {
return false;
}
$result = array();
foreach ($records as $record) {
if (strtolower($record['Name']) == strtolower($name)) {
$result [] = $record;
}
}
return ($result) ? $result : false;
}
}
$records = $r53->listResourceRecordSets("/hostedzone/$ZoneID");
/* Get IP for your hostname in Route 53 */
if (false !== ($a_result = SearchRecords($records['ResourceRecordSets'], "$hostname"))) {
/**
* if hostname for ipv4 and ipv6 is the same, a_result contains more than 1 item
* we need to get the item that corresponds to the record type
*/
$oldTTLResult = null;
$oldIPResult = null;
foreach ($a_result as $resultItem) {
if ($RecordType === $resultItem['Type']) {
$oldTTLResult = $resultItem["TTL"];
$oldIPResult = $resultItem["ResourceRecords"][0];
}
}
$OldTTL = $oldTTLResult;
$OldIP = $oldIPResult;
} else {
$OldIP = "";
}
/* Check if we need to update DNS Record */
if ($OldIP !== $NewIP) {
if (!empty($OldIP)) {
/* Your Hostname already exists, deleting and creating it again */
$changes = array();
$changes[] = $r53->prepareChange("DELETE", $hostname, $RecordType, $OldTTL, $OldIP);
$changes[] = $r53->prepareChange("CREATE", $hostname, $RecordType, $NewTTL, $NewIP);
$result = $r53->changeResourceRecordSets("/hostedzone/$ZoneID", $changes);
} else {
/* Your Hostname does not exist yet, creating it */
$changes = $r53->prepareChange("CREATE", $hostname, $RecordType, $NewTTL, $NewIP);
$result = $r53->changeResourceRecordSets("/hostedzone/$ZoneID", $changes);
}
}
$this->_checkStatus(0, $result);
break;
case 'custom':
case 'custom-v6':
if ($this->_curlIpresolveV4) {
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
}
if ($this->_curlSslVerifypeer) {
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
} else {
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
}
if ($this->_dnsUser != '') {
curl_setopt($ch, CURLOPT_USERPWD, "{$this->_dnsUser}:{$this->_dnsPass}");
}
$server = str_replace("%IP%", $this->_dnsIP, $this->_dnsUpdateURL);
curl_setopt($ch, CURLOPT_URL, $server);
break;
case 'cloudflare':
case 'cloudflare-v6':
case 'cloudflare-token':
case 'cloudflare-token-v6':
$baseUrl = 'https://api.cloudflare.com/client/v4';
$fqdn = str_replace(' ', '', $this->_dnsHost);
$recordType = ($this->_useIPv6) ? 'AAAA' : 'A';
$ttlData = intval($this->_dnsTTL) < 1 ? 1 : intval($this->_dnsTTL);
$hostData = array(
"content" => "{$this->_dnsIP}",
"type" => $recordType,
"name" => $fqdn,
"ttl" => $ttlData
);
// Determine if service is token based or user/password based and define appropriate header
if (strpos($this->_dnsService, 'token') !== false) {
$headerAuth = array(
"Authorization: Bearer {$this->_dnsPass}",
'Content-Type: application/json'
);
} else {
$headerAuth = array(
"X-Auth-Email: {$this->_dnsUser}",
"X-Auth-Key: {$this->_dnsPass}",
'Content-Type: application/json'
);
}
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_HTTPHEADER, $headerAuth);
// Get all zone info
$zonesUrl = "$baseUrl/zones";
curl_setopt($ch, CURLOPT_URL, $zonesUrl);
$output = json_decode(curl_exec($ch));
$zoneId = null; // Set default value
if (!empty($output->result)) {
// Iterate zone objects, check if $fqdn is equal to or ends with zone name
foreach ($output->result as $key => $zoneObj) {
if (preg_match("/^{$zoneObj->name}$|\.{$zoneObj->name}$/", $fqdn)) {
// Found matching zone
$zoneId = $zoneObj->id;
// Get $hostName from $fqdn, set $domainName
// These are only really used for log messages.
$hostName = preg_replace("/\.?{$zoneObj->name}$/", '', $fqdn);
$domainName = $zoneObj->name;
break;
}
}
}
if ($zoneId) { // If zone ID was found get host ID
$dnsRecordsUrl = "$zonesUrl/$zoneId/dns_records";
$getHostId = "$dnsRecordsUrl?name=$fqdn&type=$recordType";
curl_setopt($ch, CURLOPT_URL, $getHostId);
$output = json_decode(curl_exec($ch));
$recordCount = !empty($output->result) ? count($output->result) : 0;
if ($recordCount === 0) {
// create record
curl_setopt($ch, CURLOPT_URL, $dnsRecordsUrl);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST');
} elseif ($recordCount >= 1) {
// update record
$recordId = $output->result[0]->id;
$hostData["proxied"] = $output->result[0]->proxied;
curl_setopt($ch, CURLOPT_URL, "$dnsRecordsUrl/$recordId");
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'PUT');
}
curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($hostData));
if ($recordCount > 1) {
log_error("Dynamic DNS ($fqdn): Warning: multiple records for $hostName found");
}
}
break;
case 'hetzner':
case 'hetzner-v6':
$baseUrl = 'https://dns.hetzner.com/api/v1';
$fqdn = str_replace(' ', '', $this->_dnsHost);
$recordType = ($this->_useIPv6) ? 'AAAA' : 'A';
$ttlData = intval($this->_dnsTTL) < 1 ? 120 : intval($this->_dnsTTL);
$hostData = [
"value" => "{$this->_dnsIP}",
"type" => $recordType,
"name" => "",
"ttl" => $ttlData,
"zone_id" => ""
];
$headerAuth = [
"Auth-API-Token: {$this->_dnsPass}",
'Content-Type: application/json'
];
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_HTTPHEADER, $headerAuth);
// Get all zone info
$zonesUrl = "$baseUrl/zones";
curl_setopt($ch, CURLOPT_URL, $zonesUrl);
$output = json_decode(curl_exec($ch));
$zoneId = null; // Set default value
if (!empty($output->zones)) {
// Iterate zone objects, check if $fqdn is equal to or ends with zone name
foreach ($output->zones as $key => $zoneObj) {
if (preg_match("/^{$zoneObj->name}$|\.{$zoneObj->name}$/", $fqdn)) {
// Found matching zone
$zoneId = $zoneObj->id;
// Get $hostName from $fqdn, set $domainName
// These are only really used for log messages.
$hostName = preg_replace("/\.?{$zoneObj->name}$/", '', $fqdn);
$domainName = $zoneObj->name;
break;
}
}
}
if ($zoneId) { // If zone ID was found get host ID
$dnsRecordsUrl = "$baseUrl/records?zone_id=$zoneId";
curl_setopt($ch, CURLOPT_URL, $dnsRecordsUrl);
$output = json_decode(curl_exec($ch));
$recordId = null;
if (!empty($output->records)) {
// Iterate zone objects, check if $hostName exist of the same type
foreach ($output->records as $key => $recordObj) {
if (preg_match("/^{$recordObj->name}$/", $hostName)) {
if ($recordObj->type == $recordType) {
// Found matching host
$recordId = $recordObj->id;
break;
}
}
}
}
if ($recordId) { // If record ID was found, update record
$setRecordUrl = "$baseUrl/records/$recordId";
curl_setopt($ch, CURLOPT_URL, $setRecordUrl);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'PUT');
} else {
$setRecordUrl = "$baseUrl/records";
curl_setopt($ch, CURLOPT_URL, $setRecordUrl);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST');
}
$hostData["zone_id"] = $zoneId;
$hostData["name"] = $hostName;
curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($hostData));
}
break;
case 'eurodns':
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
$server = "https://update.eurodyndns.org/update/";
$port = "";
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, $server . $port . '?hostname=' . $this->_dnsHost . '&myip=' . $this->_dnsIP);
break;
case 'gratisdns':
$server = "https://admin.gratisdns.com/ddns.php";
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
if (substr_count($this->_dnsHost, ".") < 2) {
$domain = $this->_dnsHost;
$hostname = $this->_dnsHost;
} else {
list($hostname, $domain) = explode(".", $this->_dnsHost, 2);
}
curl_setopt($ch, CURLOPT_URL, $server . '?u=' . urlencode($this->_dnsUser) . '&p=' . $this->_dnsPass . '&h=' . $this->_dnsHost . '&d=' . $domain . '&i=' . $this->_dnsIP);
break;
case 'ovh-dynhost':
if (isset($this->_dnsWildcard) && $this->_dnsWildcard != "OFF") {
$this->_dnsWildcard = "ON";
}
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
$server = "https://www.ovh.com/nic/update";
$port = "";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, $server . $port . '?system=dyndns&hostname=' . $this->_dnsHost . '&myip=' . $this->_dnsIP . '&wildcard=' . $this->_dnsWildcard . '&mx=' . $this->_dnsMX . '&backmx=NO');
break;
case 'citynetwork':
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
$server = 'https://dyndns.citynetwork.se/nic/update';
$port = "";
if ($this->_dnsServer) {
$server = $this->_dnsServer;
}
if ($this->_dnsPort) {
$port = ":" . $this->_dnsPort;
}
curl_setopt($ch, CURLOPT_URL, $server . $port . '?hostname=' . $this->_dnsHost . '&myip=' . $this->_dnsIP);
break;
case 'duckdns':
$server = "https://www.duckdns.org/update";
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_URL, $server . '?domains=' . str_replace('.duckdns.org', '', $this->_dnsHost) . '&token=' . urlencode($this->_dnsUser));
break;
case 'dynv6':
$server = "https://ipv4.dynv6.com/api/update";
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_INTERFACE, $this->_dnsRequestIf);
curl_setopt($ch, CURLOPT_DNS_LOCAL_IP4, $this->_dnsIP);
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
curl_setopt($ch, CURLOPT_URL, $server . '?hostname=' . $this->_dnsHost . '&ipv4=' . $this->_dnsIP . '&token=' . $this->_dnsUser);
break;
case 'dynv6-v6':
$server = "https://ipv6.dynv6.com/api/update";
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_INTERFACE, $this->_dnsRequestIf);
curl_setopt($ch, CURLOPT_DNS_LOCAL_IP6, $this->_dnsIP);
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V6);
curl_setopt($ch, CURLOPT_URL, $server . '?hostname=' . $this->_dnsHost . '&ipv6=' . $this->_dnsIP . '&token=' . $this->_dnsUser);
break;
case 'googledomains':
$server = "https://domains.google.com/nic/update";
$post_data['hostname'] = $this->_dnsHost;
$post_data['myip'] = $this->_dnsIP;
$post_data['offline'] = 'no';
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
curl_setopt($ch, CURLOPT_POSTFIELDS, $post_data);
curl_setopt($ch, CURLOPT_URL, $server);
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
break;
case 'strato':
$server = "https://dyndns.strato.com/nic/update?hostname={$this->_dnsHost}&myip={$this->_dnsIP}";
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
curl_setopt($ch, CURLOPT_URL, $server);
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
break;
case '3322':
$server = "http://members.3322.net/dyndns/update?hostname={$this->_dnsHost}&myip={$this->_dnsIP}";
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
curl_setopt($ch, CURLOPT_URL, $server);
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
break;
case 'oray':
$server = "http://ddns.oray.com/ph/update?hostname={$this->_dnsHost}&myip={$this->_dnsIP}";
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
curl_setopt($ch, CURLOPT_URL, $server);
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
break;
case 'regfish':
case 'regfish-v6':
$family = $this->_useIPv6 ? 'ipv6' : 'ipv4';
$server = "https://dyndns.regfish.de/?fqdn={$this->_dnsHost}&{$family}={$this->_dnsIP}&forcehost=1&token=" . urlencode($this->_dnsUser);
curl_setopt($ch, CURLOPT_URL, $server);
break;
case 'linode':
case 'linode-v6':
$baseUrl = "https://api.linode.com/v4";
$fqdn = trim($this->_dnsHost);
$recordType = ($this->_useIPv6) ? 'AAAA' : 'A';
if ($this->_dnsWildcard == 'ON') {
$fqdn = "*.$fqdn";
}
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_HTTPHEADER, array(
'Accept: application/json',
'Authorization: Bearer ' . $this->_dnsPass,
'Content-Type: application/json'
));
$domainsUrl = "$baseUrl/domains";
curl_setopt($ch, CURLOPT_URL, $domainsUrl);
$output = json_decode(curl_exec($ch));
$domainId = null;
if (!empty($output->data)) {
// Find matching domain and split the hostname part from it
foreach ($output->data as $key => $domainObj) {
if (preg_match("/^{$domainObj->domain}$|\.{$domainObj->domain}$/", $fqdn)) {
$domainId = $domainObj->id;
$hostName = preg_replace("/\.?{$domainObj->domain}$/", '', $fqdn);
$domainName = $domainObj->domain;
break;
}
}
}
if ($domainId) {
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS ($fqdn): Found domain name: $domainName, ID: $domainId");
}
$dnsRecordsUrl = "$domainsUrl/$domainId/records";
curl_setopt($ch, CURLOPT_URL, $dnsRecordsUrl);
$output = json_decode(curl_exec($ch));
$recordId = null;
if (!empty($output->data)) {
// Find matching record
foreach ($output->data as $key => $recordObj) {
if ($recordObj->type == $recordType && $recordObj->name == $hostName) {
$recordId = $recordObj->id;
break;
}
}
}
$hostData = [ 'target' => "{$this->_dnsIP}" ];
if ($recordId) {
// Update record
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS ($fqdn): Updating existing record ID: $recordId");
}
curl_setopt($ch, CURLOPT_URL, "$dnsRecordsUrl/$recordId");
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'PUT');
} else {
// Create record
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS ($fqdn): Creating new record");
}
$hostData['type'] = $recordType;
$hostData['name'] = $hostName;
// Linode will round up to the nearest valid TTL
$hostData['ttl_sec'] = 0;
curl_setopt($ch, CURLOPT_URL, $dnsRecordsUrl);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST');
}
curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($hostData));
} else {
log_error("Dynamic DNS($fqdn): No zone found for domain");
}
break;
case 'azurev6':
case 'azure':
$hostname = "{$this->_dnsHost}";
$resourceid = trim($this->_dnsResourceID);
$app_id = $this->_dnsUser;
$client_secret = $this->_dnsPass;
$newip = $this->_dnsIP;
$newttl = $this->_dnsTTL;
// ensure resourceid starts with / and has no trailing /
$resourceid = '/' . trim($resourceid, '/');
// extract subscription id from resource id
preg_match('/\\/subscriptions\\/(?<sid>[^\\/]*)/', $resourceid, $result);
$subscriptionid = isset($result['sid']) ? $result['sid'] : '';
if (isset($result['sid'])) {
$subscriptionid = $result['sid'];
} else {
log_error("Azure subscription id not found in resource id ({$resourceid})");
return false;
}
// find tenant id from subscription id
curl_setopt($ch, CURLOPT_URL, "https://management.azure.com/subscriptions/" . $subscriptionid . "?api-version=2016-09-01");
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_HEADER, 1);
curl_setopt($ch, CURLOPT_NOBODY, 1);
$output = curl_exec($ch);
$pattern = '/Bearer authorization_uri="https:\\/\\/login.windows.net\\/(?<tid>[^"]*)/i';
preg_match($pattern, $output, $result);
if (isset($result['tid'])) {
$tenantid = $result['tid'];
} else {
log_error("Tenant ID not found");
return false;
}
// get an bearer token
curl_setopt($ch, CURLOPT_URL, "https://login.microsoftonline.com/" . $tenantid . "/oauth2/token");
curl_setopt($ch, CURLOPT_POST, 1);
$body = "resource=" . urlencode("https://management.core.windows.net/") . "&grant_type=client_credentials&client_id=" . $app_id . "&client_secret=" . urlencode($client_secret);
curl_setopt($ch, CURLOPT_POSTFIELDS, $body);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
$server_output = curl_exec($ch);
$httpcode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
preg_match("/\"access_token\":\"(?<tok>[^\"]*)\"/", $server_output, $result);
if (isset($result['tok'])) {
$bearertoken = $result['tok'];
} else {
log_error("no valid bearer token");
return false;
}
// Update the DNS record
if ($this->_useIPv6) {
$url = "https://management.azure.com" . $resourceid . "/AAAA/" . $hostname . "?api-version=2017-09-01";
$body = '{"properties":{"TTL":"' . $newttl . '", "AAAARecords":[{"ipv6Address":"' . $newip . '"}]}}';
} else {
$url = "https://management.azure.com" . $resourceid . "/A/" . $hostname . "?api-version=2017-09-01";
$body = '{"properties":{"TTL":"' . $newttl . '", "ARecords":[{"ipv4Address":"' . $newip . '"}]}}';
}
$request_headers = array();
$request_headers[] = 'Accept: application/json';
$request_headers[] = 'Authorization: Bearer ' . $bearertoken;
$request_headers[] = 'Content-Type: application/json';
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_USERAGENT, $this->_UserAgent);
curl_setopt($ch, CURLOPT_HTTPHEADER, $request_headers);
curl_setopt($ch, CURLOPT_POSTFIELDS, $body);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_HEADER, 1);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'PUT');
break;
case 'all-inkl':
case 'all-inkl-v6':
$server = "https://dyndns.kasserver.com/";
$url = $server . '?myip=' . $this->_dnsIP;
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsUser . ':' . $this->_dnsPass);
// fix: All-Inkl dyndns.kasserver.com only supports v4
$ipv4 = get_interface_ip($this->_dnsRequestIf);
if (!is_ipaddr($ipv4)) {
log_error("Dynamic DNS ({$this->_dnsHost}): (Error) Need a IPv4 address on $this->_dnsRequestIf!");
return false;
}
curl_setopt($ch, CURLOPT_INTERFACE, $ipv4);
// fix end
case 'godaddy':
case 'godaddy-v6':
/* Read https://developer.godaddy.com/ for API documentation */
$baseApiUrl = 'https://api.godaddy.com/v1/domains/';
$recordType = $this->_useIPv6 ? "AAAA" : "A";
$splitHost = explode('.', trim($this->_dnsHost));
$dnsDomain = '*';
if ($this->_dnsWildcard != 'ON') {
$dnsDomain = array_shift($splitHost);
}
$dnsHost = implode('.', $splitHost);
$url = $baseApiUrl . $dnsHost . '/records/' . $recordType . '/' . $dnsDomain;
/* body can contain multiple options (data, port, priority, service, ttl, weight) */
$data = array();
$data[] = array('data' => $this->_dnsIP);
if ($this->_dnsTTL) {
$data[0]['ttl'] = $this->_dnsTTL;
} else {
// minimum allowed by GoDaddy
$data[0]['ttl'] = 600;
}
$jsonBody = json_encode($data);
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS: calling $url with body: $jsonBody");
}
/* PUT JSON /v1/domains/{domain}/records/{type}/{name} */
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, "PUT");
curl_setopt($ch, CURLOPT_HTTPHEADER, array(
'Accept: application/json',
'Content-Type: application/json',
'Authorization: sso-key ' . $this->_dnsUser . ':' . $this->_dnsPass
));
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_POSTFIELDS, $jsonBody);
break;
case 'desec':
/*
* https://desec.readthedocs.io/en/latest/dyndns/update-api.html
* dnsHost should be the domain
* dnsPass should be the token, NOT the token id
* IPv6 is empty so deSEC API will not set this to the IPv6 of the sending interface if the connection is made via IPv6
*/
$server = "https://update.dedyn.io/";
$url = '?hostname=' . $this->_dnsHost . '&myipv4=' . $this->_dnsIP . '&myipv6=""';
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsHost . ':' . $this->_dnsPass);
curl_setopt($ch, CURLOPT_URL, $server . $url);
break;
case 'desec-v4-v6':
/*
* https://desec.readthedocs.io/en/latest/dyndns/update-api.html
* dnsHost should be the domain
* dnsPass should be the token, NOT the 36-character token id (https://forum.netgate.com/post/930114)
* IPv4 is determined by deSEC API via the sending interface
*/
// temporarily disable useIPv6 to get IPv4 Address
$this->_useIPv6 = false;
$ipv4 = $this->_checkIP();
$this->_useIPv6 = true;
$server = "https://update.dedyn.io/";
$url = '?hostname=' . $this->_dnsHost . '&myipv4=' . $ipv4 . '&myipv6=' . $this->_dnsIP;
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsHost . ':' . $this->_dnsPass);
curl_setopt($ch, CURLOPT_URL, $server . $url);
break;
case 'desec-v6':
/*
* https://desec.readthedocs.io/en/latest/dyndns/update-api.html
* dnsHost should be the domain
* dnsPass should be the token, NOT the 36-character token id (https://forum.netgate.com/post/930114)
*/
$server = "https://update6.dedyn.io/";
$url = '?hostname=' . $this->_dnsHost . '&myipv6=' . $this->_dnsIP;
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V6);
curl_setopt($ch, CURLOPT_USERPWD, $this->_dnsHost . ':' . $this->_dnsPass);
curl_setopt($ch, CURLOPT_URL, $server . $url);
break;
default:
break;
}
if ($this->_dnsService != 'ods' and $this->_dnsService != 'route53' and $this->_dnsService != 'route53-v6') {
$data = curl_exec($ch);
$this->_checkStatus($ch, $data);
@curl_close($ch);
}
}
/*
* Private Function (added 12 July 2005) [beta]
* Retrieve Update Status
*/
function _checkStatus($ch, $data)
{
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS ({$this->_dnsHost}): _checkStatus() starting.");
log_error("Dynamic DNS ({$this->_dnsHost}): Current Service: {$this->_dnsService}");
}
$successful_update = false;
if ($this->_dnsService != 'ods' and $this->_dnsService != 'route53' and $this->_dnsService != 'route53-v6' && @curl_error($ch)) {
$status = "Curl error occurred: " . curl_error($ch);
log_error($status);
$this->status = $status;
return;
}
switch ($this->_dnsService) {
case 'dhs':
break;
case 'noip':
case 'noip-free':
$noIpPrc = explode(' ', $data);
$code = $noIpPrc[0];
$ip = isset($noIpPrc[1]) ? $noIpPrc[1] : 'n/a';
switch ($code) {
case 'good':
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) DNS hostname update successful.";
$successful_update = true;
break;
case 'nochg':
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) IP address is current, no update performed.";
$successful_update = true;
break;
case 'nohost':
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Hostname supplied does not exist.";
break;
case 'badauth':
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Invalid Username or Password.";
break;
case 'badagent':
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Client disabled. Client should exit and not perform any more updates without user intervention.";
break;
case '!donate':
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Requested update feature only available to Enhanced subscribers.";
break;
case '911':
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) No-IP servers currently experiencing outages. Retry no sooner than 30 minutes.";
break;
case 'abuse':
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Account disabled due to violation of No-IP terms of service.";
break;
default:
$status = "Dynamic DNS ({$this->_dnsHost}): (Unknown Response)";
$this->_debug("Unknown Response: " . $data);
break;
}
break;
case 'easydns':
if (preg_match('/NOACCESS/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Authentication Failed: Username and/or Password was Incorrect.";
} elseif (preg_match('/NOSERVICE/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) No Service: Dynamic DNS Service has been disabled for this domain.";
} elseif (preg_match('/ILLEGAL INPUT/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Illegal Input: Self-Explanatory";
} elseif (preg_match('/TOOSOON/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Too Soon: Not Enough Time Has Elapsed Since Last Update";
} elseif (preg_match('/NOERROR/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) IP Updated Successfully!";
$successful_update = true;
} else {
$status = "Dynamic DNS ({$this->_dnsHost}): (Unknown Response)";
log_error("Dynamic DNS ({$this->_dnsHost}): PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'hn':
/* FIXME: add checks */
break;
case 'zoneedit':
if (preg_match('/799/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error 799) Update Failed!";
} elseif (preg_match('/700/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error 700) Update Failed!";
} elseif (preg_match('/200/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) IP Address Updated Successfully!";
$successful_update = true;
} elseif (preg_match('/201/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) IP Address Updated Successfully!";
$successful_update = true;
} else {
$status = "Dynamic DNS ({$this->_dnsHost}): (Unknown Response)";
log_error("Dynamic DNS ({$this->_dnsHost}): PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'dyns':
if (preg_match("/400/i", $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Bad Request - The URL was malformed. Required parameters were not provided.";
} elseif (preg_match('/402/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Update Too Soon - You have tried updating to quickly since last change.";
} elseif (preg_match('/403/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Database Error - There was a server-sided database error.";
} elseif (preg_match('/405/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Hostname Error - The hostname (" . $this->_dnsHost . ") doesn't belong to you.";
} elseif (preg_match('/200/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) IP Address Updated Successfully!";
$successful_update = true;
} else {
$status = "Dynamic DNS ({$this->_dnsHost}): (Unknown Response)";
log_error("Dynamic DNS ({$this->_dnsHost}): PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'ods':
if (preg_match("/299/i", $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) IP Address Updated Successfully!";
$successful_update = true;
} else {
$status = "Dynamic DNS ({$this->_dnsHost}): (Unknown Response)";
log_error("Dynamic DNS ({$this->_dnsHost}): PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'freedns':
if (preg_match("/No IP change detected.*skipping update/i", $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) No Change In IP Address";
$successful_update = true;
} elseif (preg_match("/Updated/i", $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) IP Address Changed Successfully!";
$successful_update = true;
} else {
$status = "Dynamic DNS ({$this->_dnsHost}): (Unknown Response)";
log_error("Dynamic DNS ({$this->_dnsHost}): PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'dnsexit':
if (preg_match("/IP not changed/i", $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) No Change In IP Address";
$successful_update = true;
} elseif (preg_match("/Success/i", $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) IP Address Changed Successfully!";
$successful_update = true;
} else {
$status = "Dynamic DNS ({$this->_dnsHost}): (Unknown Response)";
log_error("Dynamic DNS ({$this->_dnsHost}): PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'staticcling':
if (preg_match("/invalid ip/i", $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Bad Request - The IP provided was invalid.";
} elseif (preg_match('/required info missing/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Bad Request - Required parameters were not provided.";
} elseif (preg_match('/invalid characters/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Bad Request - Illegal characters in either the username or the password.";
} elseif (preg_match('/bad password/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Invalid password.";
} elseif (preg_match('/account locked/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) This account has been administratively locked.";
} elseif (preg_match('/update too frequent/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Updating too frequently.";
} elseif (preg_match('/DB error/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Error) Server side error.";
} elseif (preg_match('/success/i', $data)) {
$status = "Dynamic DNS ({$this->_dnsHost}): (Success) IP Address Updated Successfully!";
$successful_update = true;
} else {
$status = "Dynamic DNS ({$this->_dnsHost}): (Unknown Response)";
log_error("Dynamic DNS ({$this->_dnsHost}): PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'namecheap':
$tmp = str_replace("^M", "", $data);
$ncresponse = simplexml_load_string($tmp);
if (preg_match("/internal server error/i", $data)) {
$status = "Dynamic DNS: (Error) Server side error.";
} elseif (preg_match("/request is badly formed/i", $data)) {
$status = "Dynamic DNS: (Error) Badly Formed Request (check your settings).";
} elseif ((string)$ncresponse->ErrCount === "0") {
$status = "Dynamic DNS: (Success) IP Address Updated Successfully!";
$successful_update = true;
} elseif (isset($ncresponse->ErrCount) && is_numeric((string)$ncresponse->ErrCount) && (string)$ncresponse->ErrCount > 0) {
$status = "Dynamic DNS: (Error) ";
if (isset($ncresponse->errors)) {
foreach ($ncresponse->errors->children() as $err) {
$status .= (string)$err . " ";
}
}
$successful_update = true;
} else {
$status = "Dynamic DNS: (Unknown Response)";
log_error("Dynamic DNS: PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'route53':
case 'route53-v6':
$successful_update = true;
break;
case 'custom':
case 'custom-v6':
$successful_update = false;
if ($this->_dnsResultMatch == "") {
$successful_update = true;
} else {
$this->_dnsResultMatch = str_replace("%IP%", $this->_dnsIP, $this->_dnsResultMatch);
$matches = preg_split("/(?<!\\\\)\\|/", $this->_dnsResultMatch);
foreach ($matches as $match) {
$match = str_replace("\\|", "|", $match);
if (strcmp($match, trim($data, "\t\n\r")) == 0) {
$successful_update = true;
}
}
unset($matches);
}
if ($successful_update == true) {
$status = "Dynamic DNS: (Success) IP Address Updated Successfully!";
} else {
$status = "Dynamic DNS: (Error) Result did not match.";
}
break;
case 'cloudflare':
case 'cloudflare-v6':
case 'cloudflare-token':
case 'cloudflare-token-v6':
$output = json_decode($data);
if ($output->result->content === $this->_dnsIP) {
$status = "Dynamic DNS: (Success) {$this->_dnsHost} updated to {$this->_dnsIP}";
$successful_update = true;
} elseif ($output->errors[0]->code === 9103) {
$status = "Dynamic DNS ({$this->_dnsHost}): ERROR - Invalid Credentials! Don't forget to use API Key for password field with Cloudflare.";
} elseif (($output->success) && (!$output->result[0]->id)) {
$status = "Dynamic DNS ({$this->_dnsHost}): ERROR - Zone ID was not found.";
} else {
$status = "Dynamic DNS ({$this->_dnsHost}): UNKNOWN ERROR - {$output->errors[0]->message}";
log_error("Dynamic DNS ({$this->_dnsHost}): PAYLOAD: {$data}");
}
break;
case 'hetzner':
case 'hetzner-v6':
$http_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
$output = json_decode($data);
if ($output->record->value === $this->_dnsIP) {
$status = "Dynamic DNS: (Success) {$this->_dnsHost} updated to {$this->_dnsIP}";
$successful_update = true;
} elseif ($http_code == 401) {
$status = 'Dynamic DNS: (Error) Bad authentication attempt because of a wrong API Key.';
} elseif ($http_code == 403) {
$status = 'Dynamic DNS: (Error) Access to the resource is denied. Mainly due to a lack of permissions to access it!';
} else {
$status = 'Dynamic DNS: (Error) "Unknown Response"';
log_error("Dynamic DNS: HTTP Status: {$http_code} PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'digitalocean':
case 'digitalocean-v6':
$output = json_decode($data);
if ($output->domain_record->data === $this->_dnsIP) {
$status = "Dynamic DNS: (Success) Record ID {$this->_dnsUser} updated to {$this->_dnsIP}";
$successful_update = true;
} else {
$status = "Dynamic DNS Record ID ({$this->_dnsUser}): UNKNOWN ERROR";
log_error("Dynamic DNS Record ID ({$this->_dnsUser}): PAYLOAD: {$data}");
}
break;
case 'gandi-livedns':
$http_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
if ($http_code == 401) {
$status = 'Dynamic DNS: (Error) Bad authentication attempt because of a wrong API Key.';
} elseif ($http_code == 403) {
$status = 'Dynamic DNS: (Error) Access to the resource is denied. Mainly due to a lack of permissions to access it!';
} elseif ($http_code == 201) {
$status = 'Dynamic DNS: (Success) Record was created!';
$successful_update = true;
} elseif ($http_code == 200) {
$status = 'Dynamic DNS: (Success) Same record already exists. Nothing was changed!';
$successful_update = true;
} else {
$status = 'Dynamic DNS: (Error) "Unknown Response"';
log_error("Dynamic DNS: HTTP Status: {$http_code} PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'gratisdns':
if (preg_match('/Forkerte værdier/i', $data)) {
$status = "Dynamic DNS: (Error) Wrong values - Update could not be completed.";
} elseif (preg_match('/Bruger login: Bruger eksistere ikke/i', $data)) {
$status = "Dynamic DNS: (Error) Unknown username - User does not exist.";
} elseif (preg_match('/Bruger login: 1Fejl i kodeord/i', $data)) {
$status = "Dynamic DNS: (Error) Wrong password - Remember password is case sensitive.";
} elseif (preg_match('/Domæne kan IKKE administreres af bruger/i', $data)) {
$status = "Dynamic DNS: (Error) User unable to administer the selected domain.";
} elseif (preg_match('/OK/i', $data)) {
$status = "Dynamic DNS: (Success) IP Address Updated Successfully!";
$successful_update = true;
} else {
$status = "Dynamic DNS: (Unknown Response)";
log_error("Dynamic DNS: PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'duckdns':
if (preg_match('/OK/i', $data)) {
$status = "Dynamic DNS: (Success) IP Address Updated Successfully!";
$successful_update = true;
} else {
$status = "Dynamic DNS: (Unknown Response)";
log_error("Dynamic DNS: PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'dynv6':
case 'dynv6-v6':
/* API-Documentation: https://dynv6.com/docs/apis */
if (preg_match('/addresses updated/i', $data)) {
$status = "Dynamic DNS: (Success) IP Address Updated Successfully!";
$successful_update = true;
} elseif (preg_match('/addresses unchanged/i', $data)) {
$status = "Dynamic DNS: (Success) IP Address Unchanged!";
$successful_update = true;
} else {
$status = "Dynamic DNS: (Unknown Response)";
log_error("Dynamic DNS: PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case '3322':
case 'citynetwork':
case 'dnsomatic':
case 'dyndns':
case 'dyndns-custom':
case 'dyndns-static':
case 'eurodns':
case 'googledomains':
case 'he-net':
case 'he-net-tunnelbroker':
case 'he-net-v6':
case 'loopia':
case 'oray':
case 'ovh-dynhost':
case 'selfhost':
case 'strato':
if (preg_match('/notfqdn/i', $data)) {
$status = "Dynamic DNS: (Error) Not a FQDN";
} elseif (preg_match('/nochg/i', $data)) {
$status = "Dynamic DNS: (Success) No change in IP address";
$successful_update = true;
} elseif (preg_match('/good/i', $data)) {
$status = "Dynamic DNS: (Success) IP address updated successfully ({$this->_dnsIP})";
$successful_update = true;
} elseif (preg_match('/badauth/i', $data)) {
$status = "Dynamic DNS: (Error) Authentication failed";
} elseif (preg_match("/badip/i", $data)) {
$status = "Dynamic DNS: (Error) IP address provided is invalid";
} elseif (preg_match('/nohost/i', $data)) {
$status = "Dynamic DNS: (Error) Hostname does not exist or does not have dynamic DNS enabled";
} elseif (preg_match('/numhost/i', $data)) {
$status = "Dynamic DNS: (Error) You may update up to 20 hosts only";
} elseif (preg_match('/dnserr/i', $data)) {
$status = "Dynamic DNS: (Error) DNS error, stop updating for 30 minutes.";
} elseif (preg_match('/badagent/i', $data)) {
$status = "Dynamic DNS: (Error) Bad request";
} elseif (preg_match('/abuse/i', $data)) {
$status = "Dynamic DNS: (Error) Access has been blocked for abuse";
} elseif (preg_match('/911/i', $data)) {
$status = "Dynamic DNS: (Error) Server-side error or maintenance";
} elseif (preg_match('/yours/i', $data)) {
$status = "Dynamic DNS: (Error) Specified hostname does not exist under this username";
} else {
$status = "Dynamic DNS: (Unknown Response)";
log_error("Dynamic DNS: PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'regfish':
case 'regfish-v6':
if (preg_match('/\|100\|/', $data)) {
$status = 'Dynamic DNS: (Success) Update successful';
$successful_update = true;
} elseif (preg_match('/\|101\|/', $data)) {
$status = 'Dynamic DNS: (Success) Still up-to-date';
$successful_update = true;
} elseif (preg_match('/\|401\|/', $data)) {
$status = 'Dynamic DNS: (Error) Standard authentication failed';
} elseif (preg_match('/\|402\|/', $data)) {
$status = 'Dynamic DNS: (Error) Authentication failed';
} elseif (preg_match('/\|406\|/', $data)) {
$status = 'Dynamic DNS: (Error) Invalid resource record';
} elseif (preg_match('/\|407\|/', $data)) {
$status = 'Dynamic DNS: (Error) Invalid TTL range';
} elseif (preg_match('/\|408\|/', $data)) {
$status = 'Dynamic DNS: (Error) Invalid IPv4';
} elseif (preg_match('/\|409\|/', $data)) {
$status = 'Dynamic DNS: (Error) Invalid IPv6';
} elseif (preg_match('/\|410\|/', $data)) {
$status = 'Dynamic DNS: (Error) Unknown authentication type';
} elseif (preg_match('/\|412\|/', $data)) {
$status = 'Dynamic DNS: (Error) Domain format is wrong, missing trailing dot?';
} elseif (preg_match('/\|414\|/', $data)) {
$status = 'Dynamic DNS: (Error) Unexpected error';
} elseif (preg_match('/\|415\|/', $data)) {
$status = 'Dynamic DNS: (Error) Cannot update load balancer';
} else {
$status = "Dynamic DNS: (Unknown Response)";
log_error("Dynamic DNS: PAYLOAD: {$data}");
$this->_debug($data);
}
case 'linode':
case 'linode-v6':
$fqdn = trim($this->_dnsHost);
if ($this->_dnsWildcard == 'ON') {
$fqdn = "*.$fqdn";
}
$output = json_decode($data);
if ($output->target === $this->_dnsIP) {
$status = "Dynamic DNS: (Success) $fqdn updated to {$this->_dnsIP}";
$successful_update = true;
} elseif (!empty($output->errors)) {
$status = "Dynamic DNS ($fqdn): ERROR - Reason: {$output->errors[0]->reason}";
} else {
$status = "Dynamic DNS ($fqdn): UNKNOWN ERROR";
log_error("Dynamic DNS ($fqdn): PAYLOAD: {$data}");
}
break;
case 'azure':
case 'azurev6':
$http_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
if ($http_code == 401) {
$status = 'Dynamic DNS: (Error) User Authorization Failed';
} elseif ($http_code == 201) {
$status = 'Dynamic DNS: (Success) IP Address Changed Successfully!';
$successful_update = true;
} elseif ($http_code == 200) {
$status = 'Dynamic DNS: (Success) IP Address Changed Successfully!';
$successful_update = true;
} else {
$status = 'Dynamic DNS: (Error) "Unknown Response"';
log_error("Dynamic DNS: HTTP Status: {$http_code} PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'all-inkl':
case 'all-inkl-v6':
$http_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
if (preg_match('/good\s' . $this->_dnsIP . '/i', $data)) {
$status = "Dynamic DNS: (Success) IP Update Successfully!";
$successful_update = true;
} elseif ($http_code == 401) {
$status = "Dynamic DNS: (Error) Authentication failed!";
} elseif (preg_match('/bad\s\(dyndns_target_ip_syntax_incorrect\)/i', $data)) {
$status = "Dynamic DNS: (Error) IP Syntax incorrect ($this->_dnsIP)!";
} else {
$status = "Dynamic DNS ({$this->_dnsHost}): (Unknown Response)";
log_error("Dynamic DNS ({$this->_dnsHost}): PAYLOAD: {$data}");
$this->_debug($data);
}
break;
case 'godaddy':
case 'godaddy-v6':
/* See https://developer.godaddy.com/ for API documentation, not all codes are handled. */
$http_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
$successful_update = false;
if ($http_code == 200) {
$status = 'Dynamic DNS: (Success) IP Address Updated Successfully!';
$successful_update = true;
} elseif ($http_code == 401) {
$status = 'Dynamic DNS: (Error) Authentication info not sent or invalid';
} elseif ($http_code == 404) {
$status = 'Dynamic DNS: (Error) Resource not found';
} else {
$status = "Dynamic DNS: (Error) Repsonse not handled check the following: {$data}";
log_error("Dynamic DNS: (Error) HTTPS Status: {$http_code} PAYLOAD: {$data}");
}
break;
case 'desec':
case 'desec-v4-v6':
case 'desec-v6':
$http_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
/*
* HTTP Code 404 should not be possible due to dnsUser == dnsHost, a wrong hostname should cause HTTP 401 Unauthorized.
*/
if ($http_code == 401) {
$status = 'Dynamic DNS: (Error) Bad authentication attempt because of a wrong Password.';
} elseif ($http_code == 403) {
$status = 'Dynamic DNS: (Error) Access to the resource is denied. The selected hostname is not eligible for dynamic updates.';
} elseif ($http_code == 429) {
$status = 'Dynamic DNS: (Error) Rate limit reached. Please don\'t try more than one request per minute.';
} elseif ($http_code == 200 && preg_match('/good/i', $data)) {
$status = 'Dynamic DNS: (Success) IP Address Updated Successfully!';
$successful_update = true;
}
break;
default:
break;
}
if ($successful_update == true) {
/* Write WAN IP to cache file */
$wan_ip = $this->_checkIP();
if ($this->_useIPv6 == false && $wan_ip > 0) {
$currentTime = time();
log_error("Dynamic DNS: updating cache file {$this->_cacheFile}: {$wan_ip}");
@file_put_contents($this->_cacheFile, "{$wan_ip}|{$currentTime}");
} else {
@unlink($this->_cacheFile);
}
if ($this->_useIPv6 == true && $wan_ip > 0) {
$currentTime = time();
log_error("Dynamic DNS: updating cache file {$this->_cacheFile_v6}: {$wan_ip}");
@file_put_contents($this->_cacheFile_v6, "{$wan_ip}|{$currentTime}");
} else {
@unlink($this->_cacheFile_v6);
}
}
$this->status = $status;
log_error($status);
}
/*
* Private Function (added 12 July 05) [beta]
* Return Error, Set Last Error, and Die.
*/
function _error($errorNumber = '1')
{
switch ($errorNumber) {
case 0:
break;
case 2:
$error = 'Dynamic DNS: (ERROR!) No Dynamic DNS Service provider was selected.';
break;
case 3:
$error = 'Dynamic DNS: (ERROR!) No Username Provided.';
break;
case 4:
$error = 'Dynamic DNS: (ERROR!) No Password Provided.';
break;
case 5:
$error = 'Dynamic DNS: (ERROR!) No Hostname Provided.';
break;
case 6:
$error = 'Dynamic DNS: (ERROR!) The Dynamic DNS Service provided is not yet supported.';
break;
case 7:
$error = 'Dynamic DNS: (ERROR!) No Update URL Provided.';
break;
case 8:
$status = "Route 53: (Error) Invalid ZoneID";
break;
case 9:
$status = "Route 53: (Error) Invalid TTL";
break;
case 10:
$error = "Dynamic DNS ({$this->_dnsHost}): No change in my IP address and/or " . $this->_dnsMaxCacheAgeDays . " days has not passed. Not updating dynamic DNS entry.";
break;
default:
$error = "Dynamic DNS: (ERROR!) Unknown Response.";
/* FIXME: $data isn't in scope here */
/* $this->_debug($data); */
break;
}
$this->lastError = $error;
log_error($error);
}
/*
* Private Function (added 12 July 05) [beta]
* - Detect whether or not IP needs to be updated.
* | Written Specifically for pfSense (https://www.pfsense.org) may
* | work with other systems. pfSense base is FreeBSD.
*/
function _detectChange()
{
$currentTime = time();
$wan_ip = $this->_checkIP();
if ($wan_ip == 0) {
log_error("Dynamic DNS ({$this->_dnsHost}): Current WAN IP could not be determined, skipping update process.");
return false;
}
$log_error = "Dynamic DNS ({$this->_dnsHost}): Current WAN IP: {$wan_ip} ";
if ($this->_useIPv6 == true) {
if (file_exists($this->_cacheFile_v6)) {
$contents = file_get_contents($this->_cacheFile_v6);
list($cacheIP,$cacheTime) = explode('|', $contents);
$this->_debug($cacheIP . '/' . $cacheTime);
$initial = false;
$log_error .= "Cached IPv6: {$cacheIP} ";
} else {
$cacheIP = '::';
@file_put_contents($this->_cacheFile, "::|{$currentTime}");
$cacheTime = $currentTime;
$initial = true;
$log_error .= "No Cached IPv6 found.";
}
} else {
if (file_exists($this->_cacheFile)) {
$contents = file_get_contents($this->_cacheFile);
list($cacheIP,$cacheTime) = explode('|', $contents);
$this->_debug($cacheIP . '/' . $cacheTime);
$initial = false;
$log_error .= "Cached IP: {$cacheIP} ";
} else {
$cacheIP = '0.0.0.0';
@file_put_contents($this->_cacheFile, "0.0.0.0|{$currentTime}");
$cacheTime = $currentTime;
$initial = true;
$log_error .= "No Cached IP found.";
}
}
if ($this->_dnsVerboseLog) {
log_error($log_error);
}
// Convert seconds = days * hr/day * min/hr * sec/min
$maxCacheAgeSecs = $this->_dnsMaxCacheAgeDays * 24 * 60 * 60;
$needs_updating = false;
/* lets determine if the item needs updating */
if ($cacheIP != $wan_ip) {
$needs_updating = true;
$update_reason = "Dynamic DNS: cacheIP != wan_ip. Updating. ";
$update_reason .= "Cached IP: {$cacheIP} WAN IP: {$wan_ip} ";
}
if (($currentTime - $cacheTime) > $maxCacheAgeSecs) {
$needs_updating = true;
$this->_forceUpdateNeeded = true;
$update_reason = "Dynamic DNS: More than " . $this->_dnsMaxCacheAgeDays . " days. Updating. ";
$update_reason .= "{$currentTime} - {$cacheTime} > {$maxCacheAgeSecs} ";
}
if ($initial == true) {
$needs_updating = true;
$update_reason .= "Initial update. ";
}
/* finally if we need updating then store the
* new cache value and return true
*/
if ($needs_updating == true) {
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS ({$this->_dnsHost}): {$update_reason}");
}
return true;
}
return false;
}
/*
* Private Function (added 16 July 05) [beta]
* - Writes debug information to a file.
* - This function is only called when a unknown response
* - status is returned from a dynamic DNS service provider.
*/
function _debug($data)
{
$string = date('m-d-y h:i:s') . ' - (' . $this->_debugID . ') - [' . $this->_dnsService . '] - ' . $data . "\n";
$file = fopen($this->_debugFile, 'a');
fwrite($file, $string);
fclose($file);
}
function _checkIP()
{
$ip_address = get_dyndns_ip($this->_if, $this->_useIPv6 ? 6 : 4);
if (!is_ipaddr($ip_address)) {
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS ({$this->_dnsHost}): IP address could not be extracted");
}
$ip_address = 0;
} else {
if ($this->_dnsVerboseLog) {
log_error("Dynamic DNS ({$this->_dnsHost}): {$ip_address} extracted");
}
$this->_dnsIP = $ip_address;
}
return $ip_address;
}
}