Intro
Elixir is a dynamic, functional programming language designed for building scalable and maintainable applications. It runs on the Erlang Virtual Machine (BEAM), which is known for its low-latency, distributed, and fault-tolerant systems – traits that make Elixir especially suitable for real-time and concurrent applications.
For this tutorial, we will demonstrate how easy it is to include the IP2Location Erlang library as a dependency in an Elixir project. You can also take a look at the IP2Location Erlang codes at its GitHub repo. We will be using the IP2Location DB26 IPv6 BIN file and using the codes to query geolocation data for IPv4 or IPv6 addresses.
Pre-requisites
You’ll need to have Erlang and Elixir installed. Please follow https://elixir-lang.org/install.html for the installation steps. As the IP2Location Erlang library depends on the IP2Location BIN files, please download whichever BIN file you wish to use. You can purchase the commercial BIN files from https://www.ip2location.com/database/ip2location or download the free LITE BIN file from https://lite.ip2location.com/ip2location-lite but please note that the commercial BIN files have higher accuracy. Our demo will be on a Debian Linux machine so some of the steps will be specifically for that platform.
Creating the Elixir project
Navigate to the folder where you wish to create your project, then run the below command to create the project folder. We’ll call our demo project ip2locationdemo.
mix new ip2locationdemo

Adding the IP2Location Erlang as a dependency
Now, navigate into the demo project folder and edit the mix.exs file.
cd ip2locationdemo nano mix.exs
You’ll see the below.

Let’s insert the below line into the deps section and save the file. This will add the IP2Location Erlang library as a dependency.
{:ip2location_erlang, "~> 8.6.0"}

NOTE: You may need to update the version IP2Location Erlang library if there is a newer version.
Now run the below to actually fetch the dependency.
mix deps.get

Download and extract the IP2Location BIN file
As mentioned above, you’ll need to either get the commercial or LITE BIN file. In our case, we’ll be using the IP2Location DB26 IPV6 BIN file. The BIN file will be in the root of the demo project folder.

Modify the test code
Let’s overwrite the generated test code in lib/ip2locationdemo.ex and replace it with our code below.
nano lib/ip2locationdemo.ex
defmodule Ip2locationdemo do
  def lookup(ip) do
    bin_file = "IPV6-COUNTRY-REGION-CITY-LATITUDE-LONGITUDE-ZIPCODE-TIMEZONE-ISP-DOMAIN-NETSPEED-AREACODE-WEATHER-MOBILE-ELEVATION-USAGETYPE-ADDRESSTYPE-CATEGORY-DISTRICT-ASN.BIN"
    # Load the database
    :ip2location.new(bin_file)
    # Query an IP address
    result = :ip2location.query(ip)
    # Access specific fields
    # IO.inspect(result)
    {:ip2locationrecord, country_short, country_long, region, city, isp, latitude, longitude, domain, zip_code, time_zone, net_speed, idd_code, area_code, weather_station_code, weather_station_name, mcc, mnc, mobile_brand, elevation, usage_type, address_type, category, district, asn, as} = result
    IO.puts("country_short: #{to_string(country_short)}")
    IO.puts("country_long: #{to_string(country_long)}")
    IO.puts("region: #{to_string(region)}")
    IO.puts("city: #{to_string(city)}")
    IO.puts("isp: #{to_string(isp)}")
    IO.puts("latitude: #{to_string(latitude)}")
    IO.puts("longitude: #{to_string(longitude)}")
    IO.puts("domain: #{to_string(domain)}")
    IO.puts("zip_code: #{to_string(zip_code)}")
    IO.puts("time_zone: #{to_string(time_zone)}")
    IO.puts("net_speed: #{to_string(net_speed)}")
    IO.puts("idd_code: #{to_string(idd_code)}")
    IO.puts("area_code: #{to_string(area_code)}")
    IO.puts("weather_station_code: #{to_string(weather_station_code)}")
    IO.puts("weather_station_name: #{to_string(weather_station_name)}")
    IO.puts("mcc: #{to_string(mcc)}")
    IO.puts("mnc: #{to_string(mnc)}")
    IO.puts("mobile_brand: #{to_string(mobile_brand)}")
    IO.puts("elevation: #{to_string(elevation)}")
    IO.puts("usage_type: #{to_string(usage_type)}")
    IO.puts("address_type: #{to_string(address_type)}")
    IO.puts("category: #{to_string(category)}")
    IO.puts("district: #{to_string(district)}")
    IO.puts("asn: #{to_string(asn)}")
    IO.puts("as: #{to_string(as)}")
  end 
end

Launch the interactive shell to run the code
iex -S mix
Then, run the below to query geolocation for the IP 8.8.8.8.
Ip2locationdemo.lookup(to_charlist("8.8.8.8"))

That’s how easy it is to query the IP2Location geolocation data using Elixir with the IP2Location Erlang library.
