99-origin-dns.sh 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. #!/bin/bash -x
  2. # -*- mode: sh; sh-indentation: 2 -*-
  3. # This NetworkManager dispatcher script replicates the functionality of
  4. # NetworkManager's dns=dnsmasq however, rather than hardcoding the listening
  5. # address and /etc/resolv.conf to 127.0.0.1 it pulls the IP address from the
  6. # interface that owns the default route. This enables us to then configure pods
  7. # to use this IP address as their only resolver, where as using 127.0.0.1 inside
  8. # a pod would fail.
  9. #
  10. # To use this,
  11. # - If this host is also a master, reconfigure master dnsConfig to listen on
  12. # 8053 to avoid conflicts on port 53 and open port 8053 in the firewall
  13. # - Drop this script in /etc/NetworkManager/dispatcher.d/
  14. # - systemctl restart NetworkManager
  15. #
  16. # Test it:
  17. # host kubernetes.default.svc.cluster.local
  18. # host google.com
  19. #
  20. # TODO: I think this would be easy to add as a config option in NetworkManager
  21. # natively, look at hacking that up
  22. cd /etc/sysconfig/network-scripts
  23. . ./network-functions
  24. [ -f ../network ] && . ../network
  25. if [[ $2 =~ ^(up|dhcp4-change|dhcp6-change)$ ]]; then
  26. # If the origin-upstream-dns config file changed we need to restart
  27. NEEDS_RESTART=0
  28. UPSTREAM_DNS='/etc/dnsmasq.d/origin-upstream-dns.conf'
  29. # We'll regenerate the dnsmasq origin config in a temp file first
  30. UPSTREAM_DNS_TMP=`mktemp`
  31. UPSTREAM_DNS_TMP_SORTED=`mktemp`
  32. CURRENT_UPSTREAM_DNS_SORTED=`mktemp`
  33. NEW_RESOLV_CONF=`mktemp`
  34. NEW_NODE_RESOLV_CONF=`mktemp`
  35. ######################################################################
  36. # couldn't find an existing method to determine if the interface owns the
  37. # default route
  38. def_route=$(/sbin/ip route list match 0.0.0.0/0 | awk '{print $3 }')
  39. def_route_int=$(/sbin/ip route get to ${def_route} | awk -F 'dev' '{print $2}' | head -n1 | awk '{print $1}')
  40. def_route_ip=$(/sbin/ip route get to ${def_route} | awk -F 'src' '{print $2}' | head -n1 | awk '{print $1}')
  41. if [[ ${DEVICE_IFACE} == ${def_route_int} ]]; then
  42. if [ ! -f /etc/dnsmasq.d/origin-dns.conf ]; then
  43. cat << EOF > /etc/dnsmasq.d/origin-dns.conf
  44. no-resolv
  45. domain-needed
  46. server=/cluster.local/172.30.0.1
  47. server=/30.172.in-addr.arpa/172.30.0.1
  48. enable-dbus
  49. dns-forward-max=5000
  50. cache-size=5000
  51. min-port=1024
  52. EOF
  53. # New config file, must restart
  54. NEEDS_RESTART=1
  55. fi
  56. # If network manager doesn't know about the nameservers then the best
  57. # we can do is grab them from /etc/resolv.conf but only if we've got no
  58. # watermark
  59. if ! grep -q '99-origin-dns.sh' /etc/resolv.conf; then
  60. if [[ -z "${IP4_NAMESERVERS}" || "${IP4_NAMESERVERS}" == "${def_route_ip}" ]]; then
  61. IP4_NAMESERVERS=`grep '^nameserver[[:blank:]]' /etc/resolv.conf | awk '{ print $2 }'`
  62. fi
  63. ######################################################################
  64. # Write out default nameservers for /etc/dnsmasq.d/origin-upstream-dns.conf
  65. # and /etc/origin/node/resolv.conf in their respective formats
  66. for ns in ${IP4_NAMESERVERS}; do
  67. if [[ ! -z $ns ]]; then
  68. echo "server=${ns}" >> $UPSTREAM_DNS_TMP
  69. echo "nameserver ${ns}" >> $NEW_NODE_RESOLV_CONF
  70. fi
  71. done
  72. # Sort it in case DNS servers arrived in a different order
  73. sort $UPSTREAM_DNS_TMP > $UPSTREAM_DNS_TMP_SORTED
  74. sort $UPSTREAM_DNS > $CURRENT_UPSTREAM_DNS_SORTED
  75. # Compare to the current config file (sorted)
  76. NEW_DNS_SUM=`md5sum ${UPSTREAM_DNS_TMP_SORTED} | awk '{print $1}'`
  77. CURRENT_DNS_SUM=`md5sum ${CURRENT_UPSTREAM_DNS_SORTED} | awk '{print $1}'`
  78. if [ "${NEW_DNS_SUM}" != "${CURRENT_DNS_SUM}" ]; then
  79. # DNS has changed, copy the temp file to the proper location (-Z
  80. # sets default selinux context) and set the restart flag
  81. cp -Z $UPSTREAM_DNS_TMP $UPSTREAM_DNS
  82. NEEDS_RESTART=1
  83. fi
  84. # compare /etc/origin/node/resolv.conf checksum and replace it if different
  85. NEW_NODE_RESOLV_CONF_MD5=`md5sum ${NEW_NODE_RESOLV_CONF}`
  86. OLD_NODE_RESOLV_CONF_MD5=`md5sum /etc/origin/node/resolv.conf`
  87. if [ "${NEW_NODE_RESOLV_CONF_MD5}" != "${OLD_NODE_RESOLV_CONF_MD5}" ]; then
  88. cp -Z $NEW_NODE_RESOLV_CONF /etc/origin/node/resolv.conf
  89. fi
  90. fi
  91. if ! `systemctl -q is-active dnsmasq.service`; then
  92. NEEDS_RESTART=1
  93. fi
  94. ######################################################################
  95. if [ "${NEEDS_RESTART}" -eq "1" ]; then
  96. systemctl restart dnsmasq
  97. fi
  98. # Only if dnsmasq is running properly make it our only nameserver and place
  99. # a watermark on /etc/resolv.conf
  100. if `systemctl -q is-active dnsmasq.service`; then
  101. if ! grep -q '99-origin-dns.sh' /etc/resolv.conf; then
  102. echo "# nameserver updated by /etc/NetworkManager/dispatcher.d/99-origin-dns.sh" >> ${NEW_RESOLV_CONF}
  103. fi
  104. sed -e '/^nameserver.*$/d' /etc/resolv.conf >> ${NEW_RESOLV_CONF}
  105. echo "nameserver "${def_route_ip}"" >> ${NEW_RESOLV_CONF}
  106. if ! grep -qw search ${NEW_RESOLV_CONF}; then
  107. echo 'search cluster.local' >> ${NEW_RESOLV_CONF}
  108. elif ! grep -q 'search cluster.local' ${NEW_RESOLV_CONF}; then
  109. # cluster.local should be in first three DNS names so that glibc resolver would work
  110. sed -i -e 's/^search[[:blank:]]\(.\+\)\( cluster\.local\)\{0,1\}$/search cluster.local \1/' ${NEW_RESOLV_CONF}
  111. fi
  112. cp -Z ${NEW_RESOLV_CONF} /etc/resolv.conf
  113. fi
  114. fi
  115. # Clean up after yourself
  116. rm -f $UPSTREAM_DNS_TMP $UPSTREAM_DNS_TMP_SORTED $CURRENT_UPSTREAM_DNS_SORTED $NEW_RESOLV_CONF
  117. fi