Fast offline IP-to-country lookup using RIR data with country names and currency support
Project description
IPMapper - Offline IP-to-Country Lookup
Fast, offline IP geolocation using Regional Internet Registry (RIR) data
No API calls • No rate limits • No signup required • Download once, use forever
Motivation
IPMapper addresses the need for reliable, offline IP geolocation without dependencies on external APIs or services. Built entirely on public Regional Internet Registry (RIR) data, it provides:
- Payment processing: Determine customer currency from IP for automatic payment localization
- Feature restrictions: Implement geo-blocking or feature availability based on country codes
- Compliance: Meet data residency requirements by identifying user locations
- Analytics: Understand user geographic distribution without third-party services
- Offline environments: Work in air-gapped systems or areas with limited internet connectivity
Core Features
Data Sources
- 100% Public Data: Uses only official RIR delegated files from APNIC, ARIN, RIPE NCC, LACNIC, and AFRINIC
- No External Dependencies: Operates completely offline after initial data download
- Always Free: No API keys, accounts, or subscription fees required
Geographic Information
- Country Codes: ISO 3166-1 alpha-2 country codes (US, DE, JP, etc.)
- Country Names: Full country names ("United States of America")
- Currency Codes: ISO 4217 currency codes (USD, EUR, JPY, etc.)
- Dual Stack: Full IPv4 and IPv6 support
Technical Implementation
- Radix Tree Lookup: O(prefix length) complexity for sub-microsecond lookups
- Prefix Aggregation: 30-70% reduction in dataset size while maintaining accuracy
- Memory Efficient: Optimized data structures for minimal RAM usage
- Auto-loading: Data loads automatically on first use
Technical Architecture
Radix Tree Implementation
IPMapper uses a radix tree (compressed trie) for IP prefix lookups, providing optimal search complexity:
Time Complexity: O(k) where k = IP address bit length (32 for IPv4, 128 for IPv6)
Space Complexity: O(n×k) where n = number of unique prefixes
Why Radix Trees?
- Longest Prefix Matching: Automatically finds the most specific route for any IP
- Memory Efficient: Shared prefixes are stored only once
- Cache Friendly: Tree traversal exhibits good spatial locality
- Predictable Performance: Lookup time depends only on address length, not dataset size
Prefix Aggregation
The library implements CIDR prefix aggregation to optimize storage and lookup performance:
Aggregation Process:
- Grouping: Group prefixes by country code and IP version
- Sorting: Sort prefixes by network address for efficient processing
- Merging: Use
ipaddress.collapse_addresses()to merge adjacent prefixes - Validation: Ensure aggregation preserves country code boundaries
Benefits:
- Reduced Memory: 30-70% fewer prefixes to store
- Faster Loading: Less data to process during initialization
- Maintained Accuracy: No loss of geographic precision
- Better Cache Utilization: Fewer tree nodes improve cache hit rates
Installation
# Install from PyPI
pip install ipmapper
# Or install from source
git clone https://github.com/anxkhn/ipmapper
cd ipmapper
pip install -e .
Quick Start
First Run - Download Data
# Download and process RIR data (run once or when updating)
ipmapper update
Command Line Usage
# Basic lookup
ipmapper lookup 8.8.8.8
ipmapper lookup 2001:4860:4860::8888
# Multiple IPs with additional data
ipmapper lookup 8.8.8.8 1.1.1.1 --country-name --currency
# Quick shortcuts
ipmapper country 8.8.8.8 # Just country name
ipmapper country_code 8.8.8.8 # Just country code
ipmapper currency 8.8.8.8 # Just currency
# Output formats
ipmapper lookup 8.8.8.8 --format json
ipmapper lookup 8.8.8.8 --format csv
# Check status
ipmapper status
Python Library Usage
import ipmapper
# Simple lookups (auto-loads data on first use)
result = ipmapper.lookup('8.8.8.8')
print(result)
# {'ip': '8.8.8.8', 'country_code': 'US', 'country_name': 'United States of America', 'currency': 'USD'}
# Direct attribute access
country_name = ipmapper.get_country_name('8.8.8.8') # 'United States of America'
country_code = ipmapper.get_country_code('8.8.8.8') # 'US'
currency = ipmapper.get_country_currency('8.8.8.8') # 'USD'
# Selectively use IPv4 or IPv6 tree with IP version hints
result = ipmapper.lookup('192.168.1.1', ip_version='ipv4') # Skip IPv6 tree
result = ipmapper.lookup('2001:db8::1', ip_version='ipv6') # Skip IPv4 tree
# Advanced usage with custom data directory
lookup_engine = ipmapper.IPLookup(data_dir='/custom/path')
result = lookup_engine.lookup_full('8.8.8.8')
Data Sources & Processing
Regional Internet Registries
| RIR | Region | Data Source |
|---|---|---|
| APNIC | Asia-Pacific | delegated-apnic-extended-latest |
| ARIN | North America | delegated-arin-extended-latest |
| RIPE NCC | Europe/Middle East/Central Asia | delegated-ripencc-extended-latest |
| LACNIC | Latin America/Caribbean | delegated-lacnic-extended-latest |
| AFRINIC | Africa | delegated-afrinic-extended-latest |
Processing Pipeline
- Download: Fetch latest delegated files from all 5 RIRs
- Parse: Extract IPv4/IPv6 allocations with country codes
- Filter for
allocatedandassignedstatus only - Convert IPv4 address counts to CIDR blocks
- Parse IPv6 prefix lengths directly
- Filter for
- Deduplicate: Resolve overlapping entries
- Prefer most recent allocation date
- Use lexicographic registry ordering for ties
- Aggregate: Optimize prefix lists using CIDR aggregation
- Generate: Create optimized CSV files and metadata
CLI Reference
Core Commands
ipmapper update - Download and process RIR data
ipmapper update [OPTIONS]
Options:
--force Force re-download even if data exists
--data-dir Custom data directory (default: ~/.ipmapper)
ipmapper lookup - Look up IP addresses
ipmapper lookup [OPTIONS] IP1 [IP2 ...]
Options:
--format [table|json|csv] Output format (default: table)
--country-name Include country names
--currency Include currency codes
--data-dir Custom data directory
ipmapper status - Show local data status
ipmapper status [OPTIONS]
Options:
--data-dir Custom data directory
Quick Commands
ipmapper country IP # Get country name
ipmapper country_code IP # Get country code
ipmapper currency IP # Get currency code
Output Files
After running ipmapper update, these files are generated:
prefixes_ipv4_agg.csv- Aggregated IPv4 prefixes (format:cidr,country_code)prefixes_ipv6_agg.csv- Aggregated IPv6 prefixes (format:cidr,country_code)metadata.json- Source URLs, timestamps, checksums, and statistics
Note: Only aggregated files are generated for optimal performance. Raw files are cleaned up automatically.
Use Cases
Payment Processing
import ipmapper
def determine_currency(client_ip):
"""Automatically set payment currency based on client location"""
currency = ipmapper.get_country_currency(client_ip)
return currency or 'USD' # Default to USD
# Usage
client_ip = request.remote_addr
payment_currency = determine_currency(client_ip)
Feature Restrictions
import ipmapper
RESTRICTED_COUNTRIES = {'CN', 'IR', 'KP'} # Example restrictions
def check_feature_availability(client_ip, feature_name):
"""Check if feature is available in client's country"""
country_code = ipmapper.get_country_code(client_ip)
if country_code in RESTRICTED_COUNTRIES:
return False
return True
Analytics & Compliance
import ipmapper
from collections import Counter
def analyze_user_geography(ip_list):
"""Analyze geographic distribution of users"""
countries = []
for ip in ip_list:
country = ipmapper.get_country_name(ip)
if country:
countries.append(country)
return Counter(countries)
Development
# Clone repository
git clone https://github.com/anxkhn/ipmapper
cd ipmapper
# Install in development mode
pip install -e .
# Run from source
python -m ipmapper update
python -m ipmapper lookup 8.8.8.8
Why Always Free
This project will always remain free and open source because:
| Public Data Foundation | Built entirely on publicly available RIR allocation data |
| Educational Mission | Helps people understand internet infrastructure and IP allocation |
| Community Driven | Created for developers, by developers, with no commercial agenda |
| Transparency First | All code, data sources, and methodologies are completely open |
| Self-Contained | No ongoing infrastructure costs or API maintenance requirements |
License
MIT License - see LICENSE file for details.
Data License
The IP allocation data is derived from public RIR delegated files, available under their respective terms of use. This library inherits those terms for the data while the code remains MIT licensed.
Related Projects
- GeoLite2 - MaxMind's free geolocation database
- ip2location-lite - Commercial IP geolocation with city-level data
- python-geoip - Another Python IP geolocation library
Support
- Issues: GitHub Issues
- Discussions: GitHub Discussions
- Email: Create an issue for fastest response
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file ipmapper-1.0.1.tar.gz.
File metadata
- Download URL: ipmapper-1.0.1.tar.gz
- Upload date:
- Size: 24.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.11.11
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d321a6f2851a2d23e9d92e762a805f04ac29c31497acb306c3f89b8014c55947
|
|
| MD5 |
af0a9612897f3f3ad8d40ff3e7d16c40
|
|
| BLAKE2b-256 |
06d2d190868300c90a82ee8595bf84382762e2b9eff946ad6a1dbb4045594798
|
File details
Details for the file ipmapper-1.0.1-py3-none-any.whl.
File metadata
- Download URL: ipmapper-1.0.1-py3-none-any.whl
- Upload date:
- Size: 22.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.11.11
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
130621d151cea2a0450d31a844a1518983f0e8bd258bd06de70f2892f6c7709d
|
|
| MD5 |
a3bd5af4620eae8d3cd8541373825201
|
|
| BLAKE2b-256 |
987a3d7b00593dd9d1a96037e90eb68244a394339cdb0729f66f64c00036bf95
|