#!/usr/bin/perl -w # # Module: vyatta_update_resolv.pl # # **** License **** # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License version 2 as # published by the Free Software Foundation. # # This program 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. # # This code was originally developed by Vyatta, Inc. # Portions created by Vyatta are Copyright (C) 2007 Vyatta, Inc. # All Rights Reserved. # # Author: Marat Nepomnyashy # Date: December 2007 # Description: Script to update '/etc/resolv.conf' on commit of 'system domain-search domain' config. # # **** End License **** # use strict; use lib "/opt/vyatta/share/perl5/"; use Getopt::Long; my $change_dir = ''; my $modify_dir = ''; my $dhclient_script = 0; GetOptions("change_dir=s" => \$change_dir, "modify_dir=s" => \$modify_dir, "dhclient-script=i" => \$dhclient_script ); use Vyatta::Config; my $vc = new Vyatta::Config(); if ($change_dir ne '') { $vc->{_changes_only_dir_base} = $change_dir; } if ($modify_dir ne '') { $vc->{_new_config_dir_base} = $modify_dir; } $vc->setLevel('system'); my @domains; my $domain_name = undef; if ($dhclient_script == 1) { $vc->{_active_dir_base} = "/opt/vyatta/config/active/"; @domains = $vc->returnOrigValues('domain-search domain'); $domain_name = $vc->returnOrigValue('domain-name'); } else { @domains = $vc->returnValues('domain-search domain'); $domain_name = $vc->returnValue('domain-name'); } if ($dhclient_script == 0 && @domains > 0 && $domain_name && length($domain_name) > 0) { print STDERR "System configuration error. Both \'domain-name\' and \'domain-search\' are specified, but only one of these mutually exclusive parameters is allowed.\n"; print STDERR "System configuration commit aborted due to error(s).\n"; exit(1); } my $doms = ''; foreach my $domain (@domains) { if (length($doms) > 0) { $doms .= ' '; } $doms .= $domain; } # add domain names received from dhcp client to domain search in /etc/resolv.conf if domain-name not set in CLI if (!defined($domain_name)) { my @dhcp_interfaces_resolv_files = `ls /etc/ | grep resolv.conf.dhclient-new`; if ($#dhcp_interfaces_resolv_files >= 0) { for my $each_file (@dhcp_interfaces_resolv_files) { chomp $each_file; my $find_search = `grep "^search" /etc/$each_file 2> /dev/null | wc -l`; if ($find_search == 1) { my $search_string = `grep "^search" /etc/$each_file`; my @dhcp_domains = split(/\s+/, $search_string, 2); my $dhcp_domain = $dhcp_domains[1]; chomp $dhcp_domain; $doms .= ' ' . $dhcp_domain; } } } } my $search = ''; if (length($doms) > 0) { $search = "search\t\t$doms\t\t#line generated by $0\n"; } my $domain = ''; if ($domain_name && length($domain_name) > 0) { $domain = "domain\t\t$domain_name\t\t#line generated by $0\n"; } # update /etc/resolv.conf for name-servers received from dhcp client, only done when dhclient-script calls this script if ($dhclient_script == 1) { my @current_dhcp_nameservers; my $restart_ntp = 0; # code below to add new name-servers received from dhcp client my @dhcp_interfaces_resolv_files = `ls /etc/ | grep resolv.conf.dhclient-new`; if ($#dhcp_interfaces_resolv_files >= 0) { my $ns_count = 0; for my $each_file (@dhcp_interfaces_resolv_files) { chomp $each_file; my $find_nameserver = `grep nameserver /etc/$each_file 2> /dev/null | wc -l`; if ($find_nameserver > 0) { my @nameservers = `grep nameserver /etc/$each_file`; for my $each_nameserver (@nameservers) { my @nameserver = split(/ /, $each_nameserver, 2); my $ns = $nameserver[1]; chomp $ns; $current_dhcp_nameservers[$ns_count] = $ns; $ns_count++; my @search_ns_in_resolvconf = `grep $ns /etc/resolv.conf`; my $ns_in_resolvconf = 0; if (@search_ns_in_resolvconf > 0) { foreach my $ns_resolvconf (@search_ns_in_resolvconf) { my @resolv_ns = split(/\s+/, $ns_resolvconf); my $final_ns = $resolv_ns[1]; chomp $final_ns; if ($final_ns eq $ns) { $ns_in_resolvconf = 1; } } } if ($ns_in_resolvconf == 0) { open (APPEND, ">>/etc/resolv.conf") or die "$! error trying to overwrite"; print APPEND "nameserver\t$ns\t\t#nameserver written by $0\n"; close (APPEND); $restart_ntp = 1; } } } } } # code below to remove old name-servers from /etc/resolv.conf that were not received in this response from dhcp-server my @nameservers_dhcp_in_resolvconf = `grep 'nameserver written' /etc/resolv.conf`; my @dhcp_nameservers_in_resolvconf; my $count_nameservers_in_resolvconf = 0; for my $count_dhcp_nameserver (@nameservers_dhcp_in_resolvconf) { my @dhcp_nameserver = split(/\t/, $count_dhcp_nameserver, 3); $dhcp_nameservers_in_resolvconf[$count_nameservers_in_resolvconf] = $dhcp_nameserver[1]; $count_nameservers_in_resolvconf++; } if ($#current_dhcp_nameservers < 0) { for my $dhcpnameserver (@dhcp_nameservers_in_resolvconf) { my $cmd = "sed -i '/$dhcpnameserver\t/d' /etc/resolv.conf"; system($cmd); $restart_ntp = 1; } } else { for my $dhcpnameserver (@dhcp_nameservers_in_resolvconf) { my $found = 0; for my $currentnameserver (@current_dhcp_nameservers) { if ($dhcpnameserver eq $currentnameserver){ $found = 1; } } if ($found == 0) { my $cmd = "sed -i '/$dhcpnameserver\t/d' /etc/resolv.conf"; system($cmd); $restart_ntp = 1; } } } if ($restart_ntp == 1) { # this corresponds to what is done in name-server/node.def as a fix for bug 1300 my $cmd_ntp_restart = "if [ -f /etc/ntp.conf ] && grep -q 'server' /etc/ntp.conf; then /usr/sbin/invoke-rc.d ntp restart >&/dev/null; fi &"; system($cmd_ntp_restart); } } # The following will re-write '/etc/resolv.conf' line by line, # replacing the 'search' specifier with the latest values, # or replacing the 'domain' specifier with the latest value. my @resolv; if (-e '/etc/resolv.conf') { open (RESOLV, '</etc/resolv.conf') or die("$0: Error! Unable to open '/etc/resolv.conf' for input: $!\n"); @resolv = <RESOLV>; close (RESOLV); } my $foundSearch = 0; my $foundDomain = 0; open (RESOLV, '>/etc/resolv.conf') or die("$0: Error! Unable to open '/etc/resolv.conf' for output: $!\n"); foreach my $line (@resolv) { if ($line =~ /^search\s/) { $foundSearch = 1; if (length($search) > 0) { print RESOLV $search; } } elsif ($line =~ /^domain\s/) { $foundDomain = 1; if (length($domain) > 0) { print RESOLV $domain; } } else { print RESOLV $line; } } if ($foundSearch == 0 && length($search) > 0) { print RESOLV $search; } if ($foundDomain == 0 && length($domain) > 0) { print RESOLV $domain; } close (RESOLV);