Panji Baskoro

Self Proclaimed DevOps That want to write


How To Install Geoip2 support for Nginx in Rocky 9

Published October 10, 2024

Preface

So after some confusion reading some stackoverflow articles like this one install geoip2 nginx finnaly i understand whats wrong when i tried install the geoip2 instead of geoip in my nginx that run on top of rocky 9. Actually the problem seems like because only few people need to build customized version of nginx and add some random modules with it.

More than that, some deprecated part looks like not well documented. Based on that assumption i will try cover some things that you guys should keep eye on. Let’s start from The ingredients

prerequisites :

Maxminddb

So because i want to add a geoip detection support in my nginx, that’s mean i need a source of truth like a databases that contain or can tell people about from which nationality this User come from based on it’s IP

[root@rocky-9x ~] mmdblookup --file GeoLite2-Country_20241004/GeoLite2-Country.mmdb --ip some.some.some.some

  {
    "continent": 
      {
        "code": 
          "AS" <utf8_string>
        "geoname_id": 
          6255147 <uint32>
        "names": 
          {
            "de": 
              "Asien" <utf8_string>
            "en": 
              "Asia" <utf8_string>
            "es": 
              "Asia" <utf8_string>
            "fr": 
              "Asie" <utf8_string>
            "ja": 
              "アジア" <utf8_string>
            "pt-BR": 
              "Ásia" <utf8_string>
            "ru": 
              "Азия" <utf8_string>
            "zh-CN": 
              "亚洲" <utf8_string>
          }
      }
    "country": 
      {
        "geoname_id": 
          1643084 <uint32>
        "iso_code": 
          "ID" <utf8_string>
        "names": 
          {
            "de": 
              "Indonesien" <utf8_string>
            "en": 
              "Indonesia" <utf8_string>
            "es": 
              "Indonesia" <utf8_string>
            "fr": 
              "Indonésie" <utf8_string>
            "ja": 
              "インドネシア共和国" <utf8_string>
            "pt-BR": 
              "Indonésia" <utf8_string>
            "ru": 
              "Индонезия" <utf8_string>
            "zh-CN": 
              "印度尼西亚" <utf8_string>
          }
      }
    "registered_country": 
      {
        "geoname_id": 
          1643084 <uint32>
        "iso_code": 
          "ID" <utf8_string>
        "names": 
          {
            "de": 
              "Indonesien" <utf8_string>
            "en": 
              "Indonesia" <utf8_string>
            "es": 
              "Indonesia" <utf8_string>
            "fr": 
              "Indonésie" <utf8_string>
            "ja": 
              "インドネシア共和国" <utf8_string>
            "pt-BR": 
              "Indonésia" <utf8_string>
            "ru": 
              "Индонезия" <utf8_string>
            "zh-CN": 
              "印度尼西亚" <utf8_string>
          }
      }
  }

This is where Maxminddb taking places, so in short nginx officially support geoip checking with maxminddb as source of data, we can do a lookup checking etc based on ip and file that we download from maxmind you can download data based on city, country region etc. with various format such as mmdb and csv (for nginx mmdb format will be used).

If just wanna take a look how it’s look like using lite version is feasible just register to https://www.maxmind.com/en/account/sign-in and get to download page then download mmdb file

Sreenshot package

Rocky 9 towards maxmind

so when you want to build nginx from source and add geoip support with it, it’s already well known that we should add maxmind db geoip and a development package that allow us to call the geoip db to our nginx. The first time i try to install geoip2 i was go with GeoIP2 and GeoIP2-devel since i using rocky, and… i didn’t find any of them.

After browse some thread maybe related to it, i land on this forum rocky 9 forum pages and someone said that this package already deprecated and moved to libmaxminddb and libmaxminddb-devel but another things coming. When i tried to install it directly i can’t find it (still) after manually browse the package, turns out we need to enable CRB (Code Ready Builder) repo for rocky 9. Don’t try to install the obsolete version through remi repo or something else, it will cause collision in some cases. You can check the content of CRB here : crb rocky content Sreenshot package

you can enable it using :

yum-config-manager enable crb

Once installation finish you can just go follow the guide for install GeoIP2 support for nginx, but to make sure all works well try to run these command. It should return a json data contain geo location of ip, depends on what version downloaded (city name, country name and code with 3 character etc. )

ngx_http_geoip2_module introduction

Like we all know, nginx need a some kind of “connector” that help it to gain some feature that owned by Operating System or some specific library to be specific. So for this purpose nginx hace several modules that contain a code that consume or implemented some Library or Operating System feature like syscall etc.

If we want to use geoip2 we need to install ngx_http_geoip2_module module, you can find the source here https://github.com/leev/ngx_http_geoip2_module. We can also read the docs because somehow this repo docs more straight forward and i gain some information from it.

In my case i want this module become static module, all i need is adding this flag to my build script.

after add ngx_http_geoip2_module as static modules you don’t need to call module manually via load_module because the modules already loaded by nginx

Nginx Build From Source

Actually this section will be straight to how to run it, but i’ll try to describe some packages that will be needed with assumption the crb repo already activated

./configure --add-module=~/nginx/ngx_http_geoip2_module --with-stream
make
make install

When configure command is success the output will be like this

...
...
checking for MaxmindDB library ... found
 + ngx_geoip2_module was configured
checking for PCRE2 library ... found
checking for zlib library ... found
creating objs/Makefile

Configuration summary
  + using system PCRE2 library
  + OpenSSL library is not used
  + using system zlib library

  nginx path prefix: "/usr/local/nginx"
  nginx binary file: "/usr/local/nginx/sbin/nginx"
  nginx modules path: "/usr/local/nginx/modules"
  nginx configuration prefix: "/usr/local/nginx/conf"
  nginx configuration file: "/usr/local/nginx/conf/nginx.conf"
  nginx pid file: "/usr/local/nginx/logs/nginx.pid"
  nginx error log file: "/usr/local/nginx/logs/error.log"
  nginx http access log file: "/usr/local/nginx/logs/access.log"
  nginx http client request body temporary files: "client_body_temp"
  nginx http proxy temporary files: "proxy_temp"
  nginx http fastcgi temporary files: "fastcgi_temp"
  nginx http uwsgi temporary files: "uwsgi_temp"
  nginx http scgi temporary files: "scgi_temp"

And just continue to make if configure command succeeded and will be got output like this

...
...
objs/src/stream/ngx_stream_upstream_random_module.o \
objs/src/stream/ngx_stream_upstream_zone_module.o \
objs/addon/ngx_http_geoip2_module/ngx_http_geoip2_module.o \
objs/addon/ngx_http_geoip2_module/ngx_stream_geoip2_module.o \
objs/ngx_modules.o \
-lcrypt -lmaxminddb -lmaxminddb -lpcre2-8 -lz \
-Wl,-E
sed -e "s|%%PREFIX%%|/usr/local/nginx|" \
	-e "s|%%PID_PATH%%|/usr/local/nginx/logs/nginx.pid|" \
	-e "s|%%CONF_PATH%%|/usr/local/nginx/conf/nginx.conf|" \
	-e "s|%%ERROR_LOG_PATH%%|/usr/local/nginx/logs/error.log|" \
	< man/nginx.8 > objs/nginx.8
make[1]: Leaving directory '/root/nginx/nginx-1.27.0'

and this for make install

...
...
cp conf/scgi_params \
	'/usr/local/nginx/conf/scgi_params.default'
test -f '/usr/local/nginx/conf/nginx.conf' \
	|| cp conf/nginx.conf '/usr/local/nginx/conf/nginx.conf'
cp conf/nginx.conf '/usr/local/nginx/conf/nginx.conf.default'
test -d '/usr/local/nginx/logs' \
	|| mkdir -p '/usr/local/nginx/logs'
test -d '/usr/local/nginx/logs' \
	|| mkdir -p '/usr/local/nginx/logs'
test -d '/usr/local/nginx/html' \
	|| cp -R html '/usr/local/nginx'
test -d '/usr/local/nginx/logs' \
	|| mkdir -p '/usr/local/nginx/logs'
make[1]: Leaving directory '/root/nginx/nginx-1.27.0'

As you see above the module successfully installed when you got ngx_geoip2_module was configured messages when run configure command and as i told above i intend to install it as a static module so we don’t need mention or load module manually inside our config and the nginx have been builded inside /usr/local/nginx/sbin but you can specify the default directory with --prefix= and --sbin-path= or for more param take a look at this http://nginx.org/en/docs/configure.html

To run it just execute the executable file and run /usr/local/nginx/sbin/nginx -V you will got output like this ( I installed it with stream module, you know just in case ;) )

[root@rocky-9x nginx-1.27.0] /usr/local/nginx/sbin/nginx -V
nginx version: nginx/1.27.0
built by gcc 11.4.1 20231218 (Red Hat 11.4.1-3) (GCC) 
configure arguments: --add-module=/root/nginx/ngx_http_geoip2_module --with-stream

You can check your nginx already runnin or not with access your machine via web browser or curl command Web Accessed from internet

GeoIP setup

If you already read this far, i assume maxminddb and other things that needed to done this already prepared and installed. Actually after yapping that much this steps is quiet simple. When setup GeoIP all you need is setup header of nginx and some variables. So here is my config :

http{
...
geoip2 /usr/share/GeoIP/GeoLite2-Country.mmdb {
    auto_reload 60m;
    $geoip2_metadata_country_build metadata build_epoch;
    $geoip2_data_country_code country iso_code;
    $geoip2_data_country_name country names en;
}

fastcgi_param COUNTRY_CODE $geoip2_data_country_code;
fastcgi_param COUNTRY_NAME $geoip2_data_country_name;
...
}

after apply above config, try rung nginx -t and make sure no error appear. If your nginx got warning messages and it’s not related to geoip just ignore it but if it related to geoip try check it and look for it in google. Then reload or restart nginx to make sure changes of config applied.

One interesting part is when you build nginx with geoip modules, automatically you can see it’s will add .mmdb file by it self. Now let’s test it

Testing : set header and set log for geolocation

So after a long yapping sessions, Next it to proof that our work is done. Let’s check it already applied or not. So actually there is many way to apply it. But the simplest way that i found in internet is to change header of responses with this config;

http {
...
## for header 
add_header "X-country-code" "$geoip2_data_country_code";

## for logging contry code 

log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
                      '$status $body_bytes_sent "$http_referer" '
                      '"$http_user_agent" "$http_x_forwarded_for" - ehe "$geoip2_data_country_name" - "$geoip2_data_country_code"';

access_log  logs/access.log  main;

...
}

add this config inside the http directive then reload or restart your nginx services, then lets try it;

panji@machine :: curl -I http://some.some.some.some/
HTTP/1.1 200 OK
Server: nginx/1.27.0
Date: Sun, 06 Oct 2024 09:10:14 GMT
Last-Modified: Sun, 06 Oct 2024 08:49:39 GMT
Connection: keep-alive
ETag: "67024f23-293"
X-country-code: ID
Accept-Ranges: bytes
Content-Type: text/html
Content-Length: 659

you can see that the header that we add before mentioned to the responses. Next i try curl from various nation origin :

Hit from germany this one from germany

Hit from France this one from france

Hit from US last one from US

Also we can show from log like these :

nginx geo ip log Nginx log

Done that’s it

Conclusion

So when you need to do some kind of circus attraction sometimes to obtain something in nginx. Geo Location come handy when you need either apply a rate limiting rules with nationality constraint or maybe you just wanna redirect user from exact country to specific pages of your sites. But don’t limit your self to this kind of approach i believe there is many way to do them both. What your thoughts about it?

Cheers !