3 MDU -- Mini DDNS Updater
4 Copyright (C) 2007-2009 Jonathan Zarate
6 Licensed under GNU GPL v2 or later versions.
18 #include <sys/types.h>
19 #include <sys/socket.h>
20 #include <sys/socket.h>
21 #include <netinet/in.h>
22 #include <netinet/ip.h>
23 #include <arpa/inet.h>
30 #include <tomato_version.h>
38 #define AGENT "MDU - TEST DDNS CLIENT"
40 #define AGENT "MDU - Tomato Firmware " TOMATO_MAJOR "." TOMATO_MINOR
43 #define MAX_OPTION_LENGTH 256
44 #define BLOB_SIZE (4 * 1024)
47 #define M_UNKNOWN_ERROR__D "Unknown error (%d)."
48 #define M_UNKNOWN_RESPONSE__D "Unknown response (%d)."
49 #define M_INVALID_HOST "Invalid hostname."
50 #define M_INVALID_AUTH "Invalid authentication."
51 #define M_INVALID_PARAM__D "Invalid parameter (%d)."
52 #define M_INVALID_PARAM__S "Invalid parameter (%s)."
53 #define M_TOOSOON "Update was too soon or too frequent. Please try again later."
54 #define M_ERROR_GET_IP "Error obtaining IP address."
55 #define M_SAME_IP "The IP address is the same."
56 #define M_DOWN "Server temporarily down or under maintenance."
59 int error_exitcode
= 1;
68 static void save_cookie(void);
69 static void update_noip_refresh(void);
72 static void trimamp(char *s
)
77 if ((n
> 0) && (s
[--n
] == '&')) s
[n
] = 0;
80 static const char *get_option(const char *name
)
91 if ((c
= get_option("conf")) != NULL
) {
92 if ((f
= fopen(c
, "r")) != NULL
) {
93 while (fgets(s
, sizeof(s
), f
)) {
95 if ((s
[0] == '-') && (s
[1] == '-')) p
+= 2;
96 if ((c
= strchr(p
, ' ')) != NULL
) {
98 if (p
[n
- 1] == '\n') p
[n
- 1] = 0;
101 if (n
<= 0) continue;
102 if (n
>= MAX_OPTION_LENGTH
) exit(88);
104 if ((p
= strdup(p
)) == NULL
) exit(99);
105 f_argv
[f_argc
++] = p
;
106 if (f_argc
>= (sizeof(f_argv
) / sizeof(f_argv
[0]))) break;
115 for (i
= 0; i
< f_argc
; ++i
) {
117 if ((strncmp(c
, name
, n
) == 0) && (c
[n
] == ' ')) {
122 for (i
= 0; i
< g_argc
; ++i
) {
124 if ((p
[0] == '-') && (p
[1] == '-')) {
125 if (strcmp(p
+ 2, name
) == 0) {
127 if ((i
>= g_argc
) || (strlen(g_argv
[i
]) >= MAX_OPTION_LENGTH
)) break;
136 static const char *get_option_safe(const char *name)
138 return get_option(name) ? : "";
142 static const char *get_option_required(const char *name
)
146 if ((p
= get_option(name
)) != NULL
) return p
;
147 fprintf(stderr
, "Required option --%s is missing.\n", name
);
151 static const char *get_option_or(const char *name
, const char *alt
)
153 return get_option(name
) ? : alt
;
156 static int get_option_onoff(const char *name
, int def
)
160 if ((p
= get_option(name
)) == NULL
) return def
;
161 if ((strcmp(p
, "on") == 0) || (strcmp(p
, "1") == 0)) return 1;
162 if ((strcmp(p
, "off") == 0) || (strcmp(p
, "0") == 0)) return 0;
164 fprintf(stderr
, "--%s requires the value off/on or 0/1.\n", name
);
168 static const char *md5_string(const char *value
)
170 static char buf
[(MD5_DIGEST_BYTES
+ 1) * 2];
171 unsigned char digestbuf
[MD5_DIGEST_BYTES
];
174 md5_buffer(value
, strlen(value
), digestbuf
);
175 for (i
= 0; i
< MD5_DIGEST_BYTES
; i
++)
176 sprintf(&buf
[i
* 2], "%02x", digestbuf
[i
]);
181 static void save_msg(const char *s
)
185 if ((path
= get_option("msg")) != NULL
) {
186 f_write_string(path
, s
, FW_NEWLINE
, 0);
190 static void error(const char *fmt
, ...)
196 vsnprintf(s
, sizeof(s
), fmt
, args
);
197 s
[sizeof(s
) - 1] = 0;
200 _dprintf("%s: %s\n", __FUNCTION__
, s
);
204 exit(error_exitcode
);
207 static void success_msg(const char *msg
)
211 _dprintf("%s\n", __FUNCTION__
);
218 static void success(void)
220 success_msg("Update successful.");
225 static const char *get_dump_name(void)
228 return get_option_or("dump", "/tmp/mdu.txt");
230 return get_option("dump");
234 static int _wget(int ssl
, const char *host
, int port
, const char *request
, char *buffer
, int bufsize
, char **body
)
237 struct sockaddr_in sa
;
246 _dprintf("\n*** %s\n", host
);
248 sd
= -1; // for gcc warning
250 for (trys
= 4; trys
> 0; --trys
) {
251 _dprintf("_wget trys=%d\n", trys
);
253 for (i
= 4; i
> 0; --i
) {
254 if ((he
= gethostbyname(host
)) != NULL
) {
255 if ((sd
= socket(AF_INET
, SOCK_STREAM
, 0)) < 0) {
258 memset(&sa
, 0, sizeof(sa
));
259 sa
.sin_family
= AF_INET
;
260 sa
.sin_port
= htons(port
);
261 memcpy(&sa
.sin_addr
, he
->h_addr
, sizeof(sa
.sin_addr
));
265 ia
.s_addr
= sa
.sin_addr
.s_addr
;
267 _dprintf("[%s][%d]\n", inet_ntoa(ia
), port
);
268 _dprintf("connecting...\n");
271 if (connect_timeout(sd
, (struct sockaddr
*)&sa
, sizeof(sa
), 10) == 0) {
272 _dprintf("connected\n");
282 if (i
<= 0) return -1;
286 setsockopt(sd
, SOL_SOCKET
, SO_SNDTIMEO
, &tv
, sizeof(tv
));
287 setsockopt(sd
, SOL_SOCKET
, SO_RCVTIMEO
, &tv
, sizeof(tv
));
290 mssl_init(NULL
, NULL
);
291 f
= ssl_client_fopen(sd
);
294 f
= fdopen(sd
, "r+");
297 _dprintf("error opening\n");
303 if (fwrite(request
, 1, i
, f
) != i
) {
304 _dprintf("error writing i=%d\n", i
);
309 _dprintf("sent request\n");
311 i
= fread(buffer
, 1, bufsize
, f
);
315 _dprintf("error reading i=%d\n", i
);
320 _dprintf("recvd=[%s], i=%d\n", buffer
, i
);
325 if ((c
= get_dump_name()) != NULL
) {
326 if ((f
= fopen(c
, "a")) != NULL
) {
327 fprintf(f
, "\n[%s:%d]\nREQUEST\n", host
, port
);
329 fputs("\nREPLY\n", f
);
336 if ((sscanf(buffer
, "HTTP/1.%*d %d", &i
) == 1) && (i
>= 100) && (i
<= 999)) {
337 _dprintf("HTTP/1.* i=%d\n", i
);
338 if ((p
= strstr(buffer
, "\r\n\r\n")) != NULL
) p
+= 4;
339 else if ((p
= strstr(buffer
, "\n\n")) != NULL
) p
+= 2;
343 _dprintf("body=[%s]\n", p
);
356 static int wget(int ssl
, int static_host
, const char *host
, const char *get
, const char *header
, int auth
, char **body
)
364 if (!static_host
) host
= get_option_or("server", host
);
367 if (header
) n
+= strlen(header
);
368 if (n
> (BLOB_SIZE
- 512)) return -1; // just don't go over 512 below...
371 "GET %s HTTP/1.0\r\n"
373 "User-Agent: " AGENT
"\r\n",
376 sprintf(a
, "%s:%s", get_option_required("user"), get_option_required("pass"));
377 n
= base64_encode(a
, b
, strlen(a
));
379 sprintf(blob
+ strlen(blob
), "Authorization: Basic %s\r\n", b
);
381 if ((header
) && ((n
= strlen(header
)) > 0)) {
382 strcat(blob
, header
);
383 if (header
[n
- 1] != '\n') strcat(blob
, "\r\n");
385 strcat(blob
, "\r\n");
388 _dprintf("blob=[%s]\n", blob
);
390 port
= ssl
? 443 : 80;
391 strlcpy(a
, host
, sizeof(a
));
392 if ((p
= strrchr(a
, ':')) != NULL
) {
394 if ((n
= atoi(p
+ 1)) > 0) port
= n
;
397 if ((p
= strdup(blob
)) == NULL
) return -1;
398 n
= _wget(ssl
, a
, port
, p
, blob
, BLOB_SIZE
, body
);
401 _dprintf("%s: n=%d\n", __FUNCTION__
, n
);
407 int read_tmaddr(const char *name
, long *tm
, char *addr
)
411 if (f_read_string(name
, s
, sizeof(s
)) > 0) {
412 if (sscanf(s
, "%ld,%15s", tm
, addr
) == 2) {
413 _dprintf("%s: s=%s tm=%ld addr=%s\n", __FUNCTION__
, s
, *tm
, addr
);
414 if ((tm
> 0) && (inet_addr(addr
) != -1)) return 1;
417 _dprintf("%s: unknown=%s\n", __FUNCTION__
, s
);
423 const char *get_address(int required
)
431 static char addr
[16];
434 if ((c
= get_option("addr")) != NULL
) {
437 if ((*c
!= 0) && (strlen(c
) < 20)) {
440 if ((d
= get_option("addrcache")) != NULL
) strlcpy(cache_name
, d
, sizeof(cache_name
));
441 else sprintf(cache_name
, "%s.ip", c
);
442 if (read_tmaddr(cache_name
, &et
, addr
)) {
443 if ((et
> ut
) && ((et
- ut
) <= (10 * 60))) {
444 _dprintf("%s: Using cached address %s from %s. Expires in %ld seconds.\n", __FUNCTION__
, addr
, cache_name
, et
- ut
);
449 if (strcmp(c
, "dyndns") == 0) {
450 if ((wget(0, 1, "checkip.dyndns.org:8245", "/", NULL
, 0, &body
) != 200) &&
451 (wget(0, 1, "checkip.dyndns.org", "/", NULL
, 0, &body
) != 200)) {
452 // Current IP Address: 1.2.3.4
453 error(M_ERROR_GET_IP
);
456 else if (strcmp(c
, "zoneedit") == 0 || strcmp(c
, "szoneedit") == 0) {
457 if (wget(0, 1, "dynamic.zoneedit.com", "/checkip.html", NULL
, 0, &body
) != 200) {
458 // Current IP Address: 1.2.3.4
459 error(M_ERROR_GET_IP
);
462 else if (strcmp(c
, "tzo") == 0) {
463 if ((wget(0, 1, "echo.tzo.com:21333", "/ip.shtml", NULL
, 0, &body
) != 200) &&
464 (wget(0, 1, "echo.tzo.com", "/ip.shtml", NULL
, 0, &body
) != 200)) {
466 error(M_ERROR_GET_IP
);
469 else if (strcmp(c
, "noip") == 0) {
470 if (wget(0, 1, "ip1.dynupdate.no-ip.com", "/", NULL
, 0, &body
) != 200) {
472 error(M_ERROR_GET_IP
);
475 else if (strcmp(c
, "dnsomatic") == 0) {
476 if (wget(0, 1, "myip.dnsomatic.com", "/", NULL
, 0, &body
) != 200) {
478 error(M_ERROR_GET_IP
);
482 if ((p
= strstr(body
, "Address:")) != NULL
) {
483 // dyndns, zoneedit, tzo
484 p
+= 8; // note: tzo doesn't have a space
491 while (*p
== ' ') ++p
;
494 while (((*q
>= '0') && (*q
<= '9')) || (*q
== '.')) ++q
;
497 if ((ia
.s_addr
= inet_addr(p
)) != -1) {
499 sprintf(s
, "%ld,%s", ut
+ (10 * 60), p
);
500 f_write_string(cache_name
, s
, 0, 0);
502 _dprintf("%s: saved '%s'\n", __FUNCTION__
, s
);
506 error(M_ERROR_GET_IP
);
511 return required
? get_option_required("addr") : NULL
;
514 static void append_addr_option(char *buffer
, const char *format
)
518 if ((c
= get_address(0)) != NULL
) {
519 sprintf(buffer
+ strlen(buffer
), format
, c
);
523 // -----------------------------------------------------------------------------
529 http://www.dyndns.com/developers/specs/
537 http://test:test@members.dyndns.org/nic/update?system=dyndns&hostname=test.shacknet.nu
541 hostname=yourhost.ourdomain.ext,yourhost2.dyndns.org&
544 mx=mail.exchanger.ext&
548 Host: members.dyndns.org
549 Authorization: Basic username:pass
550 User-Agent: Company - Device - Version Number
553 static void update_dua(const char *type
, int ssl
, const char *server
, const char *path
, int reqhost
)
561 sprintf(query
, "%s?", path
? path
: get_option_required("path"));
564 if (type
) sprintf(query
+ strlen(query
), "system=%s&", type
);
567 p
= reqhost
? get_option_required("host") : get_option("host");
568 if (p
) sprintf(query
+ strlen(query
), "hostname=%s&", p
);
571 if (((p
= get_option("mx")) != NULL
) && (*p
)) {
572 sprintf(query
+ strlen(query
),
576 (get_option_onoff("backmx", 0)) ? "YES" : "NO");
580 append_addr_option(query
, "myip=%s&");
582 if (get_option_onoff("wildcard", 0)) {
583 strcat(query
, "wildcard=ON");
588 r
= wget(ssl
, 0, server
? server
: get_option_required("server"), query
, NULL
, 1, &body
);
591 if ((strstr(body
, "dnserr")) || (strstr(body
, "911"))) {
596 if ((strstr(body
, "badsys")) || (strstr(body
, "numhost")) || (strstr(body
, "ILLEGAL"))) {
597 error(M_INVALID_PARAM__D
, -1);
599 if (strstr(body
, "badagent")) {
600 error(M_INVALID_PARAM__D
, -2);
602 if ((strstr(body
, "badauth")) || (strstr(body
, "NOACCESS")) || (strstr(body
, "!donator"))) {
603 error(M_INVALID_AUTH
);
605 if ((strstr(body
, "notfqdn")) || (strstr(body
, "!yours")) || strstr(body
, "nohost") ||
606 (strstr(body
, "abuse")) || strstr(body
, "NOSERVICE")) {
607 error(M_INVALID_HOST
);
609 if (strstr(body
, "TOOSOON")) {
613 if (strstr(body
, "nochg")) {
614 if ((strcmp(get_option("service"), "noip") == 0) && (refresh
)) {
615 update_noip_refresh();
621 if ((strstr(body
, "good")) || (strstr(body
, "NOERROR"))) {
626 error(M_UNKNOWN_RESPONSE__D
, -1);
629 error(M_INVALID_AUTH
);
632 error(M_UNKNOWN_ERROR__D
, r
);
639 http://namecheap.simplekb.com/kb.aspx?show=article&articleid=27&categoryid=22
643 http://dynamicdns.park-your-domain.com/update?host=host_name&domain=domain.com&password=domain_password[&ip=your_ip]
650 <?xml version="1.0"?>
652 <IP>12.123.123.12</IP>
653 <Command>SETDNSHOST</Command>
654 <Language>eng</Language>
655 <ErrCount>0</ErrCount>
656 <ResponseCount>0</ResponseCount>
657 <MinPeriod>1</MinPeriod>
658 <MaxPeriod>10</MaxPeriod>
659 <Server>Reseller9</Server>
660 <Site>Namecheap</Site>
661 <IsLockable>True</IsLockable>
662 <IsRealTimeTLD>True</IsRealTimeTLD>
663 <TimeDifference>+03.00</TimeDifference>
664 <ExecTime>0.0625</ExecTime>
666 <debug><![CDATA[]]></debug>
667 </interface-response>"
675 <?xml version="1.0"?>
677 <Command>SETDNSHOST</Command>
678 <Language>eng</Language>
679 <ErrCount>1</ErrCount>
681 <Err1>Passwords do not match</Err1>
683 <ResponseCount>1</ResponseCount>
686 <ResponseNumber>304156</ResponseNumber>
687 <ResponseString>Validation error; invalid ; password</ResponseString>
690 <MinPeriod>1</MinPeriod>
691 <MaxPeriod>10</MaxPeriod>
692 <Server>Reseller1</Server>
694 <IsLockable>True</IsLockable>
695 <IsRealTimeTLD>True</IsRealTimeTLD>
696 <TimeDifference>+03.00</TimeDifference>
697 8<ExecTime>0.0625</ExecTime>
699 <debug><![CDATA[]]></debug>
700 </interface-response>"
703 static void update_namecheap(void)
712 sprintf(query
, "/update?host=%s&domain=%s&password=%s",
713 get_option_required("host"), get_option("user") ? : get_option_required("domain"), get_option_required("pass"));
716 append_addr_option(query
, "&ip=%s");
718 r
= wget(0, 0, "dynamicdns.park-your-domain.com", query
, NULL
, 0, &body
);
720 if (strstr(body
, "<ErrCount>0<") != NULL
) {
723 if ((p
= strstr(body
, "<Err1>")) != NULL
) {
725 if ((q
= strstr(p
, "</")) != NULL
) {
727 if ((q
- p
) >= 64) p
[64] = 0;
731 error(M_UNKNOWN_RESPONSE__D
, -1);
734 error(M_UNKNOWN_ERROR__D
, r
);
741 http://www.enom.com/help/faq_dynamicdns.asp
747 ;Machine is Reseller5<br>
760 TimeDifference=+08.00
765 ;Machine is Reseller4<br>
769 Err1=Passwords do not match
771 ResponseNumber1=304156
772 ResponseString1=Validation error; invalid ; password
779 TimeDifference=+08.00
784 static void update_enom(void)
792 // http://dynamic.name-services.com/interface.asp?Command=SetDNSHost&HostName=test&Zone=test.com&Address=1.2.3.4&DomainPassword=password
795 sprintf(query
, "/interface.asp?Command=SetDNSHost&HostName=%s&Zone=%s&DomainPassword=%s",
796 get_option_required("host"), get_option("user") ? : get_option_required("domain"), get_option_required("pass"));
799 append_addr_option(query
, "&Address=%s");
801 r
= wget(0, 0, "dynamic.name-services.com", query
, NULL
, 0, &body
);
803 if (strstr(body
, "ErrCount=0") != NULL
) {
806 if ((p
= strstr(body
, "Err1=")) != NULL
) {
808 if ((q
= strchr(p
, '\n')) != NULL
) {
810 if ((q
- p
) >= 64) p
[64] = 0;
811 if ((q
= strchr(p
, '\r')) != NULL
) *q
= 0;
815 error(M_UNKNOWN_RESPONSE__D
, -1);
818 error(M_UNKNOWN_ERROR__D
, r
);
828 http://www.dnsexit.com/Direct.sv?cmd=ipClients
839 11=fail to find foo.bar.com"
842 4=Update too often. Please wait at least 8 minutes since the last update"
844 " HTTP/1.1 200 OK" <-- extra in body?
847 static void update_dnsexit(void)
854 sprintf(query
, "/RemoteUpdate.sv?action=edit&login=%s&password=%s&host=%s",
855 get_option_required("user"), get_option_required("pass"), get_option_required("host"));
858 append_addr_option(query
, "&myip=%s");
860 r
= wget(0, 0, "www.dnsexit.com", query
, NULL
, 0, &body
);
864 if ((strstr(body
, "0=Success") != NULL
) || (strstr(body
, "1=IP") != NULL
)) {
867 if ((strstr(body
, "2=Invalid") != NULL
) || (strstr(body
, "3=User") != NULL
)) {
868 error(M_INVALID_AUTH
);
870 if ((strstr(body
, "10=Host") != NULL
) || (strstr(body
, "11=fail") != NULL
)) {
871 error(M_INVALID_HOST
);
873 if (strstr(body
, "4=Update") != NULL
) {
877 error(M_UNKNOWN_RESPONSE__D
, -1);
880 error(M_UNKNOWN_ERROR__D
, r
);
892 mytest.testdomain.com:4
896 static void update_noip(void)
904 sprintf(query, "/dns?username=%s&password=%s&",
905 get_option_required("user"), get_option_required("pass"));
908 if (((c = get_option("group")) != NULL) && (*c)) {
909 sprintf(query + strlen(query), "group=%s", c);
912 sprintf(query + strlen(query), "hostname=%s", get_option_required("host"));
916 append_addr_option(query, "&ip=%s");
918 r = wget(0, 0, "dynupdate.no-ip.com", query, NULL, 0, &body);
920 if ((c = strchr(body, ':')) != NULL) {
924 case 0: // host same addr
925 while (*c == ' ') ++c;
927 error(M_UNKNOWN_RESPONSE__D, -1);
931 case 12: // group same addr
932 case 1: // host updated
933 case 11: // group updated
936 case 2: // invalid hostname
937 case 10: // invalid group
938 case 6: // account disabled
939 case 8: // disabled hostname
940 error(M_INVALID_HOST);
942 case 3: // invalid password
943 case 4: // invalid username
944 error(M_INVALID_AUTH);
949 case 7: // invalid IP supplied
950 case 99: // client banned
951 case 100: // invalid parameter
952 case 9: // redirect type
954 error(M_INVALID_PARAM__D, r);
957 error(M_UNKNOWN_RESPONSE__D, r);
966 error(M_UNKNOWN_ERROR__D, r);
975 http://www.no-ip.com/hostactive.php?host=<host>&domain=<dom>
978 static void update_noip_refresh(void)
984 strlcpy(host
, get_option_required("host"), sizeof(host
));
985 if ((domain
= strchr(host
, '.')) != NULL
) {
989 sprintf(query
, "/hostactive.php?host=%s", host
);
990 if (domain
) sprintf(query
+ strlen(query
), "&domain=%s", domain
);
992 wget(0, 1, "www.no-ip.com", query
, NULL
, 0, NULL
);
1001 http://www.ieserver.net/tools.html
1005 http://ieserver.net/cgi-bin/dip.cgi?username=XXX&domain=XXX&password=XXX&updatehost=1
1008 domain = dip.jp, fam.cx, etc.
1012 static void update_ieserver(void)
1020 sprintf(query
, "/cgi-bin/dip.cgi?username=%s&domain=%s&password=%s&updatehost=1",
1021 get_option_required("user"), get_option_required("host"), get_option_required("pass"));
1023 r
= wget(0, 0, "ieserver.net", query
, NULL
, 0, &body
);
1025 if (strstr(body
, "<title>Error") != NULL
) {
1027 // <p>yuuzaa na mata pasuwoodo (EUC-JP)
1028 if ((p
= strstr(body
, "<p>\xA5\xE6\xA1\xBC\xA5\xB6\xA1\xBC")) != NULL
) { // <p>user
1029 error(M_INVALID_AUTH
);
1032 error(M_UNKNOWN_RESPONSE__D
, -1);
1038 error(M_UNKNOWN_ERROR__D
, r
);
1046 http://www.dyns.cx/documentation/technical/protocol/v1.1.php
1053 401 testuser not authenticated"
1057 static void update_dyns(void)
1065 sprintf(query
, "/postscript011.php?username=%s&password=%s&host=%s",
1066 get_option_required("user"), get_option_required("pass"), get_option_required("host"));
1069 append_addr_option(query
, "&ip=%s");
1072 sprintf(query
+ strlen(query
), "&devel=1");
1075 r
= wget(0, 0, "www.dyns.net", query
, NULL
, 0, &body
);
1077 while ((*body
== ' ') || (*body
== '\r') || (*body
== '\n')) {
1080 switch (r
= atoi(body
)) {
1085 error(M_INVALID_PARAM__D
, r
);
1088 error(M_INVALID_AUTH
);
1094 error(M_INVALID_HOST
);
1098 error(M_UNKNOWN_RESPONSE__D
, r
);
1101 error(M_UNKNOWN_ERROR__D
, r
);
1116 <td valign="top"><font size="4">Congratulations!
1117 You've successfully signed on with TZO.<br>
1118 </font><h1 align="center"><font size="2"
1119 face="Verdana">Here's information about your
1122 <h3 align="center"><font face="Verdana"><p>TZO Name: foobartest.tzo.com<br>
1123 IP Address: 1.2.3.4<br>
1124 Expiration: 2007-01-01 1:01:01</p>
1130 Invalid hostname or password:
1134 <td valign="top"><font size="4">Congratulations!
1135 You've successfully signed on with TZO.<br>
1136 </font><h1 align="center"><font size="2"
1137 face="Verdana">Here's information about your
1140 <h3 align="center"><font face="Verdana"><p>Error=bad authentication
1147 face="Verdana">Here's information about your
1150 <h3 align="center"><font face="Verdana"><p>Error=Try again later. Please wait at least 1 minute before any additional requests.1.2.3.4
1156 static void update_tzo(void)
1164 sprintf(query
, "/webclient/signedon.html?TZOName=%s&Email=%s&TZOKey=%s",
1165 get_option_required("host"), get_option_required("user"), get_option_required("pass"));
1168 append_addr_option(query
, "&IPAddress=%s");
1170 r
= wget(0, 0, "cgi.tzo.com", query
, NULL
, 0, &body
);
1172 if ((p
= strstr(body
, "Error=")) != NULL
) {
1174 if (strncmp(p
, "bad auth", 8) == 0) {
1175 error(M_INVALID_AUTH
);
1177 if (strncmp(p
, "Try again", 9) == 0) {
1180 error(M_UNKNOWN_ERROR__D
, -1);
1185 error(M_UNKNOWN_ERROR__D
, r
);
1194 http://www.zoneedit.com/doc/dynamic.html
1200 <SUCCESS CODE="200" TEXT="Update succeeded." ZONE="test123.com" HOST="www.test123.com" IP="1.2.3.4">"
1202 "<ERROR CODE="707" TEXT="Duplicate updates for the same host/ip, adjust client settings" ZONE="testexamplesite4321.com" HOST="test.testexamplesite4321.com">"
1204 "HTTP/1.1 401 Authorization Required
1206 <title>Authentication Failed </title>"
1208 ERROR CODE="[701-799]" TEXT="Description of the error" ZONE="Zone that Failed"
1209 ERROR CODE="702" TEXT="Update failed." ZONE="%zone%"
1210 ERROR CODE="703" TEXT="one of either parameters 'zones' or 'host' are required."
1211 ERROR CODE="705" TEXT="Zone cannot be empty" ZONE="%zone%"
1212 ERROR CODE="707" TEXT="Duplicate updates for the same host/ip, adjust client settings" ZONE="%zone%"
1213 ERROR CODE="707" TEXT="Too frequent updates for the same host, adjust client settings" ZONE="%zone%"
1214 ERROR CODE="704" TEXT="Zone must be a valid 'dotted' internet name." ZONE="%zone%"
1215 ERROR CODE="701" TEXT="Zone is not set up in this account." ZONE="%zone%"
1216 ERROR CODE="708" TEXT="Login/authorization error"
1217 SUCCESS CODE="[200-201]" TEXT="Description of the success" ZONE="Zone that Succeeded"
1218 SUCCESS CODE="200" TEXT="Update succeeded." ZONE="%zone%" IP="%dnsto%"
1219 SUCCESS CODE="201" TEXT="No records need updating." ZONE="%zone%"
1222 static void update_zoneedit(int ssl
)
1230 sprintf(query
, "/auth/dynamic.html?host=%s", get_option_required("host"));
1233 append_addr_option(query
, "&dnsto=%s");
1235 r
= wget(ssl
, 0, "dynamic.zoneedit.com", query
, NULL
, 1, &body
);
1238 if (strstr(body
, "<SUCCESS CODE")) {
1241 if ((c
= strstr(body
, "<ERROR CODE=\"")) != NULL
) {
1244 case 701: // invalid "zone"
1245 error(M_INVALID_HOST
);
1247 case 707: // update is the same ip address? / too frequent updates
1248 if (strstr(c
, "Duplicate") != NULL
) success();
1249 else error(M_TOOSOON
);
1251 case 708: // authorization error
1252 error(M_INVALID_AUTH
);
1255 error(M_UNKNOWN_RESPONSE__D
, r
);
1257 error(M_UNKNOWN_RESPONSE__D
, -1);
1260 error(M_INVALID_AUTH
);
1264 error(M_UNKNOWN_ERROR__D
, r
);
1275 http://freedns.afraid.org/dynamic/update.php?XXXXXXXXXXYYYYYYYYYYYZZZZZZZ1111222
1280 ERROR: Address 1.2.3.4 has not changed."
1282 "Updated 1 host(s) foobar.mooo.com to 1.2.3.4 in 1.234 seconds"
1284 "ERROR: Missing S/key and DataID, check your update URL."
1286 "fail, make sure you own this record, and the address does not already equal 1.2.3.4"
1291 static void update_afraid(void)
1298 sprintf(query
, "/dynamic/update.php?%s", get_option_required("ahash"));
1300 r
= wget(0, 0, "freedns.afraid.org", query
, NULL
, 0, &body
);
1302 if ((strstr(body
, "ERROR") != NULL
) || (strstr(body
, "fail") != NULL
)) {
1303 if (strstr(body
, "has not changed") != NULL
) {
1306 error(M_INVALID_AUTH
);
1308 else if ((strstr(body
, "Updated") != NULL
) && (strstr(body
, "host") != NULL
)) {
1312 error(M_UNKNOWN_RESPONSE__D
, -1);
1316 error(M_UNKNOWN_ERROR__D
, r
);
1325 X-Powered-By: PHP/5.0.1-dev
1327 Content-Type: text/html; charset=utf-8
1329 Authentication given
1330 Authentication failed: Bad Username/password
1334 static void update_everydns(void)
1341 strcpy(query
, "/index.php?ver=0.1");
1342 append_addr_option(query
, "&ip=%s");
1344 p
= get_option("host");
1345 if (p
) sprintf(query
+ strlen(query
), "&domain=%s", p
);
1347 r
= wget(0, 0, "dyn.everydns.net", query
, NULL
, 1, &body
);
1349 if ((p
= strstr(body
, "Exit code:")) != NULL
) {
1356 error(M_INVALID_AUTH
);
1359 error(M_UNKNOWN_RESPONSE__D
, r
);
1363 error(M_UNKNOWN_RESPONSE__D
, -1);
1366 error(M_UNKNOWN_ERROR__D
, r
);
1373 http://www.minidns.net/areg.php?opcode=ADD&host=bar.minidns.net&username=foo&password=topsecret&ip=1.2.3.4
1377 "okay. BAR.MINIDNS.NET mapped to 1.2.3.4."
1378 "auth_fail. Incorrect username/password/hostname."
1379 "auth_fail. Host name format error."
1382 static void update_minidns(void)
1390 sprintf(query
, "/areg.php?opcode=ADD&host=%s&username=%s&password=%s",
1391 get_option_required("host"),
1392 get_option_required("user"),
1393 get_option_required("pass"));
1396 append_addr_option(query
, "&ip=%s");
1398 r
= wget(0, 0, "www.minidns.net", query
, NULL
, 1, &body
);
1400 if (strstr(body
, "okay.") != NULL
) {
1403 else if (strstr(body
, "Host name format error") != NULL
) {
1404 error(M_INVALID_HOST
);
1406 else if (strstr(body
, "auth_fail") != NULL
) {
1407 error(M_INVALID_AUTH
);
1410 error(M_UNKNOWN_RESPONSE__D
, -1);
1414 error(M_UNKNOWN_ERROR__D
, r
);
1421 http://www.editdns.net/
1427 http://DynDNS.EditDNS.net/api/dynLinux.php?p=XXX&r=XXX
1433 static void update_editdns(void)
1440 sprintf(query
, "/api/dynLinux.php?p=%s&r=%s",
1441 get_option_required("pass"), get_option_required("host"));
1443 r
= wget(0, 1, "DynDNS.EditDNS.net", query
, NULL
, 0, &body
);
1445 if (strstr(body
, "Record has been updated") != NULL
) {
1448 if (strstr(body
, "Record already exists") != NULL
) {
1451 else if (strstr(body
, "Invalid Username") != NULL
) {
1452 error(M_INVALID_AUTH
);
1454 else if (strstr(body
, "Invalid DynRecord") != NULL
) {
1455 error(M_INVALID_HOST
);
1458 error(M_UNKNOWN_RESPONSE__D
, -1);
1462 error(M_UNKNOWN_ERROR__D
, r
);
1469 HE.net IPv6 TunnelBroker
1470 https://ipv4.tunnelbroker.net/ipv4_end.php?ipv4b=$IPV4ADDR&pass=$MD5PASS&user_id=$USERID&tunnel_id=$GTUNID
1478 Your tunnel endpoint has been updated to: X.X.X.X
1479 That IPv4 endpoint is already in use.
1481 Please enter a valid IPv4 endpoint!
1482 That user_id or password is not valid
1483 Unable to find your tunnel
1486 static void update_heipv6tb(void)
1493 sprintf(query
, "/ipv4_end.php?pass=%s&user_id=%s&tunnel_id=%s",
1494 md5_string(get_option_required("pass")),
1495 get_option_required("user"),
1496 get_option_required("host"));
1499 append_addr_option(query
, "&ipv4b=%s");
1501 r
= wget(1, 0, "ipv4.tunnelbroker.net", query
, NULL
, 0, &body
);
1503 if ((strstr(body
, "has been updated") != NULL
) || (strstr(body
, "already in use") != NULL
)) {
1506 if (strstr(body
, "is not valid") != NULL
) {
1507 error(M_INVALID_AUTH
);
1509 if (strstr(body
, "find your tunnel") != NULL
) {
1510 error(M_INVALID_PARAM__S
, "Tunnel ID");
1512 if (strstr(body
, "valid IPv4 endpoint") != NULL
) {
1513 error(M_INVALID_PARAM__S
, "IPv4 endpoint");
1516 error(M_UNKNOWN_RESPONSE__D
, -1);
1519 error(M_UNKNOWN_ERROR__D
, r
);
1529 static void update_wget(void)
1543 // http://user:pass@domain:port/path?query
1544 // https://user:pass@domain:port/path?query
1546 strcpy(url
, get_option_required("url"));
1549 if (strncasecmp(url
, "https://", 8) == 0) {
1553 else if (strncasecmp(url
, "http://", 7) != 0) {
1554 error(M_INVALID_PARAM__S
, "url");
1557 if ((p
= strchr(host
, '/')) == NULL
) {
1558 error(M_INVALID_PARAM__S
, "url");
1563 if ((c
= strstr(path
, "@IP")) != NULL
) {
1565 strcpy(c
, get_address(1));
1569 if ((c
= strrchr(host
, '@')) != NULL
) {
1571 s
[base64_encode(host
, s
, c
- host
)] = 0;
1572 sprintf(he
, "Authorization: Basic %s\r\n", s
);
1580 r
= wget(https
, 1, host
, path
, header
, 0, &body
);
1583 case 302: // redirect -- assume ok
1584 if (body
&& *body
) success_msg((char *)body
);
1588 error(M_INVALID_AUTH
);
1592 error(M_UNKNOWN_ERROR__D
, r
);
1595 // -----------------------------------------------------------------------------
1599 static void check_cookie(void)
1605 if (((c
= get_option("cookie")) == NULL
) || (!read_tmaddr(c
, &tm
, addr
))) {
1606 _dprintf("%s: no cookie\n", __FUNCTION__
);
1611 if ((c
= get_address(0)) == NULL
) {
1612 _dprintf("%s: no address specified\n", __FUNCTION__
);
1615 if (strcmp(c
, addr
) != 0) {
1616 _dprintf("%s: address is different (%s != %s)\n", __FUNCTION__
, c
, addr
);
1625 if ((now
< Y2K
) || (now
< tm
)) {
1626 _dprintf("%s: time rolled back (now=%ld, tm=%ld)\n", __FUNCTION__
, now
, tm
);
1631 _dprintf("%s: addr=%s tm=%ld (relative)\n", __FUNCTION__
, addr
, tm
);
1633 if ((c
= get_option("maxtime")) != NULL
) {
1634 u
= strtol(c
, NULL
, 0);
1637 _dprintf("%s: %s expired (%ld > %ld)\n", __FUNCTION__
, addr
, tm
, u
);
1640 _dprintf("%s: maxtime=%ld tm=%ld\n", __FUNCTION__
, u
, tm
);
1647 if ((c
= get_option("mintime")) != NULL
) {
1648 u
= strtol(c
, NULL
, 0);
1649 if ((u
> 0) && (tm
< u
)) {
1650 _dprintf("%s: %s recently updated (%ld < %ld)\n", __FUNCTION__
, addr
, tm
, u
);
1663 static void save_cookie(void)
1672 _dprintf("%s: no time", __FUNCTION__
);
1676 if ((cookie
= get_option("cookie")) == NULL
) {
1677 _dprintf("%s: no cookie\n", __FUNCTION__
);
1681 if ((c
= get_address(0)) == NULL
) {
1682 _dprintf("%s: no address specified\n", __FUNCTION__
);
1686 sprintf(s
, "%ld,%s", now
, c
);
1687 f_write_string(cookie
, s
, FW_NEWLINE
, 0);
1689 _dprintf("%s: cookie=%s\n", __FUNCTION__
, s
);
1692 int main(int argc
, char *argv
[])
1699 printf("MDU\nCopyright (C) 2007-2009 Jonathan Zarate\n\n");
1700 _dprintf("DEBUG\n");
1702 if ((blob
= malloc(BLOB_SIZE
)) == NULL
) {
1706 mkdir("/var/lib/mdu", 0700);
1707 chdir("/var/lib/mdu");
1709 if ((p = get_dump_name()) != NULL) {
1715 p
= get_option_required("service");
1716 if (strcmp(p
, "dua") == 0) {
1717 update_dua("dyndns", 0, NULL
, NULL
, 1);
1719 else if (strcmp(p
, "duas") == 0) {
1720 update_dua("dyndns", 1, NULL
, NULL
, 1);
1722 else if (strcmp(p
, "dyndns") == 0) {
1723 // test ok 9/14 -- zzz
1724 update_dua("dyndns", 0, "members.dyndns.org", "/nic/update", 1);
1726 else if (strcmp(p
, "dyndns-static") == 0) {
1727 // test ok 9/14 -- zzz
1728 update_dua("statdns", 0, "members.dyndns.org", "/nic/update", 1);
1730 else if (strcmp(p
, "dyndns-custom") == 0) {
1731 // test ok 9/14 -- zzz
1732 update_dua("custom", 0, "members.dyndns.org", "/nic/update", 1);
1734 else if (strcmp(p
, "sdyndns") == 0) {
1735 update_dua("dyndns", 1, "members.dyndns.org", "/nic/update", 1);
1737 else if (strcmp(p
, "sdyndns-static") == 0) {
1738 update_dua("statdns", 1, "members.dyndns.org", "/nic/update", 1);
1740 else if (strcmp(p
, "sdyndns-custom") == 0) {
1741 update_dua("custom", 1, "members.dyndns.org", "/nic/update", 1);
1743 else if (strcmp(p
, "easydns") == 0) {
1744 // no account, test output ok, test 401 error parse ok 9/15 -- zzz
1745 update_dua(NULL
, 0, "members.easydns.com", "/dyn/dyndns.php", 1);
1747 else if (strcmp(p
, "seasydns") == 0) {
1748 update_dua(NULL
, 1, "members.easydns.com", "/dyn/dyndns.php", 1);
1750 else if (strcmp(p
, "3322") == 0) {
1751 // no account, test output ok, test 401 error parse ok 9/16 -- zzz
1752 update_dua(NULL
, 0, "members.3322.org", "/dyndns/update", 1);
1754 else if (strcmp(p
, "3322-static") == 0) {
1755 // no account, test output ok, test 401 error parse ok 9/16 -- zzz
1756 update_dua("statdns", 0, "members.3322.org", "/dyndns/update", 1);
1758 else if (strcmp(p
, "opendns") == 0) {
1759 // test ok 9/15 -- zzz
1760 update_dua(NULL
, 1, "updates.opendns.com", "/nic/update", 0);
1762 else if (strcmp(p
, "dnsomatic") == 0) {
1763 // test ok 12/02 -- zzz
1764 update_dua(NULL
, 1, "updates.dnsomatic.com", "/nic/update", 0);
1766 else if (strcmp(p
, "noip") == 0) {
1767 update_dua(NULL
, 0, "dynupdate.no-ip.com", "/nic/update", 1);
1770 else if (strcmp(p
, "namecheap") == 0) {
1771 // test ok 9/14 -- zzz
1774 else if (strcmp(p
, "enom") == 0) {
1775 // no account, test output ok, 12/03 -- zzz
1778 else if (strcmp(p
, "dnsexit") == 0) {
1779 // test ok 9/14 -- zzz
1782 else if (strcmp(p
, "ieserver") == 0) {
1783 // test ok 9/14 -- zzz
1786 else if (strcmp(p
, "dyns") == 0) {
1787 // no account, test output ok, test 401 error parse ok 9/15 -- zzz
1790 else if (strcmp(p
, "tzo") == 0) {
1791 // test ok 9/15 -- zzz
1794 else if (strcmp(p
, "zoneedit") == 0) {
1795 // test ok 9/16 -- zzz
1798 else if (strcmp(p
, "szoneedit") == 0) {
1801 else if (strcmp(p
, "afraid") == 0) {
1802 // test ok 9/16 -- zzz
1805 else if (strcmp(p
, "everydns") == 0) {
1809 else if (strcmp(p
, "editdns") == 0) {
1812 else if (strcmp(p
, "minidns") == 0) {
1815 else if (strcmp(p
, "heipv6tb") == 0) {
1818 else if ((strcmp(p
, "wget") == 0) || (strcmp(p
, "custom") == 0)) {
1819 // test ok 9/15 -- zzz
1823 error("Unknown service");