farzai / geonames
A PHP library for downloading and converting Geonames data.
Fund package maintenance!
parsilver
Requires
- php: ^8.1
- guzzlehttp/guzzle: ^7.0
- symfony/console: ^6.0
Requires (Dev)
- laravel/pint: ^1.2
- pestphp/pest: ^2.8
- phpstan/extension-installer: ^1.1
- phpstan/phpstan-deprecation-rules: ^1.0
- phpstan/phpstan-phpunit: ^1.0
- spatie/ray: ^1.28
Suggests
- ext-zip: Required to extract the downloaded data.
- guzzlehttp/guzzle: Required to download the data from Geonames.
- mongodb/mongodb: Required for MongoDB output format support.
README
A PHP library for downloading and converting Geonames data. This library provides easy access to both Postal Codes and Gazetteer data from Geonames.
Features
- Download postal codes data for specific countries or all countries
- Download detailed geographical data (Gazetteer) including administrative divisions
- Convert data to JSON format with proper structure
- Import data directly to MongoDB with proper indexing
- Memory-efficient processing for large datasets
- Progress bars for all operations
- Support for filtering by feature types (for Gazetteer data)
Installation
composer require farzai/geonames
Usage
Postal Codes
Download postal codes for a specific country:
./bin/geonames geonames:download TH
Download postal codes for all countries:
./bin/geonames geonames:download all
Options:
--output (-o)
: Output directory (default: ./data)--format (-f)
: Output format (default: json, options: json, mongodb)--mongodb-uri
: MongoDB connection URI (default: mongodb://localhost:27017)--mongodb-db
: MongoDB database name (default: geonames)--mongodb-collection
: MongoDB collection name (default: postal_codes)
The postal codes data includes:
- Country code
- Postal code
- Place name
- Administrative divisions (state/province, county/district, community)
- Latitude and longitude
- Accuracy level
Gazetteer Data
Download geographical data for a specific country:
./bin/geonames geonames:gazetteer:download TH
Download geographical data for all countries:
./bin/geonames geonames:gazetteer:download all
Options:
--output (-o)
: Output directory (default: ./data)--format (-f)
: Output format (default: json, options: json, mongodb)--feature-class (-c)
: Filter by feature class (default: P)--mongodb-uri
: MongoDB connection URI (default: mongodb://localhost:27017)--mongodb-db
: MongoDB database name (default: geonames)--mongodb-collection
: MongoDB collection name (default: gazetteer)
Available feature classes:
A
: Country, state, regionH
: Stream, lakeL
: Parks, areaP
: City, villageR
: Road, railroadS
: Spot, building, farmT
: Mountain, hill, rockU
: UnderseaV
: Forest, heath
The Gazetteer data includes:
- Geoname ID
- Name (with ASCII and alternate names)
- Geographical coordinates
- Feature class and code
- Administrative divisions with names
- Population
- Elevation
- Digital elevation model (DEM)
- Timezone
- Modification date
Data Structure
Postal Codes Structure
JSON Format
{ "country_code": "TH", "postal_code": "10200", "place_name": "Bang Rak", "admin_name1": "Bangkok", "admin_code1": "10", "admin_name2": "", "admin_code2": "", "admin_name3": "", "admin_code3": "", "latitude": 13.7235, "longitude": 100.5147, "accuracy": 1 }
MongoDB Format
In MongoDB, the postal codes data has the same structure as JSON but includes an additional location
field for geospatial queries:
{ "country_code": "TH", "postal_code": "10200", "place_name": "Bang Rak", "admin_name1": "Bangkok", "admin_code1": "10", "admin_name2": "", "admin_code2": "", "admin_name3": "", "admin_code3": "", "latitude": 13.7235, "longitude": 100.5147, "accuracy": 1, "location": { "type": "Point", "coordinates": [100.5147, 13.7235] } }
The MongoDB collection is indexed for efficient queries:
- Compound index on
country_code
andpostal_code
(unique) - Index on
country_code
- Index on
postal_code
- Geospatial index on
location
Gazetteer Structure
JSON Format
{ "geoname_id": 1609350, "name": "Bangkok", "ascii_name": "Bangkok", "alternate_names": ["Krung Thep", "กรุงเทพมหานคร"], "latitude": 13.75, "longitude": 100.51667, "feature_class": "P", "feature_code": "PPLC", "country_code": "TH", "cc2": [], "admin1_code": "40", "admin1_name": "Bangkok", "admin2_code": "", "admin2_name": "", "admin3_code": "", "admin4_code": "", "population": 5104476, "elevation": 2, "dem": 4, "timezone": "Asia/Bangkok", "modification_date": "2023-01-12" }
MongoDB Format
In MongoDB, the gazetteer data has the same structure as JSON but includes an additional location
field for geospatial queries:
{ "geoname_id": 1609350, "name": "Bangkok", "ascii_name": "Bangkok", "alternate_names": ["Krung Thep", "กรุงเทพมหานคร"], "latitude": 13.75, "longitude": 100.51667, "location": { "type": "Point", "coordinates": [100.51667, 13.75] }, "feature_class": "P", "feature_code": "PPLC", "country_code": "TH", "cc2": [], "admin1_code": "40", "admin1_name": "Bangkok", "admin2_code": "", "admin2_name": "", "admin3_code": "", "admin4_code": "", "population": 5104476, "elevation": 2, "dem": 4, "timezone": "Asia/Bangkok", "modification_date": "2023-01-12" }
The MongoDB collection is indexed for efficient queries:
- Unique index on
geoname_id
- Index on
country_code
- Index on
feature_class
- Index on
feature_code
- Text index on
name
andascii_name
- Geospatial index on
location
MongoDB Usage Examples
Finding locations near a point
$client = new MongoDB\Client('mongodb://localhost:27017'); $collection = $client->geonames->gazetteer; // Find all places within 5km of Bangkok $result = $collection->find([ 'location' => [ '$near' => [ '$geometry' => [ 'type' => 'Point', 'coordinates' => [100.51667, 13.75] // [longitude, latitude] ], '$maxDistance' => 5000 // 5km in meters ] ] ]); foreach ($result as $place) { echo $place['name'] . ' - ' . $place['feature_code'] . PHP_EOL; }
Finding postal codes by country
$client = new MongoDB\Client('mongodb://localhost:27017'); $collection = $client->geonames->postal_codes; // Find all postal codes in Bangkok, Thailand $result = $collection->find([ 'country_code' => 'TH', 'admin_name1' => 'Bangkok' ]); foreach ($result as $postalCode) { echo $postalCode['postal_code'] . ' - ' . $postalCode['place_name'] . PHP_EOL; }
License
This package is open-sourced software licensed under the MIT license.
Credits
- Data provided by GeoNames under a Creative Commons Attribution 4.0 License
- Developed by Parsilver