31 Commits

Author SHA1 Message Date
Gregory Oschwald
7fa971aef8 Updated .gitignore 2013-10-21 11:15:22 -07:00
Gregory Oschwald
edb6940c74 Updated exception text in test 2013-10-21 11:09:27 -07:00
Gregory Oschwald
57cb77a53c Updated version in example composer.json 2013-10-21 11:04:28 -07:00
Gregory Oschwald
8aedaa5f91 Doc updates 2013-10-21 11:03:02 -07:00
Gregory Oschwald
87f6e2ad6b Require new version of MaxMind DB API 2013-10-21 11:00:02 -07:00
Gregory Oschwald
6bcd0ab21e Removed Get Satisfaction link 2013-10-21 09:10:03 -07:00
Gregory Oschwald
e03b9558a9 Tidying and doc fixes 2013-10-21 06:18:12 -07:00
Gregory Oschwald
234dcaff7c Renamed languages to locales in the code 2013-10-17 12:58:21 -07:00
Gregory Oschwald
b8188bc558 language code -> locale code 2013-10-17 10:28:31 -07:00
Gregory Oschwald
8f16ed6115 Merge branch 'leading-backslash' of github.com:BenMorel/GeoIP2-php into BenMorel-leading-backslash 2013-10-16 10:45:50 -07:00
Gregory Oschwald
8f3331002a Merge branch 'unused-imports' of github.com:BenMorel/GeoIP2-php into BenMorel-unused-imports 2013-10-16 10:43:47 -07:00
Benjamin Morel
b4b0341307 Fixed typo in exception names 2013-10-16 17:33:29 +00:00
Benjamin Morel
ee00329652 Removed unused imports 2013-10-16 17:30:18 +00:00
Benjamin Morel
e8fa9a3a24 Removed the leading backslash on imports 2013-10-16 17:27:40 +00:00
Gregory Oschwald
167433de0c Added iterface for Reader/Client.
It could use a better name.
2013-10-02 06:54:03 -07:00
Gregory Oschwald
c584dbfd6b Added more info to comment 2013-09-19 08:07:24 -07:00
Gregory Oschwald
c642a32b0c Added some comments in examples 2013-09-19 07:52:44 -07:00
Gregory Oschwald
decfda7e90 Added missing documentation on $languages param 2013-08-02 13:34:53 -07:00
Gregory Oschwald
3a560452af Added more detailed examples 2013-08-02 13:32:21 -07:00
Gregory Oschwald
8febab1fe4 Updates for v0.4.0 release 2013-07-16 12:40:34 -07:00
Gregory Oschwald
f0430b613d No reason for call_user_func_array 2013-07-16 10:34:44 -07:00
Gregory Oschwald
e3289ec416 Fix for PHP 5.3 where you can't use '$this' in a callback. 2013-07-16 10:11:06 -07:00
Gregory Oschwald
1d1dcec74c Added missing method documentation 2013-07-16 09:48:45 -07:00
Gregory Oschwald
630765f924 Make unit tests check all the methods 2013-07-16 09:44:30 -07:00
Gregory Oschwald
b55d42b5cd Documentation updates 2013-07-16 09:21:08 -07:00
Gregory Oschwald
044ca4d085 Added specific exceptions for the different types of errors. 2013-07-16 07:46:50 -07:00
Gregory Oschwald
3ba84992c8 Updated for new test path 2013-07-15 16:10:05 -07:00
Gregory Oschwald
a6c1f9378e Updated submodule 2013-07-15 16:09:31 -07:00
Gregory Oschwald
1c0817d133 Test looking up an invalid address 2013-07-15 16:05:08 -07:00
Gregory Oschwald
59a51ef34d Tests for the reader 2013-07-15 15:52:07 -07:00
Gregory Oschwald
3e0fc2a837 Basic GeoIP2 database reader 2013-07-15 15:06:51 -07:00
29 changed files with 756 additions and 225 deletions

1
.gitignore vendored
View File

@@ -1,5 +1,6 @@
_site _site
.gh-pages .gh-pages
.idea
composer.lock composer.lock
composer.phar composer.phar
phpunit.xml phpunit.xml

3
.gitmodules vendored Normal file
View File

@@ -0,0 +1,3 @@
[submodule "maxmind-db"]
path = maxmind-db
url = git://github.com/maxmind/MaxMind-DB.git

View File

@@ -1,6 +1,23 @@
CHANGELOG CHANGELOG
========= =========
0.5.0 (2013-10-21)
------------------
* Renamed $languages constructor parameters to $locales for both the Client
and Reader classes.
* Documentation and code clean-up (Ben Morel).
* Added the interface `GeoIp2\ProviderInterface`, which is implemented by both
`\GeoIp2\Database\Reader` and `\GeoIp2\WebService\Client`.
0.4.0 (2013-07-16)
------------------
* This is the first release with the GeoIP2 database reader. Please see the
`README.md` file and the `\GeoIp2\Database\Reader` class.
* The general exception classes were replaced with specific exception classes
representing particular types of errors, such as an authentication error.
0.3.0 (2013-07-12) 0.3.0 (2013-07-12)
------------------ ------------------

147
README.md
View File

@@ -8,32 +8,24 @@ release, which will be numbered 2.0.0.
You may find information on the GeoIP2 beta release process on [our You may find information on the GeoIP2 beta release process on [our
website](http://www.maxmind.com/en/geoip2_beta). website](http://www.maxmind.com/en/geoip2_beta).
To provide feedback or get support during the beta, please see the
[MaxMind Customer Community](https://getsatisfaction.com/maxmind).
## Description ## ## Description ##
Currently, this distribution provides an API for the [GeoIP2 web services] This distribution provides an API for the [GeoIP2 web services]
(http://dev.maxmind.com/geoip/geoip2/web-services). (http://dev.maxmind.com/geoip/geoip2/web-services) and the [GeoLite2
databases](http://dev.maxmind.com/geoip/geoip2/geolite2/). The commercial
In the future, this distribution will also provide the same API for the GeoIP2 databases have not yet been released as a downloadable product.
GeoIP2 downloadable databases. These databases have not yet been
released as a downloadable product.
See ``GeoIp2\WebService\Client`` for details on the web service client
API.
## Installation ## ## Installation ##
### Define Your Dependencies ### ### Define Your Dependencies ###
We recommend installing this package with [Composer](http://getcomposer.org/). We recommend installing this package with [Composer](http://getcomposer.org/).
To do this, add ```geoip2/geoip2``` to your ```composer.json``` file. To do this, add `geoip2/geoip2` to your `composer.json` file.
```json ```json
{ {
"require": { "require": {
"geoip2/geoip2": "0.3.*" "geoip2/geoip2": "0.5.*"
} }
} }
``` ```
@@ -61,10 +53,71 @@ You can autoload all dependencies by adding this to your code:
require 'vendor/autoload.php'; require 'vendor/autoload.php';
``` ```
## Usage ## ### Optional C Extension ###
To use this API, you must create a new ``\GeoIp2\WebService\Client`` The [MaxMind DB API](https://github.com/maxmind/MaxMind-DB-Reader-php)
object with your ``$userId`` and ``$licenseKey``, then you call the method includes an optional C extension that you may install to dramatically increase
the performance of lookups in GeoIP2 or GeoLite2 databases. To install, please
follow the instructions included with that API.
The extension has no effect on web-service lookups.
## Database Reader ##
### Usage ###
To use this API, you must create a new `\GeoIp2\Database\Reader` object with
the path to the database file as the first argument to the constructor. You
may then call the method corresponding to the database you are using.
If the lookup succeeds, the method call will return a model class for the
record in the database. This model in turn contains multiple container
classes for the different parts of the data such as the city in which the
IP address is located.
If the record is not found, a `\GeoIp2\Exception\AddressNotFoundException`
is returned. If the database is invalid or corrupt, a
`\MaxMind\Db\InvalidDatabaseException` will be thrown.
See the API documentation for more details.
### Example ###
```php
<?php
require_once 'vendor/autoload.php';
use GeoIp2\Database\Reader;
// This creates the Reader object, which should be reused across
// lookups.
$reader = new Reader('/usr/local/share/GeoIP/GeoIP2-City.mmdb');
// Replace "city" with the appropriate method for your database, e.g.,
// "country".
$record = $reader->city('128.101.101.101');
print($record->country->isoCode . "\n"); // 'US'
print($record->country->name . "\n"); // 'United States'
print($record->country->names['zh-CN'] . "\n"); // '美国'
print($record->mostSpecificSubdivision->name . "\n"); // 'Minnesota'
print($record->mostSpecificSubdivision->isoCode . "\n"); // 'MN'
print($record->city->name . "\n"); // 'Minneapolis'
print($record->postal->code . "\n"); // '55455'
print($record->location->latitude . "\n"); // 44.9733
print($record->location->longitude . "\n"); // -93.2323
```
## Web Service Client ##
### Usage ###
To use this API, you must create a new `\GeoIp2\WebService\Client`
object with your `$userId` and `$licenseKey`, then you call the method
corresponding to a specific end point, passing it the IP address you want to corresponding to a specific end point, passing it the IP address you want to
look up. look up.
@@ -72,41 +125,43 @@ If the request succeeds, the method call will return a model class for the end
point you called. This model in turn contains multiple record classes, each of point you called. This model in turn contains multiple record classes, each of
which represents part of the data returned by the web service. which represents part of the data returned by the web service.
If there is an error, a structured exception is thrown.
See the API documentation for more details. See the API documentation for more details.
## Example ## ### Example ###
```php ```php
<?php <?php
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
use \GeoIp2\WebService\Client; use GeoIp2\WebService\Client;
// This creates a Client object that can be reused across requests.
// Replace "42" with your user ID and "license_key" with your license
// key.
$client = new Client(42, 'abcdef123456'); $client = new Client(42, 'abcdef123456');
$omni = $client->omni('24.24.24.24');
echo $omni->country->isoCode; // Replace "city" with the method corresponding to the web service that
// you are using, e.g., "country", "cityIspOrg", "omni".
$record = $client->city('128.101.101.101');
print($record->country->isoCode . "\n"); // 'US'
print($record->country->name . "\n"); // 'United States'
print($record->country->names['zh-CN'] . "\n"); // '美国'
print($record->mostSpecificSubdivision->name . "\n"); // 'Minnesota'
print($record->mostSpecificSubdivision->isoCode . "\n"); // 'MN'
print($record->city->name . "\n"); // 'Minneapolis'
print($record->postal->code . "\n"); // '55455'
print($record->location->latitude . "\n"); // 44.9733
print($record->location->longitude . "\n"); // -93.2323
``` ```
## Exceptions ## ### What data is returned? ###
For details on the possible errors returned by the web service itself, see
the
[GeoIP2 web service docs](http://dev.maxmind.com/geoip2/geoip/web-service).
If the web service returns an explicit error document, this is thrown as a
```\GeoIp2\Exception\WebServiceException```. If some other sort of transport
error occurs, this is thrown as a ```\GeoIp2\Exception\HttpException```.
The difference is that the web service error includes an error message and
error code delivered by the web service. The latter is thrown when some sort
of unanticipated error occurs, such as the web service returning a 500 or an
invalid error document.
If the web service returns any status code besides 200, 4xx, or 5xx, this also
becomes a ```\GeoIp2\Exception\HttpException```.
Finally, if the web service returns a 200 but the body is invalid, the client
throws a ```\GeoIp2\Exception\GeoIp2Exception```.
## What data is returned? ##
While many of the end points return the same basic records, the attributes While many of the end points return the same basic records, the attributes
which can be populated vary between end points. In addition, while an end which can be populated vary between end points. In addition, while an end
@@ -120,8 +175,8 @@ See the
[GeoIP2 web service docs](http://dev.maxmind.com/geoip/geoip2/web-services) [GeoIP2 web service docs](http://dev.maxmind.com/geoip/geoip2/web-services)
for details on what data each end point may return. for details on what data each end point may return.
The only piece of data which is always returned is the ```ipAddress``` The only piece of data which is always returned is the `ipAddress`
attribute in the ``GeoIp2\Record\Traits`` record. attribute in the `GeoIp2\Record\Traits` record.
Every record class attribute has a corresponding predicate method so you can Every record class attribute has a corresponding predicate method so you can
check to see if the attribute is set. check to see if the attribute is set.
@@ -131,10 +186,10 @@ check to see if the attribute is set.
[GeoNames](http://www.geonames.org/) offers web services and downloadable [GeoNames](http://www.geonames.org/) offers web services and downloadable
databases with data on geographical features around the world, including databases with data on geographical features around the world, including
populated places. They offer both free and paid premium data. Each populated places. They offer both free and paid premium data. Each
feature is unique identified by a ```geonameId```, which is an integer. feature is unique identified by a `geonameId`, which is an integer.
Many of the records returned by the GeoIP2 web services and databases Many of the records returned by the GeoIP2 web services and databases
include a ```geonameId``` property. This is the ID of a geographical feature include a `geonameId` property. This is the ID of a geographical feature
(city, region, country, etc.) in the GeoNames database. (city, region, country, etc.) in the GeoNames database.
Some of the data that MaxMind provides is also sourced from GeoNames. We Some of the data that MaxMind provides is also sourced from GeoNames. We

View File

@@ -58,6 +58,6 @@ git push git@github.com:maxmind/GeoIP2-php.git
git push git push
cd .. cd ..
git tag $TAG git tag -a $TAG
git push git push
git push --tags git push --tags

View File

@@ -14,6 +14,7 @@
], ],
"require": { "require": {
"guzzle/guzzle": "3.*", "guzzle/guzzle": "3.*",
"maxmind-db/reader": "0.2.*",
"php": ">=5.3.1" "php": ">=5.3.1"
}, },
"require-dev": { "require-dev": {

1
maxmind-db Submodule

Submodule maxmind-db added at f7b9342b84

View File

@@ -0,0 +1,149 @@
<?php
namespace GeoIp2\Database;
use GeoIp2\Exception\AddressNotFoundException;
use GeoIp2\ProviderInterface;
use MaxMind\Db\Reader as DbReader;
/**
* Instances of this class provide a reader for the GeoIP2 database format.
* IP addresses can be looked up using the <code>country</code>
* and <code>city</code> methods. We also provide <code>cityIspOrg</code>
* and <code>omni</code> methods to ease compatibility with the web service
* client, although we may offer the ability to specify additional databases
* to replicate these web services in the future (e.g., the ISP/Org database).
*
* **Usage**
*
* The basic API for this class is the same for every database. First, you
* create a reader object, specifying a file name. You then call the method
* corresponding to the specific database, passing it the IP address you want
* to look up.
*
* If the request succeeds, the method call will return a model class for
* the method you called. This model in turn contains multiple record classes,
* each of which represents part of the data returned by the database. If
* the database does not contain the requested information, the attributes
* on the record class will have a <code>null</code> value.
*
* If the address is not in the database, an
* {@link \GeoIp2\Exception\AddressNotFoundException} exception will be
* thrown. If an invalid IP address is passed to one of the methods, a
* SPL {@link \InvalidArgumentException} will be thrown. If the database is
* corrupt or invalid, a {@link \MaxMind\Db\Reader\InvalidDatabaseException}
* will be thrown.
*
*/
class Reader implements ProviderInterface
{
private $dbReader;
private $locales;
/**
* Constructor.
*
* @param string $filename The path to the GeoIP2 database file.
* @param array $locales List of locale codes to use in name property
* from most preferred to least preferred.
* @throws \MaxMind\Db\Reader\InvalidDatabaseException if the database
* is corrupt or invalid
*/
public function __construct(
$filename,
$locales = array('en')
) {
$this->dbReader = new DbReader($filename);
$this->locales = $locales;
}
/**
* This method returns a GeoIP2 City model.
*
* @param string $ipAddress IPv4 or IPv6 address as a string.
*
* @return \GeoIp2\Model\City
*
* @throws \GeoIp2\Exception\AddressNotFoundException if the address is
* not in the database.
* @throws \MaxMind\Db\Reader\InvalidDatabaseException if the database
* is corrupt or invalid
*/
public function city($ipAddress)
{
return $this->modelFor('City', $ipAddress);
}
/**
* This method returns a GeoIP2 Country model.
*
* @param string $ipAddress IPv4 or IPv6 address as a string.
*
* @return \GeoIp2\Model\Country
*
* @throws \GeoIp2\Exception\AddressNotFoundException if the address is
* not in the database.
* @throws \MaxMind\Db\Reader\InvalidDatabaseException if the database
* is corrupt or invalid
*/
public function country($ipAddress)
{
return $this->modelFor('Country', $ipAddress);
}
/**
* This method returns a GeoIP2 City/ISP/Org model.
*
* @param string $ipAddress IPv4 or IPv6 address as a string.
*
* @return \GeoIp2\Model\CityIspOrg
*
* @throws \GeoIp2\Exception\AddressNotFoundException if the address is
* not in the database.
* @throws \MaxMind\Db\Reader\InvalidDatabaseException if the database
* is corrupt or invalid
*/
public function cityIspOrg($ipAddress)
{
return $this->modelFor('CityIspOrg', $ipAddress);
}
/**
* This method returns a GeoIP2 Omni model.
*
* @param string $ipAddress IPv4 or IPv6 address as a string.
*
* @return \GeoIp2\Model\Omni
*
* @throws \GeoIp2\Exception\AddressNotFoundException if the address is
* not in the database.
* @throws \MaxMind\Db\Reader\InvalidDatabaseException if the database
* is corrupt or invalid
*/
public function omni($ipAddress)
{
return $this->modelFor('Omni', $ipAddress);
}
private function modelFor($class, $ipAddress)
{
$record = $this->dbReader->get($ipAddress);
if ($record === null) {
throw new AddressNotFoundException(
"The address $ipAddress is not in the database."
);
}
$record['traits']['ip_address'] = $ipAddress;
$class = "GeoIp2\\Model\\" . $class;
return new $class($record, $this->locales);
}
/**
* Closes the GeoIP2 database and returns the resources to the system.
*/
public function close()
{
$this->dbReader->close();
}
}

View File

@@ -0,0 +1,10 @@
<?php
namespace GeoIp2\Exception;
/**
* This class represents a generic error.
*/
class AddressNotFoundException extends GeoIp2Exception
{
}

View File

@@ -0,0 +1,10 @@
<?php
namespace GeoIp2\Exception;
/**
* This class represents a generic error.
*/
class AuthenticationException extends GeoIp2Exception
{
}

View File

@@ -17,7 +17,7 @@ class HttpException extends GeoIp2Exception
$message, $message,
$httpStatus, $httpStatus,
$uri, $uri,
Exception $previous = null \Exception $previous = null
) { ) {
$this->uri = $uri; $this->uri = $uri;
parent::__construct($message, $httpStatus, $previous); parent::__construct($message, $httpStatus, $previous);

View File

@@ -6,7 +6,7 @@ namespace GeoIp2\Exception;
* This class represents an error returned by MaxMind's GeoIP2 * This class represents an error returned by MaxMind's GeoIP2
* web service. * web service.
*/ */
class WebServiceException extends HttpException class InvalidRequestException extends HttpException
{ {
/** /**
* The code returned by the MaxMind web service * The code returned by the MaxMind web service
@@ -18,7 +18,7 @@ class WebServiceException extends HttpException
$error, $error,
$httpStatus, $httpStatus,
$uri, $uri,
Exception $previous = null \Exception $previous = null
) { ) {
$this->error = $error; $this->error = $error;
parent::__construct($message, $httpStatus, $uri, $previous); parent::__construct($message, $httpStatus, $uri, $previous);

View File

@@ -0,0 +1,10 @@
<?php
namespace GeoIp2\Exception;
/**
* This class represents a generic error.
*/
class OutOfQueriesException extends GeoIp2Exception
{
}

View File

@@ -74,18 +74,18 @@ class City extends Country
/** /**
* @ignore * @ignore
*/ */
public function __construct($raw, $languages) public function __construct($raw, $locales)
{ {
parent::__construct($raw, $languages); parent::__construct($raw, $locales);
$this->city = new \GeoIp2\Record\City($this->get('city'), $languages); $this->city = new \GeoIp2\Record\City($this->get('city'), $locales);
$this->location = new \GeoIp2\Record\Location($this->get('location')); $this->location = new \GeoIp2\Record\Location($this->get('location'));
$this->postal = new \GeoIp2\Record\Postal($this->get('postal')); $this->postal = new \GeoIp2\Record\Postal($this->get('postal'));
$this->createSubdivisions($raw, $languages); $this->createSubdivisions($raw, $locales);
} }
private function createSubdivisions($raw, $languages) private function createSubdivisions($raw, $locales)
{ {
if (!isset($raw['subdivisions'])) { if (!isset($raw['subdivisions'])) {
return; return;
@@ -94,7 +94,7 @@ class City extends Country
foreach ($raw['subdivisions'] as $sub) { foreach ($raw['subdivisions'] as $sub) {
array_push( array_push(
$this->subdivisions, $this->subdivisions,
new \GeoIp2\Record\Subdivision($sub, $languages) new \GeoIp2\Record\Subdivision($sub, $locales)
); );
} }
} }
@@ -113,8 +113,8 @@ class City extends Country
private function mostSpecificSubdivision() private function mostSpecificSubdivision()
{ {
return empty($this->subdivisions)? return empty($this->subdivisions) ?
new \GeoIp2\Record\Subdivision(array(), $this->languages): new \GeoIp2\Record\Subdivision(array(), $this->locales) :
end($this->subdivisions); end($this->subdivisions);
} }
} }

View File

@@ -37,7 +37,7 @@ class Country
{ {
private $continent; private $continent;
private $country; private $country;
private $languages; private $locales;
private $maxmind; private $maxmind;
private $registeredCountry; private $registeredCountry;
private $representedCountry; private $representedCountry;
@@ -47,30 +47,30 @@ class Country
/** /**
* @ignore * @ignore
*/ */
public function __construct($raw, $languages) public function __construct($raw, $locales)
{ {
$this->raw = $raw; $this->raw = $raw;
$this->continent = new \GeoIp2\Record\Continent( $this->continent = new \GeoIp2\Record\Continent(
$this->get('continent'), $this->get('continent'),
$languages $locales
); );
$this->country = new \GeoIp2\Record\Country( $this->country = new \GeoIp2\Record\Country(
$this->get('country'), $this->get('country'),
$languages $locales
); );
$this->maxmind = new \GeoIp2\Record\MaxMind($this->get('maxmind')); $this->maxmind = new \GeoIp2\Record\MaxMind($this->get('maxmind'));
$this->registeredCountry = new \GeoIp2\Record\Country( $this->registeredCountry = new \GeoIp2\Record\Country(
$this->get('registered_country'), $this->get('registered_country'),
$languages $locales
); );
$this->representedCountry = new \GeoIp2\Record\RepresentedCountry( $this->representedCountry = new \GeoIp2\Record\RepresentedCountry(
$this->get('represented_country'), $this->get('represented_country'),
$languages $locales
); );
$this->traits = new \GeoIp2\Record\Traits($this->get('traits')); $this->traits = new \GeoIp2\Record\Traits($this->get('traits'));
$this->languages = $languages; $this->locales = $locales;
} }
/** /**
@@ -84,7 +84,7 @@ class Country
/** /**
* @ignore * @ignore
*/ */
public function __get ($attr) public function __get($attr)
{ {
if ($attr != "instance" && isset($this->$attr)) { if ($attr != "instance" && isset($this->$attr)) {
return $this->$attr; return $this->$attr;

View File

@@ -0,0 +1,34 @@
<?php
namespace GeoIp2;
interface ProviderInterface
{
/**
* @param ipAddress
* IPv4 or IPv6 address to lookup.
* @return \GeoIp2\Model\Country A Country model for the requested IP address.
*/
public function country($ipAddress);
/**
* @param ipAddress
* IPv4 or IPv6 address to lookup.
* @return \GeoIp2\Model\City A City model for the requested IP address.
*/
public function city($ipAddress);
/**
* @param ipAddress
* IPv4 or IPv6 address to lookup.
* @return \GeoIp2\Model\CityIspOrg A CityIspOrg model for the requested IP address.
*/
public function cityIspOrg($ipAddress);
/**
* @param ipAddress
* IPv4 or IPv6 address to lookup.
* @return \GeoIp2\Model\Omni An Omni model for the requested IP address.
*/
public function omni($ipAddress);
}

View File

@@ -4,14 +4,14 @@ namespace GeoIp2\Record;
abstract class AbstractPlaceRecord extends AbstractRecord abstract class AbstractPlaceRecord extends AbstractRecord
{ {
private $languages; private $locales;
/** /**
* @ignore * @ignore
*/ */
public function __construct($record, $languages) public function __construct($record, $locales)
{ {
$this->languages = $languages; $this->locales = $locales;
parent::__construct($record); parent::__construct($record);
} }
@@ -29,9 +29,9 @@ abstract class AbstractPlaceRecord extends AbstractRecord
private function name() private function name()
{ {
foreach ($this->languages as $language) { foreach ($this->locales as $locale) {
if (isset($this->names[$language])) { if (isset($this->names[$locale])) {
return $this->names[$language]; return $this->names[$locale];
} }
} }
} }

View File

@@ -14,10 +14,10 @@ namespace GeoIp2\Record;
* @property int $geonameId The GeoName ID for the city. This attribute * @property int $geonameId The GeoName ID for the city. This attribute
* is returned by all end points. * is returned by all end points.
* *
* @property string $name The name of the city based on the languages list * @property string $name The name of the city based on the locales list
* passed to the constructor. This attribute is returned by all end points. * passed to the constructor. This attribute is returned by all end points.
* *
* @property array $names A array map where the keys are language codes * @property array $names A array map where the keys are locale codes
* and the values are names. This attribute is returned by all end points. * and the values are names. This attribute is returned by all end points.
*/ */
class City extends AbstractPlaceRecord class City extends AbstractPlaceRecord

View File

@@ -14,10 +14,10 @@ namespace GeoIp2\Record;
* is returned by all end points. * is returned by all end points.
* *
* @property string $name Returns the name of the continent based on the * @property string $name Returns the name of the continent based on the
* languages list passed to the constructor. This attribute is returned by * locales list passed to the constructor. This attribute is returned by
* all end points. * all end points.
* *
* @property array $names An array map where the keys are language codes * @property array $names An array map where the keys are locale codes
* and the values are names. This attribute is returned by all end points. * and the values are names. This attribute is returned by all end points.
*/ */
class Continent extends AbstractPlaceRecord class Continent extends AbstractPlaceRecord

View File

@@ -18,10 +18,10 @@ namespace GeoIp2\Record;
* two-character ISO 3166-1 alpha code} for the country. This attribute is * two-character ISO 3166-1 alpha code} for the country. This attribute is
* returned by all end points. * returned by all end points.
* *
* @property string $name The name of the country based on the languages list * @property string $name The name of the country based on the locales list
* passed to the constructor. This attribute is returned by all end points. * passed to the constructor. This attribute is returned by all end points.
* *
* @property array $names An array map where the keys are language codes and * @property array $names An array map where the keys are locale codes and
* the values are names. This attribute is returned by all end points. * the values are names. This attribute is returned by all end points.
*/ */
class Country extends AbstractPlaceRecord class Country extends AbstractPlaceRecord

View File

@@ -22,10 +22,10 @@ namespace GeoIp2\Record;
* two-character ISO 3166-1 alpha code} for the country. This attribute is * two-character ISO 3166-1 alpha code} for the country. This attribute is
* returned by all end points. * returned by all end points.
* *
* @property string $name The name of the country based on the languages list * @property string $name The name of the country based on the locales list
* passed to the constructor. This attribute is returned by all end points. * passed to the constructor. This attribute is returned by all end points.
* *
* @property array $names An array map where the keys are language codes and * @property array $names An array map where the keys are locale codes and
* the values are names. This attribute is returned by all end points. * the values are names. This attribute is returned by all end points.
* *
* @property string $type A string indicating the type of entity that is * @property string $type A string indicating the type of entity that is

View File

@@ -20,14 +20,14 @@ namespace GeoIp2\Record;
* http://en.wikipedia.org/wiki/ISO_3166-2 ISO 3166-2 code}. This attribute * http://en.wikipedia.org/wiki/ISO_3166-2 ISO 3166-2 code}. This attribute
* is returned by all end points except Country. * is returned by all end points except Country.
* *
* @property string $name The name of the subdivision based on the languages * @property string $name The name of the subdivision based on the locales
* list passed to the constructor. This attribute is returned by all end * list passed to the constructor. This attribute is returned by all end
* points except Country. * points except Country.
* *
* @property array $names An array map where the keys are language codes and * @property array $names An array map where the keys are locale codes and
* the values are names. This attribute is returned by all end points except * the values are names. This attribute is returned by all end points except
* Country. * Country.
*/ */
class Subdivision extends AbstractPlaceRecord class Subdivision extends AbstractPlaceRecord
{ {
/** /**

View File

@@ -2,15 +2,15 @@
namespace GeoIp2\WebService; namespace GeoIp2\WebService;
use GeoIp2\Exception\AddressNotFoundException;
use GeoIp2\Exception\AuthenticationException;
use GeoIp2\Exception\GeoIp2Exception; use GeoIp2\Exception\GeoIp2Exception;
use GeoIp2\Exception\HttpException; use GeoIp2\Exception\HttpException;
use GeoIp2\Exception\WebServiceException; use GeoIp2\Exception\InvalidRequestException;
use GeoIp2\Model\City; use GeoIp2\Exception\OutOfQueriesException;
use GeoIp2\Model\CityIspOrg; use GeoIp2\ProviderInterface;
use GeoIp2\Model\Country;
use GeoIp2\Model\Omni;
use Guzzle\Http\Client as GuzzleClient;
use Guzzle\Common\Exception\RuntimeException; use Guzzle\Common\Exception\RuntimeException;
use Guzzle\Http\Client as GuzzleClient;
use Guzzle\Http\Exception\ClientErrorResponseException; use Guzzle\Http\Exception\ClientErrorResponseException;
use Guzzle\Http\Exception\ServerErrorResponseException; use Guzzle\Http\Exception\ServerErrorResponseException;
@@ -44,32 +44,12 @@ use Guzzle\Http\Exception\ServerErrorResponseException;
* service. * service.
* *
* If the request fails, the client class throws an exception. * If the request fails, the client class throws an exception.
*
* **Exceptions**
*
* For details on the possible errors returned by the web service itself, see
* {@link http://dev.maxmind.com/geoip2/geoip/web-services the GeoIP2 web
* service docs}.
*
* If the web service returns an explicit error document, this is thrown as a
* {@link \GeoIp2\Exception\WebServiceException}. If some other sort of
* transport error occurs, this is thrown as a {@link
* \GeoIp2\Exception\HttpException}. The difference is that the web service
* error includes an error message and error code delivered by the web
* service. The latter is thrown when some sort of unanticipated error occurs,
* such as the web service returning a 500 or an invalid error document.
*
* If the web service returns any status code besides 200, 4xx, or 5xx, this
* also becomes a {@link \GeoIp2\Exception\HttpException}.
*
* Finally, if the web service returns a 200 but the body is invalid, the
* client throws a {@link \GeoIp2\Exception\GeoIp2Exception}.
*/ */
class Client class Client implements ProviderInterface
{ {
private $userId; private $userId;
private $licenseKey; private $licenseKey;
private $languages; private $locales;
private $host; private $host;
private $guzzleClient; private $guzzleClient;
@@ -78,7 +58,7 @@ class Client
* *
* @param int $userId Your MaxMind user ID * @param int $userId Your MaxMind user ID
* @param string $licenseKey Your MaxMind license key * @param string $licenseKey Your MaxMind license key
* @param array $languages List of language codes to use in name property * @param array $locales List of locale codes to use in name property
* from most preferred to least preferred. * from most preferred to least preferred.
* @param string $host Optional host parameter * @param string $host Optional host parameter
* @param object $guzzleClient Optional Guzzle client to use (to facilitate * @param object $guzzleClient Optional Guzzle client to use (to facilitate
@@ -87,13 +67,13 @@ class Client
public function __construct( public function __construct(
$userId, $userId,
$licenseKey, $licenseKey,
$languages = array('en'), $locales = array('en'),
$host = 'geoip.maxmind.com', $host = 'geoip.maxmind.com',
$guzzleClient = null $guzzleClient = null
) { ) {
$this->userId = $userId; $this->userId = $userId;
$this->licenseKey = $licenseKey; $this->licenseKey = $licenseKey;
$this->languages = $languages; $this->locales = $locales;
$this->host = $host; $this->host = $host;
// To enable unit testing // To enable unit testing
$this->guzzleClient = $guzzleClient; $this->guzzleClient = $guzzleClient;
@@ -108,12 +88,23 @@ class Client
* *
* @return \GeoIp2\Model\City * @return \GeoIp2\Model\City
* *
* @throws \GeoIp2\Exception\GeoIp2Exception if there was a generic * @throws \GeoIp2\Exception\AddressNotFoundException if the address you
* error processing your request. * provided is not in our database (e.g., a private address).
* @throws \GeoIp2\Exception\HttpException if there was an HTTP transport * @throws \GeoIp2\Exception\AuthenticationException if there is a problem
* error. * with the user ID or license key that you provided.
* @throws \GeoIp2\Exception\WebServiceException if an error was returned * @throws \GeoIp2\Exception\OutOfQueriesException if your account is out
* by MaxMind's GeoIP2 web service. * of queries.
* @throws \GeoIp2\Exception\InvalidRequestException} if your request was
* received by the web service but is invalid for some other reason.
* This may indicate an issue with this API. Please report the error to
* MaxMind.
* @throws \GeoIp2\Exception\HttpException if an unexpected HTTP error
* code or message was returned. This could indicate a problem with the
* connection between your server and the web service or that the web
* service returned an invalid document or 500 error code.
* @throws \GeoIp2\Exception\GeoIp2Exception This serves as the parent
* class to the above exceptions. It will be thrown directly if a 200
* status code is returned but the body is invalid.
*/ */
public function city($ipAddress = 'me') public function city($ipAddress = 'me')
{ {
@@ -129,12 +120,23 @@ class Client
* *
* @return \GeoIp2\Model\Country * @return \GeoIp2\Model\Country
* *
* @throws \GeoIp2\Exception\GeoIp2Exception if there was a generic * @throws \GeoIp2\Exception\AddressNotFoundException if the address you
* error processing your request. * provided is not in our database (e.g., a private address).
* @throws \GeoIp2\Exception\HttpException if there was an HTTP transport * @throws \GeoIp2\Exception\AuthenticationException if there is a problem
* error. * with the user ID or license key that you provided.
* @throws \GeoIp2\Exception\WebServiceException if an error was returned * @throws \GeoIp2\Exception\OutOfQueriesException if your account is out
* by MaxMind's GeoIP2 web service. * of queries.
* @throws \GeoIp2\Exception\InvalidRequestException} if your request was
* received by the web service but is invalid for some other reason.
* This may indicate an issue with this API. Please report the error to
* MaxMind.
* @throws \GeoIp2\Exception\HttpException if an unexpected HTTP error
* code or message was returned. This could indicate a problem with the
* connection between your server and the web service or that the web
* service returned an invalid document or 500 error code.
* @throws \GeoIp2\Exception\GeoIp2Exception This serves as the parent
* class to the above exceptions. It will be thrown directly if a 200
* status code is returned but the body is invalid.
*/ */
public function country($ipAddress = 'me') public function country($ipAddress = 'me')
{ {
@@ -150,12 +152,23 @@ class Client
* *
* @return \GeoIp2\Model\CityIspOrg * @return \GeoIp2\Model\CityIspOrg
* *
* @throws \GeoIp2\Exception\GeoIp2Exception if there was a generic * @throws \GeoIp2\Exception\AddressNotFoundException if the address you
* error processing your request. * provided is not in our database (e.g., a private address).
* @throws \GeoIp2\Exception\HttpException if there was an HTTP transport * @throws \GeoIp2\Exception\AuthenticationException if there is a problem
* error. * with the user ID or license key that you provided.
* @throws \GeoIp2\Exception\WebServiceException if an error was returned * @throws \GeoIp2\Exception\OutOfQueriesException if your account is out
* by MaxMind's GeoIP2 web service. * of queries.
* @throws \GeoIp2\Exception\InvalidRequestException} if your request was
* received by the web service but is invalid for some other reason.
* This may indicate an issue with this API. Please report the error to
* MaxMind.
* @throws \GeoIp2\Exception\HttpException if an unexpected HTTP error
* code or message was returned. This could indicate a problem with the
* connection between your server and the web service or that the web
* service returned an invalid document or 500 error code.
* @throws \GeoIp2\Exception\GeoIp2Exception This serves as the parent
* class to the above exceptions. It will be thrown directly if a 200
* status code is returned but the body is invalid.
*/ */
public function cityIspOrg($ipAddress = 'me') public function cityIspOrg($ipAddress = 'me')
{ {
@@ -171,12 +184,23 @@ class Client
* *
* @return \GeoIp2\Model\Omni * @return \GeoIp2\Model\Omni
* *
* @throws \GeoIp2\Exception\GeoIp2Exception if there was a generic * @throws \GeoIp2\Exception\AddressNotFoundException if the address you
* error processing your request. * provided is not in our database (e.g., a private address).
* @throws \GeoIp2\Exception\HttpException if there was an HTTP transport * @throws \GeoIp2\Exception\AuthenticationException if there is a problem
* error. * with the user ID or license key that you provided.
* @throws \GeoIp2\Exception\WebServiceException if an error was returned * @throws \GeoIp2\Exception\OutOfQueriesException if your account is out
* by MaxMind's GeoIP2 web service. * of queries.
* @throws \GeoIp2\Exception\InvalidRequestException} if your request was
* received by the web service but is invalid for some other reason.
* This may indicate an issue with this API. Please report the error to
* MaxMind.
* @throws \GeoIp2\Exception\HttpException if an unexpected HTTP error
* code or message was returned. This could indicate a problem with the
* connection between your server and the web service or that the web
* service returned an invalid document or 500 error code.
* @throws \GeoIp2\Exception\GeoIp2Exception This serves as the parent
* class to the above exceptions. It will be thrown directly if a 200
* status code is returned but the body is invalid.
*/ */
public function omni($ipAddress = 'me') public function omni($ipAddress = 'me')
{ {
@@ -191,11 +215,8 @@ class Client
$this->guzzleClient : new GuzzleClient(); $this->guzzleClient : new GuzzleClient();
$request = $client->get($uri, array('Accept' => 'application/json')); $request = $client->get($uri, array('Accept' => 'application/json'));
$request->setAuth($this->userId, $this->licenseKey); $request->setAuth($this->userId, $this->licenseKey);
$ua = $request->getHeader('User-Agent'); $this->setUserAgent($request);
$ua = "GeoIP2 PHP API ($ua)";
$request->setHeader('User-Agent', $ua);
$response = null;
try { try {
$response = $request->send(); $response = $request->send();
} catch (ClientErrorResponseException $e) { } catch (ClientErrorResponseException $e) {
@@ -207,7 +228,7 @@ class Client
if ($response && $response->isSuccessful()) { if ($response && $response->isSuccessful()) {
$body = $this->handleSuccess($response, $uri); $body = $this->handleSuccess($response, $uri);
$class = "GeoIp2\\Model\\" . $class; $class = "GeoIp2\\Model\\" . $class;
return new $class($body, $this->languages); return new $class($body, $this->locales);
} else { } else {
$this->handleNon200($response, $uri); $this->handleNon200($response, $uri);
} }
@@ -237,8 +258,6 @@ class Client
{ {
$status = $response->getStatusCode(); $status = $response->getStatusCode();
$body = array();
if ($response->getContentLength() > 0) { if ($response->getContentLength() > 0) {
if (strstr($response->getContentType(), 'json')) { if (strstr($response->getContentType(), 'json')) {
try { try {
@@ -273,8 +292,7 @@ class Client
$uri $uri
); );
} }
$this->handleWebServiceError(
throw new WebServiceException(
$body['error'], $body['error'],
$body['code'], $body['code'],
$status, $status,
@@ -282,6 +300,28 @@ class Client
); );
} }
private function handleWebServiceError($message, $code, $status, $uri)
{
switch ($code) {
case 'IP_ADDRESS_NOT_FOUND':
case 'IP_ADDRESS_RESERVED':
throw new AddressNotFoundException($message);
case 'AUTHORIZATION_INVALID':
case 'LICENSE_KEY_REQUIRED':
case 'USER_ID_REQUIRED':
throw new AuthenticationException($message);
case 'OUT_OF_QUERIES':
throw new OutOfQueriesException($message);
default:
throw new InvalidRequestException(
$message,
$code,
$status,
$uri
);
}
}
private function handle5xx($response, $uri) private function handle5xx($response, $uri)
{ {
$status = $response->getStatusCode(); $status = $response->getStatusCode();
@@ -305,6 +345,13 @@ class Client
); );
} }
private function setUserAgent($request)
{
$userAgent = $request->getHeader('User-Agent');
$userAgent = "GeoIP2 PHP API ($userAgent)";
$request->setHeader('User-Agent', $userAgent);
}
private function baseUri() private function baseUri()
{ {
return 'https://' . $this->host . '/geoip/v2.0'; return 'https://' . $this->host . '/geoip/v2.0';

View File

@@ -0,0 +1,80 @@
<?php
namespace GeoIp2\Test\WebService;
use GeoIp2\Database\Reader;
class ReaderTest extends \PHPUnit_Framework_TestCase
{
public function testDefaultLocale()
{
$reader = new Reader('maxmind-db/test-data/GeoIP2-City-Test.mmdb');
// Needed for PHP 5.3
$that = $this;
$this->checkAllMethods(
function ($method) use (&$that, &$reader) {
$record = $reader->$method('81.2.69.160');
$that->assertEquals('United Kingdom', $record->country->name);
}
);
$reader->close();
}
public function testLocaleList()
{
$reader = new Reader(
'maxmind-db/test-data/GeoIP2-City-Test.mmdb',
array('xx', 'ru', 'pt-BR', 'es', 'en')
);
$that = $this;
$this->checkAllMethods(
function ($method) use (&$that, &$reader) {
$record = $reader->$method('81.2.69.160');
$that->assertEquals('Великобритания', $record->country->name);
}
);
$reader->close();
}
public function testHasIpAddress()
{
$reader = new Reader('maxmind-db/test-data/GeoIP2-City-Test.mmdb');
$that = $this;
$this->checkAllMethods(
function ($method) use (&$that, &$reader) {
$record = $reader->$method('81.2.69.160');
$that->assertEquals('81.2.69.160', $record->traits->ipAddress);
}
);
$reader->close();
}
/**
* @expectedException GeoIp2\Exception\AddressNotFoundException
* @expectedExceptionMessage The address 10.10.10.10 is not in the database.
*/
public function testUnknownAddress()
{
$reader = new Reader('maxmind-db/test-data/GeoIP2-City-Test.mmdb');
$reader->city('10.10.10.10');
$reader->close();
}
/**
* @expectedException InvalidArgumentException
* @expectedExceptionMessage is not a valid IP address
*/
public function testInvalidAddress()
{
$reader = new Reader('maxmind-db/test-data/GeoIP2-City-Test.mmdb');
$reader->city('invalid');
$reader->close();
}
public function checkAllMethods($testCb)
{
foreach (array('city', 'cityIspOrg', 'country', 'omni') as $method) {
$testCb($method);
}
}
}

View File

@@ -11,17 +11,17 @@ class CountryTest extends \PHPUnit_Framework_TestCase
'continent' => array( 'continent' => array(
'code' => 'NA', 'code' => 'NA',
'geoname_id' => 42, 'geoname_id' => 42,
'names' => array( 'en' => 'North America' ), 'names' => array('en' => 'North America'),
), ),
'country' => array( 'country' => array(
'geoname_id' => 1, 'geoname_id' => 1,
'iso_code' => 'US', 'iso_code' => 'US',
'names' => array( 'en' => 'United States of America' ), 'names' => array('en' => 'United States of America'),
), ),
'registered_country' => array( 'registered_country' => array(
'geoname_id' => 2, 'geoname_id' => 2,
'iso_code' => 'CA', 'iso_code' => 'CA',
'names' => array( 'en' => 'Canada' ), 'names' => array('en' => 'Canada'),
), ),
'traits' => array( 'traits' => array(
'ip_address' => '1.2.3.4', 'ip_address' => '1.2.3.4',
@@ -30,12 +30,12 @@ class CountryTest extends \PHPUnit_Framework_TestCase
private $model; private $model;
public function setUp () public function setUp()
{ {
$this->model = new Country($this->raw, array('en')); $this->model = new Country($this->raw, array('en'));
} }
public function testObjects () public function testObjects()
{ {
$this->assertInstanceOf( $this->assertInstanceOf(
'GeoIp2\Model\Country', 'GeoIp2\Model\Country',
@@ -80,7 +80,7 @@ class CountryTest extends \PHPUnit_Framework_TestCase
); );
$this->assertEquals( $this->assertEquals(
array( 'en' => 'North America' ), array('en' => 'North America'),
$this->model->continent->names, $this->model->continent->names,
'continent names' 'continent names'
); );
@@ -104,7 +104,7 @@ class CountryTest extends \PHPUnit_Framework_TestCase
); );
$this->assertEquals( $this->assertEquals(
array( 'en' => 'United States of America' ), array('en' => 'United States of America'),
$this->model->country->names, $this->model->country->names,
'country name' 'country name'
); );
@@ -134,7 +134,7 @@ class CountryTest extends \PHPUnit_Framework_TestCase
); );
$this->assertEquals( $this->assertEquals(
array( 'en' => 'Canada' ), array('en' => 'Canada'),
$this->model->registeredCountry->names, $this->model->registeredCountry->names,
'registered_country names' 'registered_country names'
); );
@@ -145,7 +145,7 @@ class CountryTest extends \PHPUnit_Framework_TestCase
'registered_country name is Canada' 'registered_country name is Canada'
); );
foreach (array( 'isAnonymousProxy', 'isSatelliteProvider' ) as $meth) { foreach (array('isAnonymousProxy', 'isSatelliteProvider') as $meth) {
$this->assertEquals( $this->assertEquals(
0, 0,
$this->model->traits->$meth, $this->model->traits->$meth,

View File

@@ -31,7 +31,7 @@ class NameTest extends \PHPUnit_Framework_TestCase
public function testFallback() public function testFallback()
{ {
$model = new Country($this->raw, array( 'ru', 'zh-CN', 'en' )); $model = new Country($this->raw, array('ru', 'zh-CN', 'en'));
$this->assertEquals( $this->assertEquals(
'北美洲', '北美洲',

View File

@@ -14,18 +14,18 @@ class OmniTest extends \PHPUnit_Framework_TestCase
'city' => array( 'city' => array(
'confidence' => 76, 'confidence' => 76,
'geoname_id' => 9876, 'geoname_id' => 9876,
'names' => array( 'en' => 'Minneapolis' ), 'names' => array('en' => 'Minneapolis'),
), ),
'continent' => array( 'continent' => array(
'code' => 'NA', 'code' => 'NA',
'geoname_id' => 42, 'geoname_id' => 42,
'names' => array( 'en' => 'North America' ), 'names' => array('en' => 'North America'),
), ),
'country' => array( 'country' => array(
'confidence' => 99, 'confidence' => 99,
'geoname_id' => 1, 'geoname_id' => 1,
'iso_code' => 'US', 'iso_code' => 'US',
'names' => array( 'en' => 'United States of America' ), 'names' => array('en' => 'United States of America'),
), ),
'location' => array( 'location' => array(
'accuracy_radius' => 1500, 'accuracy_radius' => 1500,
@@ -42,19 +42,19 @@ class OmniTest extends \PHPUnit_Framework_TestCase
'registered_country' => array( 'registered_country' => array(
'geoname_id' => 2, 'geoname_id' => 2,
'iso_code' => 'CA', 'iso_code' => 'CA',
'names' => array( 'en' => 'Canada' ), 'names' => array('en' => 'Canada'),
), ),
'represented_country' => array( 'represented_country' => array(
'geoname_id' => 3, 'geoname_id' => 3,
'iso_code' => 'GB', 'iso_code' => 'GB',
'names' => array( 'en' => 'United Kingdom' ), 'names' => array('en' => 'United Kingdom'),
), ),
'subdivisions' => array( 'subdivisions' => array(
array( array(
'confidence' => 88, 'confidence' => 88,
'geoname_id' => 574635, 'geoname_id' => 574635,
'iso_code' => 'MN', 'iso_code' => 'MN',
'names' => array( 'en' => 'Minnesota' ), 'names' => array('en' => 'Minnesota'),
) )
), ),
'traits' => array( 'traits' => array(
@@ -145,7 +145,7 @@ class OmniTest extends \PHPUnit_Framework_TestCase
public function testEmptyObjects() public function testEmptyObjects()
{ {
$raw = array( 'traits' => array( 'ip_address' => '5.6.7.8' ) ); $raw = array('traits' => array('ip_address' => '5.6.7.8'));
$model = new Omni($raw, array('en')); $model = new Omni($raw, array('en'));
@@ -220,14 +220,14 @@ class OmniTest extends \PHPUnit_Framework_TestCase
public function testUnknown() public function testUnknown()
{ {
$raw = array( $raw = array(
'new_top_level' => array( 'foo' => 42 ), 'new_top_level' => array('foo' => 42),
'city' => array( 'city' => array(
'confidence' => 76, 'confidence' => 76,
'geoname_id_id' => 9876, 'geoname_id_id' => 9876,
'names' => array( 'en' => 'Minneapolis' ), 'names' => array('en' => 'Minneapolis'),
'population' => 50, 'population' => 50,
), ),
'traits' => array( 'ip_address' => '5.6.7.8' ) 'traits' => array('ip_address' => '5.6.7.8')
); );
// checking whether there are exceptions with unknown keys // checking whether there are exceptions with unknown keys

View File

@@ -15,12 +15,12 @@ class ClientTest extends \PHPUnit_Framework_TestCase
'continent' => array( 'continent' => array(
'code' => 'NA', 'code' => 'NA',
'geoname_id' => 42, 'geoname_id' => 42,
'names' => array( 'en' => 'North America' ), 'names' => array('en' => 'North America'),
), ),
'country' => array( 'country' => array(
'geoname_id' => 1, 'geoname_id' => 1,
'iso_code' => 'US', 'iso_code' => 'US',
'names' => array( 'en' => 'United States of America' ), 'names' => array('en' => 'United States of America'),
), ),
'maxmind' => array('queries_remaining' => 11), 'maxmind' => array('queries_remaining' => 11),
'traits' => array( 'traits' => array(
@@ -44,7 +44,7 @@ class ClientTest extends \PHPUnit_Framework_TestCase
), ),
'1.2.3.5' => $this->response('country', 200), '1.2.3.5' => $this->response('country', 200),
'2.2.3.5' => $this->response('country', 200, 'bad body'), '2.2.3.5' => $this->response('country', 200, 'bad body'),
'1.2.3.6'=> $this->response( '1.2.3.6' => $this->response(
'error', 'error',
400, 400,
array( array(
@@ -59,7 +59,7 @@ class ClientTest extends \PHPUnit_Framework_TestCase
'1.2.3.8' => $this->response( '1.2.3.8' => $this->response(
'error', 'error',
400, 400,
array( 'weird' => 42 ) array('weird' => 42)
), ),
'1.2.3.9' => $this->response( '1.2.3.9' => $this->response(
'error', 'error',
@@ -82,6 +82,54 @@ class ClientTest extends \PHPUnit_Framework_TestCase
null, null,
'text/plain' 'text/plain'
), ),
'1.2.3.13' => $this->response(
'error',
404,
array(
'code' => 'IP_ADDRESS_NOT_FOUND',
'error' => 'The address "1.2.3.13" is not in our database.'
)
),
'1.2.3.14' => $this->response(
'error',
400,
array(
'code' => 'IP_ADDRESS_RESERVED',
'error' => 'The address "1.2.3.14" is a private address.'
)
),
'1.2.3.15' => $this->response(
'error',
401,
array(
'code' => 'AUTHORIZATION_INVALID',
'error' => 'A user ID and license key are required to use this service'
)
),
'1.2.3.16' => $this->response(
'error',
401,
array(
'code' => 'LICENSE_KEY_REQUIRED',
'error' => 'A license key is required to use this service'
)
),
'1.2.3.17' => $this->response(
'error',
401,
array(
'code' => 'USER_ID_REQUIRED',
'error' => 'A user ID is required to use this service'
)
),
'1.2.3.18' => $this->response(
'error',
402,
array(
'code' => 'OUT_OF_QUERIES',
'error' => 'The license key you have provided is out of queries.'
)
),
); );
return $responses[$ip]; return $responses[$ip];
} }
@@ -130,7 +178,7 @@ class ClientTest extends \PHPUnit_Framework_TestCase
); );
$this->assertEquals( $this->assertEquals(
array( 'en' => 'United States of America' ), array('en' => 'United States of America'),
$country->country->names, $country->country->names,
'country names' 'country names'
); );
@@ -207,7 +255,7 @@ class ClientTest extends \PHPUnit_Framework_TestCase
/** /**
* @expectedException GeoIp2\Exception\WebServiceException * @expectedException GeoIp2\Exception\InvalidRequestException
* @expectedExceptionCode 400 * @expectedExceptionCode 400
* @expectedExceptionMessage The value "1.2.3" is not a valid ip address * @expectedExceptionMessage The value "1.2.3" is not a valid ip address
*/ */
@@ -216,7 +264,6 @@ class ClientTest extends \PHPUnit_Framework_TestCase
$client = $this->client($this->getResponse('1.2.3.6')); $client = $this->client($this->getResponse('1.2.3.6'));
$client->country('1.2.3.6'); $client->country('1.2.3.6');
} }
/** /**
@@ -293,6 +340,72 @@ class ClientTest extends \PHPUnit_Framework_TestCase
$client->country('1.2.3.12'); $client->country('1.2.3.12');
} }
/**
* @expectedException GeoIp2\Exception\AddressNotFoundException
* @expectedExceptionMessage The address "1.2.3.13" is not in our database.
*/
public function testAddressNotFoundException()
{
$client = $this->client($this->getResponse('1.2.3.13'));
$client->country('1.2.3.13');
}
/**
* @expectedException GeoIp2\Exception\AddressNotFoundException
* @expectedExceptionMessage The address "1.2.3.14" is a private address.
*/
public function testAddressReservedException()
{
$client = $this->client($this->getResponse('1.2.3.14'));
$client->country('1.2.3.14');
}
/**
* @expectedException GeoIp2\Exception\AuthenticationException
* @expectedExceptionMessage A user ID and license key are required to use this service
*/
public function testAuthorizationException()
{
$client = $this->client($this->getResponse('1.2.3.15'));
$client->country('1.2.3.15');
}
/**
* @expectedException GeoIp2\Exception\AuthenticationException
* @expectedExceptionMessage A license key is required to use this service
*/
public function testMissingLicenseKeyException()
{
$client = $this->client($this->getResponse('1.2.3.16'));
$client->country('1.2.3.16');
}
/**
* @expectedException GeoIp2\Exception\AuthenticationException
* @expectedExceptionMessage A user ID is required to use this service
*/
public function testMissingUserIdException()
{
$client = $this->client($this->getResponse('1.2.3.17'));
$client->country('1.2.3.17');
}
/**
* @expectedException GeoIp2\Exception\OutOfQueriesException
* @expectedExceptionMessage The license key you have provided is out of queries.
*/
public function testOutOfQueriesException()
{
$client = $this->client($this->getResponse('1.2.3.18'));
$client->country('1.2.3.18');
}
public function testParams() public function testParams()
{ {
$plugin = new MockPlugin(); $plugin = new MockPlugin();
@@ -337,7 +450,7 @@ class ClientTest extends \PHPUnit_Framework_TestCase
} }
private function client($response, $languages = array('en')) private function client($response, $locales = array('en'))
{ {
$plugin = new MockPlugin(); $plugin = new MockPlugin();
$plugin->addResponse($response); $plugin->addResponse($response);
@@ -347,7 +460,7 @@ class ClientTest extends \PHPUnit_Framework_TestCase
$client = new Client( $client = new Client(
42, 42,
'abcdef123456', 'abcdef123456',
$languages, $locales,
'geoip.maxmind.com', 'geoip.maxmind.com',
$guzzleClient $guzzleClient
); );
@@ -365,7 +478,7 @@ class ClientTest extends \PHPUnit_Framework_TestCase
$headers = array(); $headers = array();
if ($contentType) { if ($contentType) {
$headers['Content-Type'] = $contentType; $headers['Content-Type'] = $contentType;
} elseif ($status == 200 || ( $status >= 400 && $status < 500 )) { } elseif ($status == 200 || ($status >= 400 && $status < 500)) {
$headers['Content-Type'] = 'application/vnd.maxmind.com-' $headers['Content-Type'] = 'application/vnd.maxmind.com-'
. $endpoint . '+json; charset=UTF-8; version=1.0;'; . $endpoint . '+json; charset=UTF-8; version=1.0;';
} }

View File

@@ -1,6 +1,6 @@
<?php <?php
if (!$loader = @include __DIR__.'/../vendor/autoload.php') { if (!$loader = @include __DIR__ . '/../vendor/autoload.php') {
die('Project dependencies missing'); die('Project dependencies missing');
} }