/
gdnsd-plugin-geoip.podin
614 lines (516 loc) · 25.2 KB
/
gdnsd-plugin-geoip.podin
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
=head1 NAME
gdnsd-plugin-geoip - gdnsd meta-plugin for GSLB + failover via MaxMind's
GeoIP2 databases
=head1 SYNOPSIS
Minimal example gdnsd config file using this plugin:
plugins => { geoip => {
maps => {
my_prod_map => {
geoip2_db => GeoIP2-City.mmdb,
datacenters => [dc-03, dc-02, dc-01, dc-fail],
map => {
EU => {
DE => [dc-03, dc-01, dc-fail],
CH => [dc-01, dc-03, dc-fail]
},
NA => { MX => [dc-02, dc-fail] }
}
},
my_auto_map => {
geoip2_db => GeoIP2-Country.mmdb,
datacenters => [dc1, dc2],
auto_dc_coords => {
dc1 => [ 38.9, -77 ],
dc2 => [ 50.1, 8.7 ],
}
}
},
resources => {
prod_www => {
map => my_prod_map
service_types => up
dcmap => {
dc-01 => 192.0.2.1,
dc-02 => { lb01 => 192.0.2.2, lb02 => 192.0.2.3 },
dc-03 => [ 192.0.2.4, 192.0.2.5, 192.0.2.6 ],
dc-fail => last.resort.cname.example.net.
}
}
corp_www => {
map => my_auto_map
dcmap => {
dc1 => 192.0.2.100,
dc2 => 192.0.2.101
}
}
}
}}
Example zonefile RRs in zone example.com:
www 600 DYNA geoip!prod_www
www-dc01 600 DYNA geoip!prod_www/dc-01
www.corp 600 DYNA geoip!corp_www
=head1 DESCRIPTION
B<gdnsd-plugin-geoip> uses MaxMind's GeoIP2 binary databases to
map address and CNAME results based on geography and
monitored service availability. It fully supports both
IPv6 and the emerging edns-client-subnet standard. If a request
contains the edns-client-subnet option with a source netmask greater
than zero, the edns-client-subnet information will be used instead
of the source IP of the request (the IP of the querying cache).
It supports the GeoIP2 format databases, which typically end in F<.mmdb>. It
does not supports the legacy GeoIP1 format databases (which typically end in
F<.dat>).
It can also be used with no GeoIP database at all, in which case the only
network-mapping input comes from the C<nets> config data or an external
C<nets> file, which explicitly map subnets to datacenter lists.
This plugin can operate in an automatic distance-based mode (using a City-level
database's coordinate information) It can also operate coordinate-free and rely
on the user to configure a hierarchical map of cascading default
user-location-to-datacenter mappings, starting at the continent level.
The two modes can also be effectively mixed at geographic boundaries.
For each C<map> you define (which maps geographic location codes to
preference-ordered lists of your datacenter locations), this plugin merges
all of the raw GeoIP subnets into the largest possible supernets which
contain identical responses in your configuration. These in turn are used
to set larger edns-client-subnet scope masks than you'd see simply
returning raw GeoIP results.
=head1 PLUGIN_METAFO
The documentation for L<gdnsd-plugin-metafo(8)> is required reading for
understanding the geoip plugin documentation here. The geoip plugin is an
exact superset of the metafo plugin, and re-uses almost all of the metafo
plugin's source code. Metafo does failover along a single, global, ordered
list of datacenters. What plugin_geoip adds on top of the functionality
of metafo is the ability to have the order of the datacenter failover list
become dynamic per-request based on geographic hints derived from the
client's network address.
=head1 FILE LOCATIONS
The configuration of this plugin can reference several external
configuration and/or data files. By default, all files referenced in
this plugin's configuration are loaded from the F<geoip> subdirectory
of the daemon's configuration directory (default F<@GDNSD_DEFPATH_CONFIG@>).
You can load from other locations by specifying absolute file paths.
=head1 CONFIGURATION - TOP-LEVEL
The top level of the geoip plugin's configuration (i.e. C<plugins =E<gt> {
geoip =E<gt> { ... } }>) supports only three explicit keys. One is the
optional setting C<undefined_datacenters_ok>.
The other two are required and expanded upon in detail in the next two
sections: C<maps>, and C<resources>. The C<maps> section defines one or more
named mappings of location information from GeoIP binary databases to ordered
subsets of datacenter names. The C<resources> section defines one or more
named resources, each of which references one of the named maps and resolves
datacenter names to specific sets of addresses or CNAMEs.
Any other keys present at this level will be inherited down inside of each
per-resource hash inside the C<resources> stanza, acting as per-resource
defaults for anything not defined explicitly there.
=head2 C<undefined_datacenters_ok = false>
Boolean, default false. If set to true, geoip resources are allowed to leave
some of the datacenters specified in their C<map> undefined in their
resource-level C<dcmap>. For example, a map M might define 3 datacenters
named A, B, and C, but a resource using map M might only define result
addresses for datacenters B and C in its C<dcmap>. This would otherwise be a
hard configuration error.
B<!!! DANGER !!!> - Setting this value to true is a good way to shoot yourself
in the foot if you're not very careful about how your maps and resources are
configured with respect to each other, especially in "City Auto Mode". Maps
are calculated without any knowledge of the resources that use them. If a
specific network or location maps to a list of datacenters which contains none
of the defined datacenters for a given resource, the results of runtime
queries for that resource from that location or network will be the empty set
(no answer records at all). This is virtually guaranteed to happen in "City
Auto Mode" if the number of undefined datacenters in a resource is greater
than or equal to the map's C<auto_dc_limit>.
=head1 CONFIGURATION - MAPS
All C<maps>-level configuration keys are the names of the maps you
choose to define. A map, conceptually, is a mapping between geography
and/or network topology to varying ordered datacenter sub-sets. The value
of each named map must be a hash, and the following configuration keys
apply within:
=head2 C<geoip2_db = GeoIP2-City.mmdb>
String, filename, optional. This is the filename of a MaxMind GeoIP2 format
database. It should contain either the City or Country data model. There is
no distinction made here for the IP version, and it is normal for these
databases to contain both IPv4 and IPv6 data together. If one or the other is
missing, clients using that address family will be defaulted.
=head2 C<datacenters = [ one, two, three, ... ]>
Array of strings, required. This is the total set of datacenter names used
by this map. You must define at least one datacenter name (although 2 or
more would be infinitely more useful). At this time, there is a maximum
limit of 254 datacenter names per map, although this could be raised if
anyone requires it. The order specified here is the fallback default
result ordering in various default cases (e.g. if no explicit top-level map
default list is given).
=head2 C<ignore_ecs = true>
Boolean, default false. If this is set to C<true>, all resources using
this map will ignore EDNS Client Subnet (ECS) information when
performing lookups against the map, relying solely on the DNS source IP
for the lookup. If the client provided ECS in such a query, the
response will also contain the ECS option to signal that we're ECS aware
in general, but the response scope mask will be set to zero to signal
the cache that the ECS data wasn't used and the result is globally
cacheable.
This is useful for situations in which the map is intentionally meant to
operate solely on recursor IPs rather than ECS IPs, but other maps
serviced by the same authserver do make use of ECS data, and thus the
global B<edns_client_subnet> config option can't be used to disable
handling the option completely for the whole server.
=head2 C<nets = { ... }>
Key-value hash, optional (see below for alternate form). If specified, the
contents should be key-value pairs of C<network/netmask> mapped to a
datacenter name (or an array of datacenter names). Any
network-to-datacenter mappings specified here will override mappings
determined via GeoIP. Note that it is illegal to specify networks in the
IPv4-like subspaces of IPv6 other than v4compat, but it is legal to specify
actual IPv4 networks (which are treated identically to v4compat). See the
section on IPv4 Compatible Addresses later in this document for more
details. The order of the networks is unimportant; they will always be
sorted and inserted such that an entry which is a subnet of another entry
is not obliterated by the parent supernet.
nets => {
10.0.0.0/8 => [ dc1, dc2 ],
192.0.2.128/25 => dc3
2001:DB8::/32 => [ dc4, dc5, dc6 ],
}
In the case that one entry is a subnet of another with a different result
dclist, the entries are merged correctly such that the supernet surrounds
the subnet. In the case of an exact duplicate entry (or an effective one,
after merging smaller subnets) with a different dclist, it is arbitrary
which one "wins" and the condition is warned about. If you care about
this case, you should sanitize your nets data beforehand with an external
tool and/or parse for the warning message in log outputs.
=head2 C<nets = nets_file_name>
String pathname, optional. A variant of the above, but the contents of the
key-value hash are loaded from the named external file. This makes life
easier for external tools and scripts generating large sets of nets entries
(e.g. from BGP data). The file will be monitored for changes and reloaded
at runtime much like the GeoIP databases.
=head2 C<map = { ... }>
Key-value hash, optional. This is the heart of a named map which uses
GeoIP: the map itself, which maps places to ordered lists of datacenters.
It requires C<geoip2_db> is also specified, and makes no sense without it.
This is a nested key-value hash. At each level, the keys are location
codes (continent, country, region/subdivision, or city information depending
on depth), and the values are either an ordered datacenter array (e.g. C<[
dc03, dc01, dc04 ]>), or a sub-hash containing a deeper level of distinction.
At each layer, a special key named C<default> is available, which sets the
default for everything within the current scope. The top-level default itself
defaults to the ordered list from C<datacenters> in the normal case. If
the entire C<map> stanza is missing or empty, you just get the default
behavior of C<default>. A datacenter array can also be empty, which
implies that this location is mapped to receive no response data (the
server will still respond to the query, and will not issue an NXDOMAIN. It
will simply be a NODATA/NOERROR response like you'd get if there were no
records of this type, but could be records of other types for the same
name).
=head2 GeoIP2 Location Data Hierarchy
The top level of the map hierarchy is comprised of MaxMind's seven
continent codes: C<AF> for Africa, C<AS> for Asia, C<NA> for North America,
C<SA> for South America, C<EU> for Europe, C<OC> for Oceania, and C<AN> for
Antarctica. The next level is the ISO 3166-1 2-letter country code.
From here there are a number of Subdivision levels, the count of which varies
for different network database entries. In the US, for example, there is only
one level of subdivision data for the US States. In the Czech
Republic there are two levels of subdivision: first into 14 regions, and then
further into 91 districts. Subdivisions are all specified using their
ISO 3166-2 codes directly.
After all subdivision levels, the final level is the City level. The City
names are all in the UTF-8 character set. Currently this plugin only uses the
English city names from the database, even though other languages may be
available depending on the database.
As a pragmatic answer to the issues that can arise with multiple subdivision
layers, the map automatically searches deeper in the database data when no map
match is found at a given level of the map hierarchy beneath the Country level.
This means you can skip over any levels of Subdivision detail in your map that
are irrelevant to you.
For example, this targets the New Zealand regional council subdivision of
Otago without explicitly specifying the enclosing subdivision for the South
Island:
{ OC => { NZ => { OTA => [...] } } }
As another example, this works correctly for targeting the city of Paris
without caring about what layers of subdivisions lie between it and FR:
{ EU => { FR => { Paris => [...] } } }
=head1 CONFIGURATION - MAPS - CITY AUTO MODE
"City-auto-mode" is a special mode of operation that automatically maps out
the world to your datacenters based on coordinate math, so that you don't
have to manually construct a complex hierarchical C<map>. It can still be
mixed with C<map> of course, allowing you to use auto-mode for only select
geographic areas if you wish (or disabling it for select areas by
specifying manual lists). The key parameter is C<auto_dc_coords>, which
enables city-auto-mode. This requires a City-level GeoIP2 database; the
Country ones don't contain coordinate information.
=over 4
=item C<auto_dc_coords = { ... }>
Key-value hash, optional. If this option is specified, the whole map's
basic mode of operation changes to "city-auto-mode". The contents of the
hash are a key for each datacenter named in C<datacenters>, with their
values set to an array of C<[lat, lon]> in decimal degree units. When
city-auto-mode is enabled by this, the following configuration-validation
changes occur from the default, static-mapping mode: the loaded GeoIP2
database(s) are required be City-level databases, and the special keyword
C<auto> becomes a legal "datacenter list" in the C<map> stanza.
With city-auto-mode enabled, the top-level map C<default> defaults to
C<auto>, but can be overridden with a manual list. For any location that
maps to C<auto>, the coordinates specified here in C<auto_dc_coords> will
be compared with the coordinates from the City-level database(s) to
determine an automatic distance-sorted datacenter list.
If you omit one or more defined datacenters from the coordinate list in
C<auto_dc_coords>, those datacenters will not be used in automatic results,
but will still be available for manual use via C<map> and/or C<nets>
entries.
=item C<auto_dc_limit = N>
Unsigned integer, optional, default 3. When city-auto-mode is in effect,
this is the upper length limit for auto-generated lists. 3 is a reasonable
default even if you have a considerably longer set of datacenters, as this
provides a primary as well as two fallbacks. Raising this to a large
number in the presence of a long datacenter list will cause the set of
unique result datacenter lists to increase rapidly, and thus reduce the
optimization of the final result database for edns-client-subnet purposes.
It's really not worth raising this value in almost any case, unless you
really need to handle more than 3 random datacenters going offline at the
same time and still have clients fail elsewhere. The value zero is treated
as unlimited (highly un-recommended).
=back
Under city-auto-mode, when the top-level default is (explicitly or
implicitly) C<auto>, there is still a fallback static ordering which is the
whole ordered C<datacenters> list, which is the normal static default
C<default> when not in city-auto-mode. This fallback is used when no
location information is available at all (e.g. IPv6 client vs IPv4 GeoIP
DB, Anonymous Proxies, etc).
=head1 MAP TESTING
A binary program C<gdnsd_geoip_test> is included. This can be used
directly from the commandline, parses the relevant bits of your gdnsd
config file for geoip map info, and then provides datacenter list results
for IP address + map combinations supplied by the user. Useful for
debugging your maps and testing the mapping of client IPs. It has a
separate manpage L<gdnsd_geoip_test(1)>.
=head1 CONFIGURATION - RESOURCES
Resource-level configuration within the C<resources> stanza is nearly
identical to the resources configuration of the metafo plugin, with all of
the same basic behaviors about synthesizing or directly referencing the
configuration of other plugins per-datacenter.
One difference is that metafo's per-resource C<datacenters> array is replaced
with C<map =E<gt> mapname>, which references one of the maps defined in the
C<maps> stanza, described in detail earlier. The set of defined datacenters in
the C<dcmap> stanza must match the total set of datacenters defined by the
referenced map, unless C<undefined_datacenters_ok> is set to C<true> (see
warnings and documentation above).
The C<skip_first> flag can also be set per resource, and is much more useful
with the geoip plugin than it is with the basic metafo plugin. If this flag is
set, the first datacenter in the failover list for a given lookup will be
skipped, allowing the definition of a "second choice" resource using the same
basic map definition as the first choice. In this case the original first
choice is *never* a possible answer, and the rest of the logic (e.g. skipping
datacenters marked as down) proceeds as normal with the remaining list. If the
map entry and/or the resource definition have already reduced the effective
datacenter count to one, the flag has no effect.
=head1 META-PLUGIN INTERACTION
Both of the meta-plugins (C<metafo> and C<geoip>) can reference their own
as well as each others' resources by direct reference within a C<dcmap>, so
long as a resource does not directly refer to itself. This allows
plugin-layering configurations such as geoip -> metafo -> weighted, or
metafo -> geoip -> multifo, or even metafo -> metafo -> simplefo, etc.
Bear in mind that once you begin using inter-meta-plugin references, you
could create a reference loop. gdnsd does not currently detect or prevent
such loops, and they will cause complete runtime failure when queried,
probably by running out of stack space during recursion.
Additionally, C<geoip> can synthesize configuration for C<metafo>
resources, but the reverse does not hold; C<metafo> cannot synthesize
configuration for C<geoip> resources.
=head1 IPv4 Compatible Addresses
This plugin knows of six different relatively-trivial ways to map IPv4
addresses into the IPv6 address space. These are shown below in as much
detail matters to this plugin, with C<NNNN:NNNN> in place of the copied
IPv4 address bytes:
::0000:NNNN:NNNN/96 # RFC 4291 - v4compat (deprecated)
::ffff:NNNN:NNNN/96 # RFC 4291 - v4mapped
::ffff:0000:NNNN:NNNN/96 # RFC 2765 - SIIT (obsoleted)
64:ff9b::NNNN:NNNN/96 # RFC 6052 - Well-Known Prefix
2001:0000:X:NNNN:NNNN/32 # RFC 4380 - Teredo (IPv4 bits are flipped)
2002:NNNN:NNNN::/16 # RFC 3056 - 6to4
(in the Teredo case above, "X" represents some variable non-zero bytes
that occupy the center 64 bits of the address).
All of this plugin's internal lookup databases are IPv6 databases, and any
IPv4-like information is always stored in the v4compat space within these
databases. When doing runtime lookups all other v4-like addresses (raw
IPv4 addresses, v4mapped, SIIT, WKP, Teredo, and 6to4) are converted to the
canonical v4compat IPv6 representation before querying the internal
databases. The other representations (v4mapped, SIIT, WKP, Teredo, 6to4) are
Undefined internally, and will never be referenced at lookup-time due to
the v4compat conversion mentioned earlier.
The C<nets> stanza is not allowed to specify entries in the five undefined
v4-like IPv6 spaces (those other than v4compat). Specify those networks as
normal IPv4 networks or v4compat networks instead. Legitimate IPv6 C<nets>
entries which happen to be a supernet of any v4-like spaces will *not*
unduly affect v4-like lookups. There is no functional difference between
v4compat and native v4 forms in C<nets>, e.g. C<192.0.2.0/24> and
C<::C000:0200/120> are completely identical.
GeoIP databases that are natively IPv4-only get all of their data loaded
into the v4compat space only. For normal IPv6 GeoIP databases, by default we
load the v4compat space directly (which is where MaxMind stores IPv4 data in
their IPv6 databases), but ignore the v4mapped/SIIT/Teredo/6to4 spaces
(some of which are empty in MaxMind's databases, and some of which simply
alias the v4compat space).
=head1 ANOTHER CONFIG EXAMPLE
A relatively-maximal example config, showing the interaction of valid
C<maps> and C<resources> sections:
service_types => {
xmpp_svc => { plugin => "tcp_connect", ... }
www_svc => { plugin => "http_status", ... }
}
plugins => {
geoip => {
maps => {
my_prod_map => {
geoip2_db => GeoIP2-City.mmdb,
datacenters => [us-01, de-01, sg-01],
map => {
# Hierarchy is Continent -> Country -> Region -> City
NA => {
US => {
Dallas => [sg-01],
}
}
SA => [us-01, sg-01, de-01],
EU => {
default => [de-01, us-01, sg-01],
CH => {
Geneve => {
Geneva => [sg-01],
}
}
}
AF => [de-01, us-01, sg-01],
AS => [sg-01, de-01, us-01],
OC => [sg-01, us-01, de-01],
}
nets => {
10.0.0.0/8 => [ de-01 ],
2001:DB8::/32 => [ us-01 ],
}
}
my_auto_map => {
geoip2_db => GeoIP2-City.mmdb,
datacenters => [us-01, de-01, sg-01],
auto_dc_coords => {
us-01 => [ 38.9, -77 ],
de-01 => [ 50.1, 8.7 ],
sg-01 => [ 1.3, 103.9 ],
}
}
}
resources => {
prod_app => {
map => my_auto_map
# these two are inherited multifo config keys
# for all of the dcmap below:
service_types => [www_svc, xmpp_svc],
up_thresh => 0.4,
dcmap => {
us-01 => {
lb01 => 192.0.2.1,
lb02 => 192.0.2.2,
lb03 => 192.0.2.3,
lb01.v6 => 2001:DB8::1,
lb02.v6 => 2001:DB8::2,
lb03.v6 => 2001:DB8::3,
},
sg-01 => {
lb01 => 192.0.2.4,
lb02 => 192.0.2.5,
lb03 => 192.0.2.6,
lb01.v6 => 2001:DB8::4,
lb02.v6 => 2001:DB8::5,
lb03.v6 => 2001:DB8::6,
},
de-01 => {
lb01 => 192.0.2.7,
lb02 => 192.0.2.8,
lb03 => 192.0.2.9,
lb01.v6 => 2001:DB8::7,
lb02.v6 => 2001:DB8::8,
lb03.v6 => 2001:DB8::9,
},
}
},
prod_cdn => {
map => my_prod_map,
dcmap => {
us-01 => us-cdn-provider.example.net.
sg-01 => asia-cdn-provider.example.net.
de-01 => europe-cdn-provider.example.net.
}
}
}
}
}
Example zonefile RRs in zone example.com:
app 600 DYNA geoip!prod_app
app.us 600 DYNA geoip!prod_app/us-01
app.sg 600 DYNA geoip!prod_app/sg-01
app.de 600 DYNA geoip!prod_app/de-01
content 600 DYNC geoip!prod_cdn
=head1 EXAMPLE OF METAFO->GEOIP CITY-AUTO-MODE w/ LAST RESORT CNAME
plugins => {
geoip => {
maps => {
auto_map => {
geoip2_db => GeoIP2-City.mmdb,
datacenters => [dc1, dc2, dc3, dc4],
auto_dc_coords => {
dc1 => [ 38.9, -77 ],
dc2 => [ 50.1, 8.7 ],
dc3 => [ 20.2, 88.9 ],
dc4 => [ 39.0, -20 ],
},
# only fail through the nearest 2 before giving up:
auto_dc_limit => 2,
}
},
resources => {
www_real => {
map => my_auto_map,
service_types => [ http, xmpp ],
dcmap => {
dc1 => 192.0.2.100,
dc2 => 192.0.2.101,
dc3 => 192.0.2.102,
dc4 => 192.0.2.103
}
}
}
},
metafo => {
resources => {
www => {
datacenters => [ real, backup ],
dcmap => {
real => %geoip!www_real,
backup => backup-host.example.net.
}
}
}
}
}
And in the example.com zonefile:
; This tries through the closest 2/4 datacenters to
; the client from the geoip map, and if both of
; those are down it returns a CNAME to backup-host.example.net.
; for a downtime message or something:
www DYNC metafo!www
=head1 SEE ALSO
L<gdnsd-plugin-metafo(8)>, L<gdnsd_geoip_test(1)>, L<gdnsd.config(5)>,
L<gdnsd.zonefile(5)>, L<gdnsd(8)>
The gdnsd manual.
=head1 COPYRIGHT AND LICENSE
Copyright (c) 2012 Brandon L Black <blblack@gmail.com>
This file is part of gdnsd.
gdnsd is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
gdnsd is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with gdnsd. If not, see <http://www.gnu.org/licenses/>.
=cut