| Install | |
|---|---|
composer require nnjeim/world |

The World is a Laravel package that provides a comprehensive list of countries, states, cities, timezones, currencies, languages and IP geolocation. You can access the data using the World Facade or through defined API routes.
🌍 New in v1.1.38: IP Geolocation Module
Detect user location from IP address with automatic fallback to free API.
World::geolocate() · GET /api/geolocate
Learn more →
First, set your application environment to local:
set APP_ENV=local
Then, install the package via composer:
composer require nnjeim/world
Optionally, set the WORLD_DB_CONNECTION environment variable to your desired database connection.
Run the following Artisan command to automate the installation process:
php artisan world:install
If you prefer to install the package manually, follow these steps:
php artisan vendor:publish --tag=world --force
php artisan migrate
php artisan db:seed --class=WorldSeeder
⚠️ Important: When upgrading to a new version, you must re-publish the configuration file to ensure all new features work correctly:
php artisan vendor:publish --tag=world --force
This command updates the config/world.php file with new configuration options. Failing to run this command may result in missing features or errors.
World::geolocate() to detect location from client IPGET /api/geolocate with automatic IP detection from headersphp artisan world:geoip to download GeoLite2 database⚠️ Required: After upgrading, run
php artisan vendor:publish --tag=world --forceto update your configuration file.
For detailed information on recent changes, please see the CHANGELOG.
We welcome contributions! For details on how to get started, please review our CONTRIBUTING guidlines.
Explore the API examples on our live site:
List all countries:
https://world.bmbc.cloud/api/countries
Search for a country:
https://world.bmbc.cloud/api/countries?search=rom
Get states by country code:
https://world.bmbc.cloud/api/states?filters[country_code]=RO&fields=cities
Use the World facade:
use Nnjeim\World\World;
$action = World::countries();
if ($action->success) {
$countries = $action->data;
}
response (object)
{
"success": true,
"data": [
{
"id": 1,
"name": "Afghanistan"
},
{
"id": 2,
"name": "Åland Islands"
},
.
.
.
],
}
Use the API countries endpoint:
https://myDomain.local/api/countries
Use the World facade:
use Nnjeim\World\World;
$action = World::countries([
'fields' => 'states,cities',
'filters' => [
'iso2' => 'FR',
]
]);
if ($action->success) {
$countries = $action->data;
}
Response:
(object)
{
"success": true,
"data": [
"id": 77,
"name": "France",
"states": [
{
"id": 1271,
"name": "Alo"
},
{
"id": 1272,
"name": "Alsace"
},
.
.
.
],
"cities": [
{
"id": 25148,
"name": "Abondance"
},
{
"id": 25149,
"name": "Abrest"
},
.
.
.
]
],
}
Use the API countries endpoint:
https://myDomain.local/api/countries?fields=states,cities&filters[iso2]=FR
use Nnjeim\World\WorldHelper;
new class {
protected $world;
public function __construct(WorldHelper $world) {
$this->world = $world;
}
$action = $this->world->cities([
'filters' => [
'country_id' => 182,
],
]);
if ($action->success) {
$cities = $action->data;
}
}
Use the API cities endpoint:
https://myDomain.local/api/cities?filters[country_code]=RO
| Name | Description |
|---|---|
| countries | lists all the world countries |
| states | lists all the states |
| cities | lists all the cities |
| timezones | lists all the timezones |
| currencies | lists all the currencies |
| languages | lists all the languages |
| geolocate | geolocates an IP address |
An action response is formed as below:
success (boolean)message (string)data (instance of Illuminate\Support\Collection)errors (array)fields*: comma seperated string (countries table fields in addition to states, cities, currency and timezones).filters*: array of keys (countries table fields) and their corresponding values.search*: string.fields*: comma seperated string (states table fields in addition to country and states).filters*: array of keys (states table fields) and their corresponding values.search*: string.fields*: comma seperated string (cities table fields in addition to country and state).filters*: array of keys (cities table fields) and their corresponding values.search*: string.fields*: comma seperated string (timezones table fields in addition to country).filters*: array of keys (timezones table fields) and their corresponding values.search*: string.fields*: comma seperated string (currencies table fields in addition to country).filters*: array of keys (currencies table fields) and their corresponding values.search*: string.fields*: comma seperated string (languages table fields).filters*: array of keys (languages table fields) and their corresponding values.search*: string.Geolocate an IP address. Returns country, state, city, coordinates, and timezone information linked to existing World data.
ip*: string (optional - auto-detects from request if not provided).Data Sources:
The package automatically falls back to ip-api.com if the GeoLite2 database is not installed. For production use, we recommend downloading the GeoLite2 database:
# Get a free license key at: https://www.maxmind.com/en/geolite2/signup
php artisan world:geoip --license=YOUR_LICENSE_KEY
Or set the MAXMIND_LICENSE_KEY environment variable and run:
php artisan world:geoip
To disable the fallback API, set WORLD_GEOLOCATE_FALLBACK_API=false in your .env file.
Usage:
use Nnjeim\World\World;
// Auto-detect IP from request
$action = World::geolocate();
// Geolocate specific IP
$action = World::geolocate(['ip' => '8.8.8.8']);
if ($action->success) {
$location = $action->data;
// $location['country'], $location['state'], $location['city'], etc.
}
How IP Detection Works:
The client IP is automatically detected from request headers in this order:
CF-Connecting-IP (Cloudflare)X-Forwarded-For (Standard proxy header)X-Real-IP (Nginx proxy)CLIENT-IP (Generic)request()->ip() fallbackResponse Format:
{
"success": true,
"message": "geolocations",
"data": {
"ip": "8.8.8.8",
"country": {
"id": 236,
"iso2": "US",
"iso3": "USA",
"name": "United States",
"phone_code": "1",
"region": "Americas",
"subregion": "Northern America"
},
"state": {
"id": 4808,
"name": "Virginia",
"state_code": "VA"
},
"city": {
"id": 147562,
"name": "Ashburn"
},
"coordinates": {
"latitude": 39.03,
"longitude": -77.5,
"accuracy_radius": null
},
"timezone": {
"id": 404,
"name": "America/New_York"
},
"postal_code": "20149"
},
"response_time": "690 ms"
}
Note: The
idfields link to existing records in the World database tables. If a matching record is not found, theidfield will be omitted and only the raw geolocation data will be returned.
All routes can be prefixed by any string. Ex.: admin, api...
| Method | GET |
| Route | /{prefix}/countries |
| Parameters* | comma seperated fields (countries table fields in addition to states, cities, currency and timezones), array filters, string search |
| Example | /api/countries?fields=iso2,cities&filters[phone_code]=44 |
| response | success, message, data |
| Method | GET |
| Route | /{prefix}/states |
| Parameters* | comma seperated fields (states table fields in addition to country and cities), array filters, string search |
| Example | /api/states?fields=country,cities&filters[country_code]=RO |
| response | success, message, data |
| Method | GET |
| Route | /{prefix}/cities |
| Parameters* | comma seperated fields (cities table fields in addition to country and state), array filters, string search |
| Example | /api/cities?fields=country,state&filters[country_code]=RO |
| response | success, message, data |
| Method | GET |
| Route | /{prefix}/timezones |
| Parameters* | comma seperated fields (timezones table fields in addition to the country), array filters, string search |
| Example | /api/timezones?fields=country&filters[country_code]=RO |
| response | success, message, data |
| Method | GET |
| Route | /{prefix}/currencies |
| Parameters* | comma seperated fields (currencies table fields in addition to the country), array filters, string search |
| Example | /api/currencies?fields=code&filters[country_code]=RO |
| response | success, message, data |
| Method | GET |
| Route | /{prefix}/languages |
| Parameters* | comma seperated fields, string search |
| Example | /api/languages?fields=dir |
| response | success, message, data |
| Method | GET |
| Route | /{prefix}/geolocate |
| Parameters* | ip (optional - auto-detects from request) |
| Example | /api/geolocate or /api/geolocate?ip=8.8.8.8 |
| response | success, message, data (ip, country, state, city, coordinates, timezone, postal_code) |
The available locales are
ar, az, bn, br, de, en, es, fr, hy, it, ja, kr, ne, nl, pl, pt, ro, ru, sw, tr and zh.
The default locale is en.
Header option
accept-language=locale
Alternatively, you can use specific locale with the World Facade setLocale('locale') helper method. Example:
World::setLocale('zh')->countries();

The configuration for the World package is located in the world.php file.
If you're upgrading from a previous version, you may want to re-publish the config file:
php artisan vendor:publish --tag=world --force
By default, this package uses the default database connection, but it's possible to customize it
using the WORLD_DB_CONNECTION variable in your .env file.
Countries can be restricted while seeding the database either by adding the ISO2 country codes in the allowed_countries or disallowed_countries array lists.
A list of the accepted locales which relate to the localized lang/ files.
The states, cities, timezones, currencies and languages modules can be optionally disabled.
Please note that the cities module depends on the states module.
If you don't wish to use the packages as an API service, you can disable all the routes by assigning false to routes.
It offers the ability to enable or disable the database fields.
When changing this configuration the database should be dropped and the seeder should be re-run.
Requirements
Browse to the package root folder and run:
composer install # installs the package dev dependencies
composer test
* optional