Nginx Setting server_names_hash_max_size and server_names_hash_bucket_size

The list of server names that nginx serves is stored in a hash table for fast lookup. As you increase the number of entries, you have to increase the size of the hash table and/or the number of hash buckets in the table.

Given the nature of your setup, I can't think of any way for you to easily reduce the number of server names you're storing in the table. I will suggest, though, that you not "restart" nginx, but rather simply have it reload its configuration. For instance:

service nginx reload

Just some technical details that I dug form the source code:

  • The general recommendation would be to keep both values as small as possible.
  • If nginx complains increase max_size first as long as it complains. If the number exceeds some big number (32769 for instance), increase bucket_size to multiple of default value on your platform as long as it complains. If it does not complain anymore, decrease max_size back as long as it does not complain. Now you have the best setup for your set of server names (each set of server_names may need different setup).
  • Bigger max_size means more memory consumed (once per worker or server, please comment if you know).
  • Bigger bucket_size means more CPU cycles (for every domain name lookup) and more transfers from main memory to cache.
  • max_size is not related to number of server_names directly, if number of servers doubles, you may need to increase max_size 10 times or even more to avoid collisions. If you cannot avoid them, you have to increase bucket_size.
  • bucket_size is said to be increased to the next power of two, from the source code I would judge it should be enough to make it multiple of default value, this should keep transfers to cache optimal.
  • Average domain name should fit into 32 bytes even with hash array overhead. If you increase bucket_size to 512 bytes, it would accommodate 16 domain names with colliding hash key. This is not something that you want, if collision happens it searches linearly. You want to have as less collisions as possible.
  • If you have max_size less than 10000 and small bucket_size, you can come across long loading time because nginx would try to find optimal hash size in a loop.
  • If you have max_size bigger than 10000, there will be "only" 1000 loops performed before it would complain.

Increase the "server_names_hash_bucket_size" configuration inside your nginx.conf.

I had it 64 and changed to 128.

Problem solved.


@Michael Hampton is absolutely right with his answer. This hash table is constructed and compiled during restart or reload and afterwards it's running very fast. I guess this hash table could grow a lot more without degrading the performance noticeable. But I'd suggest using a size which is power of two, like 4096, due to the nature of C code.