source: trunk/dns-rpc.cgi@ 498

Last change on this file since 498 was 498, checked in by Kris Deugau, 12 years ago

/trunk

Add a line in dns-rpc.cgi's addRec() and updateRec() to allow RPC
callers to use the text labels for DNS types instead of having to
know the underlying integer value. Should work for all types
including the pseudotypes.

  • Property svn:executable set to *
  • Property svn:keywords set to Date Rev Author Id
File size: 21.2 KB
RevLine 
[216]1#!/usr/bin/perl -w -T
[262]2# XMLRPC interface to manipulate most DNS DB entities
3##
[200]4# $Id: dns-rpc.cgi 498 2013-05-06 02:41:31Z kdeugau $
[496]5# Copyright 2012,2013 Kris Deugau <kdeugau@deepnet.cx>
[262]6#
7# This program is free software: you can redistribute it and/or modify
8# it under the terms of the GNU General Public License as published by
9# the Free Software Foundation, either version 3 of the License, or
10# (at your option) any later version.
11#
12# This program is distributed in the hope that it will be useful,
13# but WITHOUT ANY WARRANTY; without even the implied warranty of
14# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15# GNU General Public License for more details.
16#
17# You should have received a copy of the GNU General Public License
18# along with this program. If not, see <http://www.gnu.org/licenses/>.
19##
[119]20
21use strict;
22use warnings;
[216]23
24# don't remove! required for GNU/FHS-ish install from tarball
25use lib '.'; ##uselib##
[490]26use DNSDB;
[216]27
[490]28use FCGI;
[119]29#use Frontier::RPC2;
30use Frontier::Responder;
31
32## We need to handle a couple of things globally, rather than pasting the same bit into *every* sub.
33## So, let's subclass Frontier::RPC2 + Frontier::Responder, so we can override the single sub in each
34## that needs kicking
35#### hmm. put this in a separate file?
36#package DNSDB::RPC;
37#our @ISA = ("Frontier::RPC2", "Frontier::Responder");
38#package main;
39
[468]40my $dnsdb = DNSDB->new();
[191]41
[119]42my $methods = {
43 'dnsdb.addDomain' => \&addDomain,
[401]44 'dnsdb.delZone' => \&delZone,
[405]45 'dnsdb.addRDNS' => \&addRDNS,
[121]46 'dnsdb.addGroup' => \&addGroup,
47 'dnsdb.delGroup' => \&delGroup,
48 'dnsdb.addUser' => \&addUser,
49 'dnsdb.updateUser' => \&updateUser,
50 'dnsdb.delUser' => \&delUser,
[447]51 'dnsdb.getLocDropdown' => \&getLocDropdown,
[121]52 'dnsdb.getSOA' => \&getSOA,
[123]53 'dnsdb.getRecLine' => \&getRecLine,
[495]54 'dnsdb.getRecList' => \&getRecList,
[123]55 'dnsdb.getRecCount' => \&getRecCount,
56 'dnsdb.addRec' => \&addRec,
[405]57 'dnsdb.updateRec' => \&updateRec,
[453]58 'dnsdb.addOrUpdateRevRec' => \&addOrUpdateRevRec,
[123]59 'dnsdb.delRec' => \&delRec,
[459]60 'dnsdb.delByCIDR' => \&delByCIDR,
[452]61#sub getLogCount {}
62#sub getLogEntries {}
63 'dnsdb.getRevPattern' => \&getRevPattern,
[405]64 'dnsdb.zoneStatus' => \&zoneStatus,
[452]65 'dnsdb.getZonesByCIDR' => \&getZonesByCIDR,
[121]66
[119]67 'dnsdb.getMethods' => \&get_method_list
68};
69
[490]70my $reqcnt = 0;
71
72while (FCGI::accept >= 0) {
73 my $res = Frontier::Responder->new(
[119]74 methods => $methods
75 );
76
[490]77 # "Can't do that" errors
78 if (!$dnsdb) {
79 print "Content-type: text/xml\n\n".$res->{_decode}->encode_fault(5, $dnsdb->err);
80 } else {
81 print $res->answer;
82 }
83 last if $reqcnt++ > $dnsdb->{maxfcgi};
84} # while FCGI::accept
[119]85
86
87exit;
88
89##
90## Subs below here
91##
92
[490]93# Check RPC ACL
[401]94sub _aclcheck {
95 my $subsys = shift;
[486]96 return 1 if grep /$ENV{REMOTE_ADDR}/, @{$dnsdb->{rpcacl}{$subsys}};
[401]97 return 0;
98}
99
[405]100# Let's see if we can factor these out of the RPC method subs
101sub _commoncheck {
102 my $argref = shift;
103 my $needslog = shift;
104
105 die "Missing remote system name\n" if !$argref->{rpcsystem};
106 die "Access denied\n" if !_aclcheck($argref->{rpcsystem});
107 if ($needslog) {
108 die "Missing remote username\n" if !$argref->{rpcuser};
109 die "Couldn't set userdata for logging\n"
[486]110 unless $dnsdb->initRPC(username => $argref->{rpcuser}, rpcsys => $argref->{rpcsystem},
111 fullname => ($argref->{fullname} ? $argref->{fullname} : $argref->{rpcuser}) );
[405]112 }
113}
114
[453]115# set location to the zone's default location if none is specified
116sub _loccheck {
117 my $argref = shift;
118 if (!$argref->{location} && $argref->{defrec} eq 'n') {
[477]119 $argref->{location} = $dnsdb->getZoneLocation($argref->{revrec}, $argref->{parent_id});
[453]120 }
121}
122
123# set ttl to zone defailt minttl if none is specified
124sub _ttlcheck {
125 my $argref = shift;
126 if (!$argref->{ttl}) {
[486]127 my $tmp = $dnsdb->getSOA($argref->{defrec}, $argref->{revrec}, $argref->{parent_id});
[453]128 $argref->{ttl} = $tmp->{minttl};
129 }
130}
131
[119]132#sub connectDB {
133#sub finish {
134#sub initGlobals {
135#sub initPermissions {
136#sub getPermissions {
137#sub changePermissions {
138#sub comparePermissions {
139#sub changeGroup {
140#sub _log {
141
142sub addDomain {
143 my %args = @_;
144
[405]145 _commoncheck(\%args, 'y');
[119]146
[477]147 my ($code, $msg) = $dnsdb->addDomain($args{domain}, $args{group}, $args{state});
[119]148 die $msg if $code eq 'FAIL';
149 return $msg; # domain ID
150}
151
[401]152sub delZone {
[119]153 my %args = @_;
154
[405]155 _commoncheck(\%args, 'y');
156 die "Need forward/reverse zone flag\n" if !$args{revrec};
[119]157
[121]158 my ($code,$msg);
[405]159 # Let's be nice; delete based on zone id OR zone name. Saves an RPC call round-trip, maybe.
160 if ($args{zone} =~ /^\d+$/) {
[477]161 ($code,$msg) = $dnsdb->delZone($args{zone}, $args{revrec});
[119]162 } else {
[405]163 my $zoneid;
[477]164 $zoneid = $dnsdb->domainID($args{zone}) if $args{revrec} eq 'n';
165 $zoneid = $dnsdb->revID($args{zone}) if $args{revrec} eq 'y';
166 die "Can't find zone: $dnsdb->errstr\n" if !$zoneid;
167 ($code,$msg) = $dnsdb->delZone($zoneid, $args{revrec});
[119]168 }
169 die $msg if $code eq 'FAIL';
[405]170 return $msg;
[119]171}
172
[405]173#sub domainName {}
174#sub revName {}
175#sub domainID {}
176#sub revID {}
[119]177
[405]178sub addRDNS {
179 my %args = @_;
180
181 _commoncheck(\%args, 'y');
182
[477]183 my ($code, $msg) = $dnsdb->addRDNS($args{revzone}, $args{revpatt}, $args{group}, $args{state}, $args{defloc});
[447]184 die "$msg\n" if $code eq 'FAIL';
[405]185 return $msg; # domain ID
186}
187
188#sub getZoneCount {}
189#sub getZoneList {}
190#sub getZoneLocation {}
191
[119]192sub addGroup {
193 my %args = @_;
194
[405]195 _commoncheck(\%args, 'y');
[407]196 die "Missing new group name\n" if !$args{groupname};
197 die "Missing parent group ID\n" if !$args{parent_id};
[119]198
[407]199# not sure how to usefully represent permissions via RPC. :/
[121]200# not to mention, permissions are checked at the UI layer, not the DB layer.
[119]201 my $perms = {domain_edit => 1, domain_create => 1, domain_delete => 1,
202 record_edit => 1, record_create => 1, record_delete => 1
203 };
204## optional $inhert arg?
[476]205 my ($code,$msg) = $dnsdb->addGroup($args{groupname}, $args{parent_id}, $perms);
[119]206 die $msg if $code eq 'FAIL';
207 return $msg;
208}
209
210sub delGroup {
211 my %args = @_;
212
[405]213 _commoncheck(\%args, 'y');
[407]214 die "Missing group ID or name to remove\n" if !$args{group};
[119]215
[121]216 my ($code,$msg);
[119]217 # Let's be nice; delete based on groupid OR group name. Saves an RPC call round-trip, maybe.
218 if ($args{group} =~ /^\d+$/) {
[476]219 ($code,$msg) = $dnsdb->delGroup($args{group});
[119]220 } else {
[486]221 my $grpid = $dnsdb->groupID($args{group});
[407]222 die "Can't find group\n" if !$grpid;
[476]223 ($code,$msg) = $dnsdb->delGroup($grpid);
[119]224 }
225 die $msg if $code eq 'FAIL';
[407]226 return $msg;
[119]227}
228
[405]229#sub getChildren {}
230#sub groupName {}
231#sub getGroupCount {}
232#sub getGroupList {}
233#sub groupID {}
[119]234
235sub addUser {
236 my %args = @_;
237
[405]238 _commoncheck(\%args, 'y');
[119]239
[409]240# not sure how to usefully represent permissions via RPC. :/
[121]241# not to mention, permissions are checked at the UI layer, not the DB layer.
[119]242 # bend and twist; get those arguments in in the right order!
243 $args{type} = 'u' if !$args{type};
244 $args{permstring} = 'i' if !defined($args{permstring});
245 my @userargs = ($args{username}, $args{group}, $args{pass}, $args{state}, $args{type}, $args{permstring});
246 for my $argname ('fname','lname','phone') {
247 last if !$args{$argname};
248 push @userargs, $args{$argname};
249 }
[479]250 my ($code,$msg) = $dnsdb->addUser(@userargs);
[119]251 die $msg if $code eq 'FAIL';
252 return $msg;
253}
254
[405]255#sub getUserCount {}
256#sub getUserList {}
257#sub getUserDropdown {}
258#sub checkUser {}
[119]259
260sub updateUser {
261 my %args = @_;
262
[405]263 _commoncheck(\%args, 'y');
[119]264
[401]265 die "Missing UID\n" if !$args{uid};
[121]266
[119]267 # bend and twist; get those arguments in in the right order!
[411]268 $args{type} = 'u' if !$args{type};
[119]269 my @userargs = ($args{uid}, $args{username}, $args{group}, $args{pass}, $args{state}, $args{type});
270 for my $argname ('fname','lname','phone') {
271 last if !$args{$argname};
272 push @userargs, $args{$argname};
273 }
274##fixme: also underlying in DNSDB::updateUser(): no way to just update this or that attribute;
275# have to pass them all in to be overwritten
[479]276 my ($code,$msg) = $dnsdb->updateUser(@userargs);
[119]277 die $msg if $code eq 'FAIL';
[412]278 return $msg;
[119]279}
280
281sub delUser {
282 my %args = @_;
283
[405]284 _commoncheck(\%args, 'y');
[119]285
[401]286 die "Missing UID\n" if !$args{uid};
[479]287 my ($code,$msg) = $dnsdb->delUser($args{uid});
[119]288 die $msg if $code eq 'FAIL';
[412]289 return $msg;
[119]290}
291
[405]292#sub userFullName {}
293#sub userStatus {}
294#sub getUserData {}
[119]295
[405]296#sub addLoc {}
297#sub updateLoc {}
298#sub delLoc {}
299#sub getLoc {}
300#sub getLocCount {}
301#sub getLocList {}
302
[447]303sub getLocDropdown {
304 my %args = @_;
305
306 _commoncheck(\%args);
307 $args{defloc} = '' if !$args{defloc};
308
[480]309 my $ret = $dnsdb->getLocDropdown($args{group}, $args{defloc});
[447]310 return $ret;
311}
312
[119]313sub getSOA {
314 my %args = @_;
315
[405]316 _commoncheck(\%args);
[121]317
[481]318 my $ret = $dnsdb->getSOA($args{defrec}, $args{revrec}, $args{id});
[413]319 if (!$ret) {
320 if ($args{defrec} eq 'y') {
[401]321 die "No default SOA record in group\n";
[121]322 } else {
[413]323 die "No SOA record in zone\n";
[121]324 }
325 }
[413]326 return $ret;
[119]327}
328
[405]329#sub updateSOA {}
330
[119]331sub getRecLine {
332 my %args = @_;
333
[405]334 _commoncheck(\%args);
[123]335
[481]336 my $ret = $dnsdb->getRecLine($args{defrec}, $args{revrec}, $args{id});
[123]337
[481]338 die $dnsdb->errstr if !$ret;
[123]339
340 return $ret;
[119]341}
342
[495]343sub getRecList {
[119]344 my %args = @_;
345
[405]346 _commoncheck(\%args);
[123]347
[405]348 # set some optional args
[123]349 $args{nrecs} = 'all' if !$args{nrecs};
350 $args{nstart} = 0 if !$args{nstart};
351## for order, need to map input to column names
352 $args{order} = 'host' if !$args{order};
353 $args{direction} = 'ASC' if !$args{direction};
354
[495]355 my $ret = $dnsdb->getRecList(defrec => $args{defrec}, revrec => $args{revrec}, id => $args{id},
[401]356 offset => $args{offset}, sortby => $args{sortby}, sortorder => $args{sortorder},
[481]357 filter => $args{filter});
[123]358
[481]359 die $dnsdb->errstr if !$ret;
[123]360
361 return $ret;
[119]362}
363
[123]364sub getRecCount {
365 my %args = @_;
[119]366
[405]367 _commoncheck(\%args);
[123]368
[405]369 # set some optional args
370 $args{nrecs} = 'all' if !$args{nrecs};
371 $args{nstart} = 0 if !$args{nstart};
372## for order, need to map input to column names
373 $args{order} = 'host' if !$args{order};
374 $args{direction} = 'ASC' if !$args{direction};
375
[481]376 my $ret = $dnsdb->getRecCount($args{defrec}, $args{revrec}, $args{id}, $args{filter});
[405]377
[481]378 die $dnsdb->errstr if !$ret;
[405]379
380 return $ret;
[123]381}
382
[119]383sub addRec {
384 my %args = @_;
385
[405]386 _commoncheck(\%args, 'y');
[123]387
[453]388 _loccheck(\%args);
389 _ttlcheck(\%args);
[123]390
[498]391 # allow passing text types rather than DNS integer IDs
392 $args{type} = $DNSDB::reverse_typemap{$args{type}} if $args{type} !~ /^d+$/;
393
[481]394 my @recargs = ($args{defrec}, $args{revrec}, $args{parent_id},
[426]395 \$args{name}, \$args{type}, \$args{address}, $args{ttl}, $args{location});
396 if ($args{type} == $DNSDB::reverse_typemap{MX} or $args{type} == $DNSDB::reverse_typemap{SRV}) {
397 push @recargs, $args{distance};
398 if ($args{type} == $DNSDB::reverse_typemap{SRV}) {
399 push @recargs, $args{weight};
400 push @recargs, $args{port};
401 }
402 }
403
[481]404 my ($code, $msg) = $dnsdb->addRec(@recargs);
[426]405
[123]406 die $msg if $code eq 'FAIL';
[426]407 return $msg;
[119]408}
409
410sub updateRec {
411 my %args = @_;
412
[405]413 _commoncheck(\%args, 'y');
[123]414
[452]415 # get old line, so we can update only the bits that the caller passed to change
416 # note we subbed address for val since it's a little more caller-friendly
[481]417 my $oldrec = $dnsdb->getRecLine($args{defrec}, $args{revrec}, $args{id});
[452]418 foreach my $field (qw(name type address ttl location distance weight port)) {
419 $args{$field} = $oldrec->{$field} if !$args{$field} && defined($oldrec->{$field});
420 }
421
[498]422 # allow passing text types rather than DNS integer IDs
423 $args{type} = $DNSDB::reverse_typemap{$args{type}} if $args{type} !~ /^d+$/;
424
[405]425 # note dist, weight, port are not required on all types; will be ignored if not needed.
[426]426 # parent_id is the "primary" zone we're updating; necessary for forward/reverse voodoo
[481]427 my ($code, $msg) = $dnsdb->updateRec($args{defrec}, $args{revrec}, $args{id}, $args{parent_id},
[426]428 \$args{name}, \$args{type}, \$args{address}, $args{ttl}, $args{location},
429 $args{distance}, $args{weight}, $args{port});
[123]430
431 die $msg if $code eq 'FAIL';
[426]432 return $msg;
[119]433}
434
[453]435# Takes a passed CIDR block and DNS pattern; adds a new record or updates the record(s) affected
436sub addOrUpdateRevRec {
437 my %args = @_;
438
439 _commoncheck(\%args, 'y');
[459]440 my $cidr = new NetAddr::IP $args{cidr};
[453]441
[477]442 my $zonelist = $dnsdb->getZonesByCIDR(%args);
[453]443 if (scalar(@$zonelist) == 0) {
444 # enhh.... WTF?
445 } elsif (scalar(@$zonelist) == 1) {
446 # check if the single zone returned is bigger than the CIDR. if so, we can just add a record
447 my $zone = new NetAddr::IP $zonelist->[0]->{revnet};
[459]448 if ($zone->contains($cidr)) {
[454]449 # We need to strip the CIDR mask on IPv4 /32 assignments, or we just add a new record all the time.
[459]450 my $filt = ($cidr->{isv6} || $cidr->masklen != 32 ? "$cidr" : $cidr->addr);
[495]451 my $reclist = $dnsdb->getRecList(defrec => 'n', revrec => 'y',
[454]452 id => $zonelist->[0]->{rdns_id}, filter => $filt);
[453]453 if (scalar(@$reclist) == 0) {
454 # Aren't Magic Numbers Fun? See pseudotype list in dnsadmin.
[460]455 my $type = ($cidr->{isv6} ? 65284 : ($cidr->masklen == 32 ? 65280 : 65283) );
[453]456 addRec(defrec =>'n', revrec => 'y', parent_id => $zonelist->[0]->{rdns_id}, type => $type,
[459]457 address => "$cidr", %args);
[453]458 } else {
[459]459 my $flag = 0;
[453]460 foreach my $rec (@$reclist) {
[454]461 # pure PTR plus composite types
462 next unless $rec->{type} == 12 || $rec->{type} == 65280 || $rec->{type} == 65281
463 || $rec->{type} == 65282 || $rec->{type} == 65283 || $rec->{type} == 65284;
464 next unless $rec->{val} eq $filt; # make sure we really update the record we want to update.
[481]465 $dnsdb->updateRec(defrec =>'n', revrec => 'y', id => $rec->{record_id},
[453]466 parent_id => $zonelist->[0]->{rdns_id}, %args);
[459]467 $flag = 1;
[453]468 last; # only do one record.
469 }
[459]470 unless ($flag) {
471 # Nothing was updated, so we didn't really have a match. Add as per @$reclist==0
472 # Aren't Magic Numbers Fun? See pseudotype list in dnsadmin.
473 my $type = ($cidr->{isv6} ? 65282 : ($cidr->masklen == 32 ? 65280 : 65283) );
[481]474 $dnsdb->addRec(defrec =>'n', revrec => 'y', parent_id => $zonelist->[0]->{rdns_id}, type => $type,
[459]475 address => "$cidr", %args);
476 }
[453]477 }
478 } else {
479 # ebbeh? CIDR is only partly represented in DNS. This needs manual intervention.
480 } # done single-zone-contains-$cidr
481 } else {
482 # Overlapping reverse zones shouldn't be possible, so if we're here we've got a CIDR
483 # that spans multiple reverse zones (eg, /23 CIDR -> 2 /24 rzones)
484 foreach my $zdata (@$zonelist) {
[495]485 my $reclist = $dnsdb->getRecList(defrec => 'n', revrec => 'y',
[453]486 id => $zdata->{rdns_id}, filter => $zdata->{revnet});
487 if (scalar(@$reclist) == 0) {
488 my $type = ($args{cidr}->{isv6} ? 65282 : ($args{cidr}->masklen == 32 ? 65280 : 65283) );
[481]489 $dnsdb->addRec(defrec =>'n', revrec => 'y', parent_id => $zdata->{rdns_id}, type => $type,
[453]490 address => "$args{cidr}", %args);
491 } else {
492 foreach my $rec (@$reclist) {
[454]493 # only the composite and/or template types; pure PTR or nontemplate composite
494 # types are nominally impossible here.
[453]495 next unless $rec->{type} == 65282 || $rec->{type} == 65283 || $rec->{type} == 65284;
[481]496 $dnsdb->updateRec(defrec =>'n', revrec => 'y', id => $rec->{record_id},
[453]497 parent_id => $zdata->{rdns_id}, %args);
498 last; # only do one record.
499 }
500 }
501 } # iterate zones within $cidr
502 } # done $cidr-contains-zones
503}
504
[119]505sub delRec {
506 my %args = @_;
507
[405]508 _commoncheck(\%args, 'y');
[123]509
[481]510 my ($code, $msg) = $dnsdb->delRec($args{defrec}, $args{recrev}, $args{id});
[123]511
512 die $msg if $code eq 'FAIL';
[426]513 return $msg;
[119]514}
515
[459]516sub delByCIDR {
517 my %args = @_;
518
519 _commoncheck(\%args, 'y');
520
521 # much like addOrUpdateRevRec()
[477]522 my $zonelist = $dnsdb->getZonesByCIDR(%args);
[459]523 my $cidr = new NetAddr::IP $args{cidr};
524
525 if (scalar(@$zonelist) == 0) {
526 # enhh.... WTF?
527 } elsif (scalar(@$zonelist) == 1) {
528
529 # check if the single zone returned is bigger than the CIDR
530 my $zone = new NetAddr::IP $zonelist->[0]->{revnet};
531 if ($zone->contains($cidr)) {
532
533 if ($args{delsubs}) {
534 # Delete ALL EVARYTHING!!one11!! in $args{cidr}
[495]535 my $reclist = $dnsdb->getRecList(defrec => 'n', revrec => 'y', id => $zonelist->[0]->{rdns_id});
[459]536 foreach my $rec (@$reclist) {
537 my $reccidr = new NetAddr::IP $rec->{val};
538 next unless $cidr->contains($reccidr);
[460]539 next unless $rec->{type} == 12 || $rec->{type} == 65280 || $rec->{type} == 65281 ||
540 $rec->{type} == 65282 || $rec->{type} == 65283 ||$rec->{type} == 65284;
[459]541 ##fixme: multiple records, wanna wax'em all, how to report errors?
542 if ($args{delforward} ||
543 $rec->{type} == 12 || $rec->{type} == 65282 ||
544 $rec->{type} == 65283 || $rec->{type} == 65284) {
[481]545 my ($code,$msg) = $dnsdb->delRec('n', 'y', $rec->{record_id});
[459]546 } else {
[481]547 my $ret = $dnsdb->downconvert($rec->{record_id}, $DNSDB::reverse_typemap{A});
[459]548 }
549 }
[460]550 if ($args{parpatt} && $zone == $cidr) {
551 # Edge case; we've just gone and axed all the records in the reverse zone.
552 # Re-add one to match the parent if we've been given a pattern to use.
[481]553 $dnsdb->addRec(defrec =>'n', revrec => 'y', parent_id => $zonelist->[0]->{rdns_id},
[460]554 type => ($zone->{isv6} ? 65284 : 65283), address => "$cidr", %args);
555 }
[459]556
557 } else {
558 # Selectively delete only exact matches on $args{cidr}
559
560 # We need to strip the CIDR mask on IPv4 /32 assignments, or we can't find single-IP records
561 my $filt = ($cidr->{isv6} || $cidr->masklen != 32 ? "$cidr" : $cidr->addr);
[495]562 my $reclist = $dnsdb->getRecList(defrec => 'n', revrec => 'y',
[459]563 id => $zonelist->[0]->{rdns_id}, filter => $filt, sortby => 'val', sortorder => 'DESC');
564 foreach my $rec (@$reclist) {
565 my $reccidr = new NetAddr::IP $rec->{val};
566 next unless $cidr == $reccidr;
[460]567 next unless $rec->{type} == 12 || $rec->{type} == 65280 || $rec->{type} == 65281 ||
568 $rec->{type} == 65282 || $rec->{type} == 65283 ||$rec->{type} == 65284;
[459]569 if ($args{delforward} || $rec->{type} == 12) {
[481]570 my ($code,$msg) = $dnsdb->delRec('n', 'y', $rec->{record_id});
[459]571 die $msg if $code eq 'FAIL';
572 return $msg;
573 } else {
[481]574 my $ret = $dnsdb->downconvert($rec->{record_id}, $DNSDB::reverse_typemap{A});
575 die $dnsdb->errstr if !$ret;
[459]576 return "A+PTR for $args{cidr} split and PTR removed";
577 }
578 } # foreach @$reclist
579 }
580
581 } else { # $cidr > $zone but we only have one zone
582 # ebbeh? CIDR is only partly represented in DNS. This needs manual intervention.
583 return "Warning: $args{cidr} is only partly represented in DNS. Check and remove DNS records manually.";
584 } # done single-zone-contains-$cidr
585
586 } else { # multiple zones nominally "contain" $cidr
587 # Overlapping reverse zones shouldn't be possible, so if we're here we've got a CIDR
588 # that spans multiple reverse zones (eg, /23 CIDR -> 2 /24 rzones)
589 foreach my $zdata (@$zonelist) {
[495]590 my $reclist = $dnsdb->getRecList(defrec => 'n', revrec => 'y', id => $zdata->{rdns_id});
[459]591 if (scalar(@$reclist) == 0) {
[460]592# nothing to do? or do we (re)add a record based on the parent?
593# yes, yes we do, past the close of the else
594# my $type = ($args{cidr}->{isv6} ? 65282 : ($args{cidr}->masklen == 32 ? 65280 : 65283) );
595# addRec(defrec =>'n', revrec => 'y', parent_id => $zdata->{rdns_id}, type => $type,
596# address => "$args{cidr}", %args);
[459]597 } else {
598 foreach my $rec (@$reclist) {
[460]599 next unless $rec->{type} == 12 || $rec->{type} == 65280 || $rec->{type} == 65281 ||
600 $rec->{type} == 65282 || $rec->{type} == 65283 || $rec->{type} == 65284;
601 # Template types are only useful when attached to a reverse zone.
602##fixme ..... or ARE THEY?
603 if ($args{delforward} ||
604 $rec->{type} == 12 || $rec->{type} == 65282 ||
605 $rec->{type} == 65283 || $rec->{type} == 65284) {
[481]606 my ($code,$msg) = $dnsdb->delRec('n', 'y', $rec->{record_id});
[460]607 } else {
[481]608 my $ret = $dnsdb->downconvert($rec->{record_id}, $DNSDB::reverse_typemap{A});
[460]609 }
[459]610 } # foreach @$reclist
[460]611 } # nrecs != 0
612 if ($args{parpatt}) {
613 # We've just gone and axed all the records in the reverse zone.
614 # Re-add one to match the parent if we've been given a pattern to use.
[481]615 $dnsdb->addRec(defrec =>'n', revrec => 'y', parent_id => $zdata->{rdns_id},
[460]616 type => ($cidr->{isv6} ? 65284 : 65283),
617 address => $zdata->{revnet}, name => $args{parpatt}, %args);
[459]618 }
619 } # iterate zones within $cidr
620 } # done $cidr-contains-zones
621
622} # end delByCIDR()
623
[405]624#sub getLogCount {}
625#sub getLogEntries {}
[452]626
627sub getRevPattern {
628 my %args = @_;
629
630 _commoncheck(\%args, 'y');
631
[483]632 return $dnsdb->getRevPattern($args{cidr}, $args{group});
[452]633}
634
[405]635#sub getTypelist {}
636#sub parentID {}
637#sub isParent {}
[123]638
[405]639sub zoneStatus {
[123]640 my %args = @_;
641
[405]642 _commoncheck(\%args, 'y');
[123]643
[477]644 my @arglist = ($args{zoneid});
[123]645 push @arglist, $args{status} if defined($args{status});
646
[477]647 my $status = $dnsdb->zoneStatus(@arglist);
[123]648}
649
[452]650# Get a list of hashes referencing the reverse zone(s) for a passed CIDR block
651sub getZonesByCIDR {
652 my %args = @_;
653
654 _commoncheck(\%args, 'y');
655
[477]656 return $dnsdb->getZonesByCIDR(%args);
[452]657}
658
[405]659#sub importAXFR {}
660#sub importBIND {}
661#sub import_tinydns {}
662#sub export {}
663#sub __export_tiny {}
664#sub _printrec_tiny {}
665#sub mailNotify {}
[119]666
667sub get_method_list {
668 my @methods = keys %{$methods};
669 return \@methods;
670}
Note: See TracBrowser for help on using the repository browser.