51

How can I get the /etc/hosts file to refer to another configuration file for it's list of hosts?

Example /etc/hosts:

 ## My Hosts
 127.0.0.1   localhost
 255.255.255.255 broadcasthost

 #Other Configurations
 <Link to /myPath/to/MyConfig/ConfigFile.txt>

 #Other Addresses
 3.3.3.3 MyAwesomeDomain.com
 4.4.4.4 SomeplaceIWantToGoTo.com

ConfigFile.txt

##My additional Hosts
1.1.1.1 SomeLocation.com
2.2.2.2 AnotherLocation.com

How do I add a link/Reference to /etc/hosts file such that ConfigFile.txt will be loaded?

tshepang
  • 65,642
DogEatDog
  • 685

5 Answers5

56

You can't. The format for /etc/hosts is quite simple, and doesn't support including extra files.

There are a couple approaches you could use instead:

  • Set up a (possibly local-only) DNS server. Some of these give a lot of flexibility, and you can definitely spread your host files over multiple files, or even machines. Others (such as dnsmasq) offer less (but still sufficient) flexibility, but are easy to set up. If you're trying to include the same list of hosts on a bunch of machines, then DNS is probably the right answer.

  • Set up some other name service (NIS, LDAP, etc.). Check the glibc NSS docs for what is supported. Personally, I think you should use DNS in most all cases.

  • Make yourself an /etc/hosts.d directory or similar, and write some scripts to concatenate them all together (most trivial: cat /etc/hosts.d/*.conf > /etc/hosts, though you'll probably want a little better to e.g., override the default sort by current locale), and run that script at boot, or from cron, or manually whenever you update the files.

Personally, at both home and work, to have machine names resolvable from every device, I run BIND 9. That does involve a few hours to learn, though.

derobert
  • 109,670
  • Thanks! I was wondering why each attempt was failing. It would be so much easier if it were possible, but I'll use the /etc/hosts.d – DogEatDog Jan 07 '13 at 16:53
  • 9
    @DogEatDog I recommend running a local DNS server. Dnsmasq is very easy to set up; on user-friendly distributions, you install the package and it just works. – Gilles 'SO- stop being evil' Jan 07 '13 at 19:17
  • @Gilles has it right - dnsmasq likely does exactly what you want on a home network. It lets you assign static IPs to all your LAN resources by MAC address, grant them hostnames, and resolve requests for them - all while letting unknown hosts pass through to an upstream DNS server. – moodboom Nov 11 '16 at 17:46
  • @moodboom & Gilles Thank you for suggesting that, I've noted it in the answer. Not sure if I was using dnsmasq back in 2013, but I certainly am now. – derobert Nov 11 '16 at 18:05
  • Glob expansions are sorted (at least in Bash, according to LC_COLLATE), so the caveat in your answer does not apply. – l0b0 Mar 20 '18 at 23:56
  • @l0b0 that's a good point, though you probably don't want to sort by LC_COLLATE, at least not without setting it to C or POSIX. Having your config file suddenly in a different order because you changed the system locale would be surprising. Edited the answer to incorporate. – derobert Mar 21 '18 at 01:05
  • @derobert The point is that's what happens, not whether you'd want that or not. – l0b0 Mar 21 '18 at 01:38
  • @l0b0 did you look at my edit to the answer? I think that edit makes it clear that * will sort the files in locale order (what happens, both in bash and per POSIX). – derobert Mar 21 '18 at 01:41
3

It's not possible, as the format is usually deeply coded into platforms' libc. However it is imaginable that an OS adds this feature, making it a non cross-platform solution.

Alternatively, you can automatically update a certain block in your hosts file. This is particularly helpful if you have a script that dynamically outputs host entries for a certain project or so (possibly with changing IPs).

Here's an example: you want to create hosts from Terraform state via terraform-inventory.

Relevant inventory output (for instance, mapping an EC2 "Name" tag to groups of exactly one host each):

$ terraform-inventory --list | jq 'with_entries(select(.key | match("^name_")))'
{
  "name_myhost-a": [
    "10.101.118.131"
  ],
  "name_myhost-b": [
    "10.101.111.189"
  ]
}

print-updated-hosts-entries.sh

#!/bin/sh
exec terraform-inventory --list | \
    jq -r 'to_entries |
           map(select(.key | match("^name_"))) |
           map(.value[0] + " " + .key[5:]) |
           join("\n")'

Script output:

./print-updated-hosts-entries.sh
10.101.118.131 myhost-a
10.101.111.189 myhost-b

And the command line to update a marked block in /etc/hosts with the script output:

sudo cp /etc/hosts "/etc/hosts.bak.$(date +%Y%m%d%H%M%S)" && \
    (
        sed -n '1,/^# MYMARKER BEGIN/{/^# MYMARKER BEGIN/!p;}' /etc/hosts; \
        echo "# MYMARKER BEGIN"; \
        ./print-updated-hosts-entries.sh; \
        echo "# MYMARKER END"; \
        sed -n '/^# MYMARKER END/,${/^# MYMARKER END/!p;}' /etc/hosts; \
    ) | \
    sudo tee /etc/hosts.new | \
    sed -n '/^# MYMARKER BEGIN/,/^# MYMARKER END/p' && \
        sudo mv /etc/hosts.new /etc/hosts

Explanation:

  • The first line obviously creates a backup
  • The subshell in parentheses has two sed calls to print all lines before and after the marker begin/end, respectively. We insert the markers in any case, putting the script output in between those lines. Even if the script fails, we still have to surrounding content of /etc/hosts (and the backup in a catastrophic scenario).
  • sudo tee /etc/hosts.new writes the piped content into a new file
  • sed -n '/^# MYMARKER BEGIN/,/^# MYMARKER END/p' prints the updated block for convenience
  • sudo mv /etc/hosts.new /etc/hosts moves the new file into place. This must be done in a separate step because if the pipe buffer runs out of space, tee /etc/hosts would start writing the file while the existing content is still being read.
AndiDog
  • 241
  • I used the concept in this answer. However, it becomes much simpler if you don't respect the position of the block, and instead move it to the end. I used the following to update the file: sed -i '/# MYMARKER BEGIN/,/# MYMARKER END/d' /etc/hosts; (echo; echo '# MYMARKER BEGIN'; ./print-updated-hosts-entries.sh; echo '# MYMARKER END';) | tee -a /etc/hosts; (sudo will be needed for calling this script) – alx - recommends codidact Apr 23 '21 at 09:41
1

I’ve had a similar requirement for me to work seamlessly from different locations with dynamic IPs, especially since I use virtual machines. My simple solution to this is as follows:

  1. Create a script to update host entries for each entity. E.g., if I have a host called ‘myhost.net’, with sub-domains ‘app.myhost.net’, I would invoke that script and supply an IP address that will be written for each domain and saved in a host file called ‘my host’. The same script has a flag to delete or update a single host file.

  2. Create another script to merge those files into /etc/hosts. This script will find all the host files you’ve created and merge them into the single hosts file at /etc/hosts.

This has worked for me perfectly. I work remotely and sometimes need to move between locations for any reason or another, and would be burdened by the demand to edit that one file so many times. Instead I just run ONE script. Yes, one script. The first script automatically runs the other one to do the merging automatically.

0

I've written a small bash snippet to handle addition and removal of hosts in /etc/hosts. It can be found https://github.com/danfruehauf/Scripts/blob/master/mock-hosts/mock-hosts.sh

Simply include it in your .bashrc, make sure you have permissions to modify /etc/hosts and off you go.

-1

there some tips that more easy :

cretate file named "personal-hosts"

store all custom host on that file.

then apend text file to "hosts" with

cat personal-hosts >> hosts

this simple trick helping me so much to add unblocked add server on super huge host with more than 120k host list