Для того, чтобы BIND начал отвечать на IPv6-запросы, нужно совсем немного.
Сначала — нужно убедиться, что в конфигурационном файле BIND упоминается loopback-зона, то есть имеются такие строки:
zone "0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.ip6.arpa" {
type master;
file "standard/loopback6.arpa";
allow-transfer { localhost; };
};
Содержимое файла loopback6.arpa выглядит так:
$ORIGIN 0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.ip6.arpa.
$TTL 6h
@ IN SOA localhost. root.localhost. (
1 ; serial
1h ; refresh
30m ; retry
7d ; expiration
1h ) ; minimum
NS localhost.
1 PTR localhost.
И — нужно проверить наличие AAAA-записи в файле localhost:
$ORIGIN localhost.
$TTL 6h
@ IN SOA localhost. root.localhost. (
1 ; serial
1h ; refresh
30m ; retry
7d ; expiration
1h ) ; minimum
NS localhost.
A 127.0.0.1
AAAA ::1 ; — эта строка должна быть!
Как правило, все вышеупомянутые записи делаются автоматически при инсталяции BIND.
Затем нужно добавить в конфигурационный файл:
options {
........
listen-on-v6 { any; }; --- нужно добавить (или перечислить только нужное вам)
allow-recursion { clients; }; --- как правило, уже есть
};
и скорректировать (в нашем случае) acl «clients», чтобы сервер отвечал на рекурсивные запросы только от “своих” клиентов:
acl clients {
......
2001:0db8::/32;
......
};
Все остальные необходимые настройки (loopback-адреса и зоны) уже имелись в дистрибутиве.
Кстати, в файле root.hint уже должны быть IPv6-адреса для нескольких корневых серверов.
Затем можно приступать к добавлению записей в файлы прямых зон и к созданию новых файлов реверсных зон для IPv6.
Поискав («погуглив») информацию на эту тему, обнаружил, что большинство поступает с «прямыми» зонами просто, то есть к любой записи А добавляют запись АААА, и мы решили сделать так же. Тут, главное, не запутаться в адресах… И — все больше приходит осознание того, что поскольку IPv6-адреса запомнить очень сложно (свой префикс выучил только на второй день), то роль DNS возрастает. Имена-то запоминаются легче.
Перед «наполнением» файлов зон АААА-записями начинаем планировать распределение подсетей. Для начала решили сохранить нынешнюю структуру, то есть если у нас есть отдельная подсеть IPv4 с любой маской, то ей будет соответствовать отдельная подсеть IPv6/64. Маска /64 для end-юзерских подсетей выбрана в соответствии с рекомендациями RIPE, хотя подсетей такого размера в IPv4 у нас нет, в нашем блоке адресов /20 размер самой большой подсети составляет /23. Правда, некоторые подсети IPv4 хотелось бы обьединить, а некоторые разделить, но такую реструктуризацию мы будем делать уже в пространстве IPv6.
По нашим оценкам, 16-битного поля, выделяемого в структуре адреса IPv6 под номер подсети, нам вполне хватит на ближайшие годы и пятилетки.
Итак, начинаем «заводить» первые записи в прямые зоны.
Выбираем вторую подсеть (то есть, проще говоря, устанавливаем 64й бит в «1») и добавляем в «прямой» зоне (скажем, domain.ru) две записи:
test1 AAAA 2001:0db8:0000:0001:0000:0000:0000:0001
test2 AAAA 2001:0db8:0000:0001:0000:0000:0000:0002
Перезапускаю BIND и проверяю nslookup-ом на самом сервере:
dig test1.domain.ru aaaa
;; <<>> DiG 9.4.2-P2 <<>> test1.bestcom.ru aaaa
;; global options: printcmd
;; Got answer:
;; << HEADER >> - opcode: QUERY, status: NOERROR, id: 35937
;; flags: qr aa rd ra; QUERY: 1, ANSWER: 1, AUTHORITY: 4, ADDITIONAL: 4
;; QUESTION SECTION:
;test1.domain.ru. IN AAAA
;; ANSWER SECTION:
test1.domain.ru. 600 IN AAAA 2001:0db8:0:1::1
Вроде все верно, ответ получен. Запрос к хосту test2 также проходит нормально.
Теперь проверяю со своей машины из офисной сети (учтите, дело было в 2012 году).
Microsoft Windows XP [Version 5.1.2600]
(C) Copyright 1985-2001 Microsoft Corp.
C:> nslookup
server xxx.yyy.zzz.ttt
Default Server: ns.domain.ru
Address: xxx.yyy.zzz.ttt
set q=aaaa
test1.domain.ru
Server: ns.domain.ru
Address: xxx.yyy.zzz.ttt
test1.domain.ru AAAA IPv6 address = 2001:0db8:0:1::1
…
test2.domain.ru
Server: ns.domain.ru
Address: xxxx.yyyy.zzzz.tttt
test2.domain.ru AAAA IPv6 address = 2001:0db8:0:1::2
…
>
exit
И здесь все нормально! Теперь — создаем реверсные зоны. Кстати, делегирование обьекта domain для IPv6-префикса у нас еще не оформлено… Ну ничего, не забудем.
Создаем первую реверсную зону. Название зоны выбираем как 2001:0db8, благо UNIX не возражает против такого имени файла.
Файл реверсной зоны для IPv6 устроен так же, как и для IPv4, его начало выглядит так:
$ORIGIN .
$TTL 7200 ; 2 hours
8.b.d.0.1.0.0.2.ip6.arpa IN SOA ns.domain.ru mail.domain.ru. (
2010121801 ; serial
18000 ; refresh (5 hours)
3600 ; retry (1 hour)
1728000 ; expire (2 weeks 6 days)
172800 ; minimum (2 days)
)
NS ns.domain.ru.
NS ns1.domain.ru.
; subnet 2001:0db8::/32
$ORIGIN 8.b.d.0.1.0.0.2.ip6.arpa.
… далее идут записи зоны …
1.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.1.0.0.0.0.0.0.0 IN PTR test1.domain.ru.
2.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.1.0.0.0.0.0.0.0 IN PTR test2.domain.ru.
Для упрощения (автоматизации) преобразований можно использовать утилиту из состава Unix-Linux, она сразу преобразует адрес в ptr-формат:
$ host -n 2001:db8::1
Host 1.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.8.b.d.0.1.0.0.2.ip6.arpa not found: 3(NXDOMAIN)
Проверяю тут же при помощи dig:
dig @ns.mydomain.ru 2xxx:yyyy:0:1::1 ptr
…… и получаю сообщение об ошибке!
А вот nslookup работает!
nslookup
set q=ptr
2xxx:yyyy:0:1::3
Server: 123.12.123.1
Address: 123.12.123.1#53
1.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.1.0.0.0.0.0.0.0.y.y.y.y.x.x.x.2.ip6.arpa name = test1.domain.ru.
То есть dig под UNIX не понимает, что это — структура адреса в формате IPv6…. Нужно будет проверить на самой свежей версии UNIX.
Проверил. Оказалось, что версия не при чем. А чтение документации показало, что вот такой формат команды работает:
dig @8.8.8.8 -x 2001:db8:0:1::1
Спасибо Google DNS, здорово выручает при проверках.
Тот же nslookup под WinXP при получении такой команды выдает:
Unrecognized command: 2001:0db8:0:1::1
— то есть он не понимает, что это адрес в формате IPv6.
(Нужно будет проверить это на Висте и 7-ке. - написано в 2012 году)
Кстати, если на DNS-серверах, которые являются авторитарными для каких-то зон, поднять IPv6, то также нужно на вышестоящих серверах (в данном случае — в РУ-центре) добавить IPv6-адреса к данным своих DNS-серверов, то есть сделать «glue ns records». Процедура несложная.
Ниже все вышеизложенное приведено в компактном виде:
(примечание: каких-либо сведений о конфигурировании адресов интерфейсов и роутинга здесь не будет)
Получить от RIR (у нас — от RIPE) блок адресов IPv6. Как и все остальные действия с RIPE, данную процедуру можно осуществить 3 способами:
ИМХО, лучше это сделать через LIR-portal, где есть специальная форма для первого заказа IPv6-адресов
Если пойти третьим путем, то обычным путем получаем шаблон :
whois -h whois.ripe.net — ‘-t inet6num’
(так делается на BSD)
В ответ приходит:
inet6num: [mandatory] [single] [primary/look-up key]
netname: [mandatory] [single] [lookup key]
descr: [mandatory] [multiple] [ ]
country: [mandatory] [multiple] [ ]
org: [optional] [single] [inverse key]
admin-c: [mandatory] [multiple] [inverse key]
tech-c: [mandatory] [multiple] [inverse key]
status: [mandatory] [single] [ ]
mnt-by: [mandatory] [multiple] [inverse key]
mnt-lower: [optional] [multiple] [inverse key]
mnt-routes: [optional] [multiple] [inverse key]
mnt-domains: [optional] [multiple] [inverse key]
mnt-irt: [optional] [multiple] [inverse key]
changed: [mandatory] [multiple] [ ]
source: [mandatory] [single] [ ]
Далее заполняем его, добавляем аутентификационные данные, и отправляем.
Для обращения к реверсным зонам требуется оформить в RIPE обьект domain
.
Предположим, что нам выдали 2001:0db8::/32 (вроде бы всем LIR дают /32) Сразу заказываем обьект domain
, для этого нужно знать имена своих dns-серверов. Обьект domain
можно сделать любым из трех способов, мы делали через Webupdates.
Каких-либо конфигурационных изменений в настройках DNS, связанных с IPv6, на этом этапе делать не нужно. Точнее — не требуется.
Шаблон можно получить так: (пробел перед ключом -t
НЕ ОШИБКА!!!)
whois -h whois.ripe.net — ‘ -t domain’
Вот образец-макет того, что отправляют в RIPE:
% Information related to '8.b.d.0.1.0.0.2.ip6.arpa'
domain: 8.b.d.0.1.0.0.2.ip6.arpa
descr: Reverse delegation for 2001:0db8::/32
nserver: ns1.server.zz
nserver: ns2.server.zz
admin-c: PERSON1-RIPE
tech-c: PERSON2-RIPE
zone-c: PERSON3-RIPE
source: RIPE # Filtered
mnt-by: EARTH-MNT
Сразу же можно (и, имхо, желательно) завести обьект route6
Команда для получения шаблона обьекта:
whois -h whois.ripe.net — ‘ -t route6’
А заполняем примерно так: (аутентификация не показана)
route6: 2001:0db8::/32
descr: AS23456 IPv6 block
origin: AS23456
mnt-by: PERSON-MNT
source: RIPE
Не забыть про аутентификацию!
В обьекте route6 есть еще один интересный атрибут: в нем можно указать пингуемый адрес, то есть адрес из вашего блока, который отвечает на пинги из всех сетей. Знание такого адреса очень полезно при отладке, при трассировке и проч. После «подьема» IPv6 BGP мы сразу же завели такой адрес у себя.
В шаблоне это выглядит так:
pingable: [optional] [multiple] [ ]
ping-hdl: [optional] [multiple] [inverse key]
И — вот так это выглядит в жизни, например:
pingable: 2001:0db8::1
ping-hdl: PERSON1-RIPE
Второй атрибут также имеет ценность: если адрес не отвечает на ICMP-запросы, а вы уверены, что у вас все нормально — то можно написать письмо ответственному и указать на недостатки (разумеется, если суметь извлечь почтовый адрес).
Далее — переходим к DNS и формируем реверсные зоны. Наша зона (в соответствии с префиксом 2001:0db8::/32) будет выглядеть так: 8.b.d.0.1.0.0.2.ip6.arpa
, соответственно так назовем и файл. Прописываем это дело в конфигурационый файл и переходим к файлу зоны:
SOA-запись ничем не отличается от обычной, записи для ns-серверов тоже обычные
$ORIGIN .
$TTL 7200 ; 2 hours
8.b.d.0.1.0.0.2.ip6.arpa IN SOA ns1.domain.zz. dnsmaster.domain.zz. (
2011010201 ; serial
18000 ; refresh (5 hours)
3600 ; retry (1 hour)
1728000 ; expire (2 weeks 6 days)
172800 ; minimum (2 days)
)
NS ns1.domain.zz.
NS ns2.domain.zz.
Реверсная запись для нашего «пингуемого» адреса (пусть его имя будет ping6.domain.zz) будет выглядеть так:
1.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.8.b.d.0.1.0.0.2.ip6.arpa IN PTR ping6.domain.zz.
Не правда ли, длинновато получилось? Не всегда умещается на одной строке (см.выше насчет частичной автоматизации).
Спасает ORIGIN. Если взять «стандартную» для IPv6-документации подсеть 2001:0db8::/64, то первый, второй и третий адрес будут выглядеть так:
;subnet 2001:0db8::/64
$ORIGIN 0.0.0.0.0.0.0.0.8.b.d.0.1.0.0.2.ip6.arpa.
1.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.ip6.arpa IN PTR pingable.domain.zz
2.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.ip6.arpa IN PTR 2.domain.zz
3.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.ip6.arpa IN PTR 3.domain.zz
Отметим, что адрес 2001:0db8::/64 также допустим, и его реверсная запись будет такой:
0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.8.b.d.0.1.0.0.2.ip6.arpa NS zero.domain.zz
Использовать автозаполнение последовательно идущих записей нет смысла, так как зона получится длиной от Земли до Луны.
RIPE выдает для LIR подсети /32, то есть по 4 миллиона адресов на 1 квадратный метр земной поверхности. Конечно же, можно все реверсные записи для одного блока «запихать» в один файл, но:
— он станет огромным — будет сложно искать и редактировать нужную запись — придется перезагружать огромный файл
А если клиент попросит делегировать ему его подсеть?
То есть имеет смысл делать делегирование реверсных субзон и делать отдельные файлы на подсети, так как при этом будет меньше работы (делегирование) и файлы зон станут короче (соответственно, будут быстрее находитьcя записи). И работать станет проще. Даже если это делать только для себя и не делегировать клиентам.
Предположим, вы хотите делегировать подсети «a» и «b» (для /64 префикса) на сервер ns3.server.zz, то есть вы хотите, что бы реверсные запросы к подсетям 2001:0db8:0:a::/64 и 2001:0db8:0:b::/64 направлялись к серверу ns3.server.zz. В отличие от IPv4 это делается достаточно просто.
Во-первых, на сервере ns1.domain.zz в начале файла зоны 8.b.d.0.1.0.0.2.ip6.arpa делаем такие записи:
; delegate 2001:0db8:0000:000a::/64
a.0.0.0.0.0.0.0.8.b.d.0.1.0.0.2.ip6.arpa NS ns3.server.zz
; delegate 2001:0db8:0000:000b::/64
b.0.0.0.0.0.0.0.8.b.d.0.1.0.0.2.ip6.arpa NS ns3.server.zz
Затем, на сервере ns3.server.zz в конфигурационном файле обьявляем две мастер-зоны:
a.0.0.0.0.0.0.0.8.b.d.0.1.0.0.2.ip6.arpa
и
b.0.0.0.0.0.0.0.8.b.d.0.1.0.0.2.ip6.arpa
(обычным образом, как и любую другую зону)
И — создаем файлы для этих зон, полностью аналогично зоне 2001:0db8::/32
Например, фрагмент первой зоны может выглядеть так (soa-запись пропущена, она обычная)
; subnet 2001:0db8:0:a::/64
$ORIGIN a.0.0.0.0.0.0.0.8.b.d.0.1.0.0.2.ip6.arpa.
1.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0 IN PTR host-1.subnet-a.server.zz.
a.b.0.0.0.0.0.0.0.0.0.0.0.0.0.0 IN PTR host-ba.subnet-a.server.zz.
1.3.d.0.0.0.0.0.0.0.0.0.0.0.0.0 IN PTR host-d13.subnet-a.server.zz.
Сравним IPv4 и IPv6:
Параметр IPv4 IPv6
Префикс «классической» зоны /24 Нет такого понятия
Делегирование от RIR Блоками по /24 Одним блоком
Делегирование клиентам По /24 — просто, меньше — со сложностями По 4-битной границе — просто, не по границе — аналогично v4
Использование ORIGIN Можно использовать Можно считать жизненно необходимым
Формирование реверсной записи вручную Можно, несложно Можно, очень трудоемко
Количество файлов зон /24 — один файл (как правило) По вашему выбору, с учетом 4-битной границы
Справедливости ради следует отметить, что возможно и «классическое» делегирование реверсных субзон по любой границе, то есть использование CNAME, о чем сказано на сайте RIPE в разделе FAQ по реверсным зонам.
Вручную формировать реверсные записи для IPv6-адресов достаточно тоскливо, мы решили попытаться облегчить этот процесс и написали небольшой скрипт на Perl, который из IPv6-адреса делает реверсную запись.
Скрипт умеет:
— делать синтаксический анализ IPv6-адреса и сообщать о найденных ошибках — выдавать параметр ORIGIN — формировать на выходе файл, который несложно преобразовать в файл зоны
Файл скрипта называется ip6.pl и должен иметь исполняемый статус, в противном случае нужно вызывать его через интерпретатор Perl. Скрипту передается один параметр — имя файла, содержащего IPv6-адреса в любом возможном формате. В архиве приведено три таких файла:
aaaa-bad — файл с IPv6-адресами, содержащий ошибки (для демонстрационных целей)
aaaa — файл с IPv6-адресами, не содержащими ошибок (тоже для демонстраии)
aaaa-o — файл, используюемый для построения блока реверсных адресов с ORIGIN
Пример вызова скрипта:
./ip6.pl aaaa
После завершения работы скрипта результаты будут выведены на экран, и при успешном завершении будет сформирован файл ip6.arpa с несложной структурой.
Сообщения об ошибках также выдаются на экран.
Некоторые соображения: делать реверсные записи для каждого ipv6-адреса, например при помощи $GENERATE можно, но вряд ли стоит: такая зона будет не просто очень большой, а ОГРОМНОЙ (полагая, что она /64), а про вторую причину, по которой не следует этого делать, может рассказать любой инженер, занимающийся электронной почтой: дело в том, что практически все почтовые серверы настроены так, что проверяют наличие реверсной записи (именно наличие, а не ее совпадение с тем, что идет в заголовке протокола или почты), и при ее отсутствии не принимают почту. Делается это в первую очередь из-за спама от машин, зараженных вирусом. А если у такой машины есть реверсная запись в ДНС, то спам может и пройти.
Другой «автоматизированный» выход — это динамические обновления в DNS. Но, опять-таки, нельзя по вышеизложенной причине создавать такую запись всем подряд. А вот как селектировать нужных клиентов — это вопрос, например можно попытаться создать для них отдельный класс в DNS, хотя по трудоемкости это аналогично ручному конфигурированию реверсных записей. Да и не все DNS-клиенты поддерживают динамические обновления. При использовании DHCP-сервера задачу динамической регистрации клиента в DNS с созданием прямой и реверсной записи можно возложить на этот сервер.
И последнее: появился новый термин — «nibble», он же ПОЛУБАЙТ, то есть 4 бита, граница разумного делегирования зон и подсетей в IPv6.
Сконфигурировать Unbound тоже несложно, в конфигурационный файл нужно добавить строки:
do-ipv6: yes # (можно и не добавлять, по умолчанию это разрешено)
interface: 2001:db8::53:53 #(IPv6-адрес вашего интерфейса, если не задать — то сервер будет работать на всех интерфейсах)
access-control: ::1 allow
access-control: 2001:db8::/32 allow
В последней строке (или в нескольких таких строках) укажите «свои» сети, то есть блоки адресов, из которых разрешены рекурсивные запросы (как и для IPv4).
Вместо директивы interface: можно использовать директиву ip-address:
Также несложно конфигурируется NSD:
do-ip6: yes
interface: 2001:db8::a:b:c # также можно использовать директиву ip-address:
Внутри блоков директив, описывающих зоны, можно также задать IPv6-адреса для трансфера зон по IPv6:
Для secondary zone:
allow-notify: 2001:db8::123:2 # принимать уведомления от этого сервера о появлении изменений в зоне
request-xfr: 2001:db8::123:2 # а отсюда slave должен считывать информацию о зоне
Для primary zone:
notify: 2001:db8::1:73 # уведомлять этот сервер о появлении изменений в зоне
provide-xfr: 2001:db8::1:73 # на этот адрес разрешено отдавать копию зоны для secondary
.. и не забыть перезапустить сервер.