Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Issues/#25 #33

Merged
merged 37 commits into from
Apr 3, 2015
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
37 commits
Select commit Hold shift + click to select a range
cabf6a3
Add Mutator Hydrator for hydrating Exif objects
Mar 29, 2015
8f6cdc0
Add separate mappers for mapping data to Exif object format
Mar 29, 2015
bf6ad93
AdapterAbstract now uses Hydrator for setOptions; Removed obsolete me…
Mar 29, 2015
bb6f4e2
Extracted mapping of ExifTool data to separate mapper & updated Exift…
Mar 29, 2015
ae92380
Extracted mapping of native data to separate mapper & updatd Native a…
Mar 29, 2015
0ba6276
Bugfixed some small inconsistencies
Mar 29, 2015
fb7763f
Added mutators to Exif class & updated unit tests
Mar 29, 2015
ec812d9
Removed blank lines to conform to PSR2
Mar 29, 2015
3c36d12
Add unit test for Exiftool::setNumeric
Mar 30, 2015
2daf6e1
Add extra unit tests for Reader class
Mar 30, 2015
a288667
Add some tests for Exif class
Mar 30, 2015
d16cacb
Remove strict coverage option
Mar 30, 2015
4e0ee69
Made coverage more explicit
Mar 30, 2015
5eb8ca5
Add @covers annotation for Exif tests
Mar 30, 2015
cf21a3f
Add unit tests for Mutator class
Apr 1, 2015
2b7fbc6
Add extra unit tests for AdapterAbstract
Apr 1, 2015
281aebd
Add coverage for interfaces & exceptions
Apr 1, 2015
8214151
Beter interface coverage?
Apr 1, 2015
f4eaad8
Add coverage ignore tags
Apr 1, 2015
836ac7c
Add unit test for Exif mutators
Apr 2, 2015
06a75fd
Improved coverage annotations on Exif class
Apr 2, 2015
52bbeef
Removed unnecessary initialization of variable in Reader class
Apr 2, 2015
8fca517
Removed indenting level in Native mapper class
Apr 2, 2015
5899952
Improved docblocks & removed unused variable
Apr 2, 2015
32b4358
Changed docblocks for AbstractAdapter constructor
Apr 2, 2015
874d645
Removed obsolete ; in docblocks
Apr 2, 2015
86c4dac
Unit tests for Exiftool mapper
Apr 2, 2015
7b16cd4
Removed incorrect coverage tag
Apr 2, 2015
a2a0272
Unit test for Native mapper
Apr 3, 2015
cc6fbb7
Exclude Interfaces from coverage report
Apr 3, 2015
6dbc594
Added @covers annotation for the NoAdapterException
Apr 3, 2015
a862bea
Tried mocking proc_open function for coverage
Apr 3, 2015
795fab5
Corrected stubbing of root function
Apr 3, 2015
476367b
Add GPS code
Apr 3, 2015
1e2e623
More unit tests for Exif (GPS added)
Apr 3, 2015
401686c
Unit tests for GPS stuff
Apr 3, 2015
1d4a96b
Covered extra paths for Exiftool mapper test
Apr 3, 2015
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
112 changes: 59 additions & 53 deletions lib/PHPExif/Adapter/AdapterAbstract.php
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,9 @@

namespace PHPExif\Adapter;

use PHPExif\Mapper\MapperInterface;
use PHPExif\Hydrator\HydratorInterface;

/**
* PHP Exif Reader Adapter Abstract
*
Expand All @@ -21,10 +24,25 @@
*/
abstract class AdapterAbstract implements AdapterInterface
{
/**
* @var string
*/
protected $hydratorClass = '\\PHPExif\\Hydrator\\Mutator';

/**
* @var \PHPExif\Mapper\MapperInterface
*/
protected $mapper;

/**
* @var \PHPExif\Hydrator\HydratorInterface
*/
protected $hydrator;

/**
* Class constructor
*
* @param array $data Optional array of data to initialize the object with
* @param array $options Optional array of data to initialize the object with
*/
public function __construct(array $options = array())
{
Expand All @@ -34,88 +52,76 @@ public function __construct(array $options = array())
}

/**
* Set array of options in the current object
* Mutator for the data mapper
*
* @param array $options
* @return \PHPExif\Reader\AdapterAbstract
* @param \PHPExif\Mapper\MapperInterface $mapper
* @return \PHPExif\Adapter\AdapterInterface
*/
public function setOptions(array $options)
public function setMapper(MapperInterface $mapper)
{
foreach ($options as $property => $value) {
$setter = $this->determinePropertySetter($property);
if (method_exists($this, $setter)) {
$this->$setter($value);
}
}
$this->mapper = $mapper;

return $this;
}

/**
* Detemines the name of the getter method for given property name
* Accessor for the data mapper
*
* @param string $property The property to determine the getter for
* @return string The name of the getter method
* @return \PHPExif\Mapper\MapperInterface
*/
protected function determinePropertyGetter($property)
public function getMapper()
{
$method = 'get' . ucfirst($property);
return $method;
if (null === $this->mapper) {
// lazy load one
$mapper = new $this->mapperClass;

$this->setMapper($mapper);
}

return $this->mapper;
}

/**
* Detemines the name of the setter method for given property name
* Mutator for the hydrator
*
* @param string $property The property to determine the setter for
* @return string The name of the setter method
* @param \PHPExif\Hydrator\HydratorInterface $hydrator
* @return \PHPExif\Adapter\AdapterInterface
*/
protected function determinePropertySetter($property)
public function setHydrator(HydratorInterface $hydrator)
{
$method = 'set' . ucfirst($property);
return $method;
$this->hydrator = $hydrator;

return $this;
}

/**
* Get a list of the class constants prefixed with given $type
* Accessor for the data hydrator
*
* @param string $type
* @return array
* @return \PHPExif\Hydrator\HydratorInterface
*/
public function getClassConstantsOfType($type)
public function getHydrator()
{
$class = new \ReflectionClass(get_called_class());
$constants = $class->getConstants();

$list = array();
$type = strtoupper($type) . '_';
foreach ($constants as $key => $value) {
if (strpos($key, $type) === 0) {
$list[$key] = $value;
}
if (null === $this->hydrator) {
// lazy load one
$hydrator = new $this->hydratorClass;

$this->setHydrator($hydrator);
}
return $list;

return $this->hydrator;
}

/**
* Returns an array notation of current instance
* Set array of options in the current object
*
* @return array
* @param array $options
* @return \PHPExif\Reader\AdapterAbstract
*/
public function toArray()
public function setOptions(array $options)
{
$rc = new \ReflectionClass(get_class($this));
$properties = $rc->getProperties();
$arrResult = array();

foreach ($properties as $rp) {
/* @var $rp \ReflectionProperty */
$getter = $this->determinePropertyGetter($rp->getName());
if (!method_exists($this, $getter)) {
continue;
}
$arrResult[$rp->getName()] = $this->$getter();
}
$hydrator = $this->getHydrator();
$hydrator->hydrate($this, $options);

return $arrResult;
return $this;
}
}
1 change: 1 addition & 0 deletions lib/PHPExif/Adapter/AdapterInterface.php
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
* @license http://github.com/miljar/PHPExif/blob/master/LICENSE MIT License
* @category PHPExif
* @package Reader
* @codeCoverageIgnore
*/

namespace PHPExif\Adapter;
Expand Down
110 changes: 15 additions & 95 deletions lib/PHPExif/Adapter/Exiftool.php
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,6 @@
use PHPExif\Exif;
use InvalidArgumentException;
use RuntimeException;
use DateTime;

/**
* PHP Exif Exiftool Reader Adapter
Expand All @@ -40,6 +39,11 @@ class Exiftool extends AdapterAbstract
*/
protected $numeric = true;

/**
* @var string
*/
protected $mapperClass = '\\PHPExif\\Mapper\\Exiftool';

/**
* Setter for the exiftool binary path
*
Expand Down Expand Up @@ -109,8 +113,16 @@ public function getExifFromFile($file)
);

$data = json_decode($result, true);
$mappedData = $this->mapData(reset($data));
$exif = new Exif($mappedData);

// map the data:
$mapper = $this->getMapper();
$mapper->setNumeric($this->numeric);
$mappedData = $mapper->mapRawData(reset($data));

// hydrate a new Exif object
$exif = new Exif();
$hydrator = $this->getHydrator();
$hydrator->hydrate($exif, $mappedData);
$exif->setRawData(reset($data));

return $exif;
Expand Down Expand Up @@ -148,96 +160,4 @@ protected function getCliOutput($command)

return $result;
}

/**
* Maps native data to Exif format
*
* @param array $source
* @return array
*/
public function mapData(array $source)
{
$focalLength = false;
if (isset($source['FocalLength'])) {
$focalLengthParts = explode(' ', $source['FocalLength']);
$focalLength = (int) reset($focalLengthParts);
}

$exposureTime = false;
if (isset($source['ExposureTime'])) {
$exposureTime = '1/' . round(1 / $source['ExposureTime']);
}

$caption = false;
if (isset($source['Caption'])) {
$caption = $source['Caption'];
} elseif (isset($source['Caption-Abstract'])) {
$caption = $source['Caption-Abstract'];
}

$gpsLocation = false;
if (isset($source['GPSLatitudeRef']) && isset($source['GPSLongitudeRef'])) {
$latitude = $this->extractGPSCoordinates($source['GPSLatitude']);
$longitude = $this->extractGPSCoordinates($source['GPSLongitude']);

if ($latitude !== false && $longitude !== false) {
$gpsLocation = sprintf(
'%s,%s',
(strtoupper($source['GPSLatitudeRef'][0]) === 'S' ? -1 : 1) * $latitude,
(strtoupper($source['GPSLongitudeRef'][0]) === 'W' ? -1 : 1) * $longitude
);
}
}

return array(
Exif::APERTURE => (!isset($source['Aperture'])) ?
false : sprintf('f/%01.1f', $source['Aperture']),
Exif::AUTHOR => (!isset($source['Artist'])) ? false : $source['Artist'],
Exif::CAMERA => (!isset($source['Model'])) ? false : $source['Model'],
Exif::CAPTION => $caption,
Exif::COLORSPACE => (!isset($source[Exif::COLORSPACE]) ? false : $source[Exif::COLORSPACE]),
Exif::COPYRIGHT => (!isset($source['Copyright'])) ? false : $source['Copyright'],
Exif::CREATION_DATE => (!isset($source['CreateDate'])) ?
false : DateTime::createFromFormat('Y:m:d H:i:s', $source['CreateDate']),
Exif::CREDIT => (!isset($source['Credit'])) ? false : $source['Credit'],
Exif::EXPOSURE => $exposureTime,
Exif::FILESIZE => (!isset($source[Exif::FILESIZE]) ? false : $source[Exif::FILESIZE]),
Exif::FOCAL_LENGTH => $focalLength,
Exif::FOCAL_DISTANCE => (!isset($source['ApproximateFocusDistance'])) ?
false : sprintf('%1$sm', $source['ApproximateFocusDistance']),
Exif::HEADLINE => (!isset($source['Headline'])) ? false : $source['Headline'],
Exif::HEIGHT => (!isset($source['ImageHeight'])) ? false : $source['ImageHeight'],
Exif::HORIZONTAL_RESOLUTION => (!isset($source['XResolution'])) ? false : $source['XResolution'],
Exif::ISO => (!isset($source['ISO'])) ? false : $source['ISO'],
Exif::JOB_TITLE => (!isset($source['JobTitle'])) ? false : $source['JobTitle'],
Exif::KEYWORDS => (!isset($source['Keywords'])) ? false : $source['Keywords'],
Exif::MIMETYPE => (!isset($source['MIMEType'])) ? false : $source['MIMEType'],
Exif::ORIENTATION => (!isset($source['Orientation'])) ? false : $source['Orientation'],
Exif::SOFTWARE => (!isset($source['Software'])) ? false : $source['Software'],
Exif::SOURCE => (!isset($source['Source'])) ? false : $source['Source'],
Exif::TITLE => (!isset($source['Title'])) ? false : $source['Title'],
Exif::VERTICAL_RESOLUTION => (!isset($source['YResolution'])) ? false : $source['YResolution'],
Exif::WIDTH => (!isset($source['ImageWidth'])) ? false : $source['ImageWidth'],
Exif::GPS => $gpsLocation,
);
}

/**
* Extract GPS coordinates from formatted string
*
* @param string $coordinates
* @return array
*/
protected function extractGPSCoordinates($coordinates)
{
if ($this->numeric === true) {
return abs((float) $coordinates);
} else {
if (!preg_match('!^([0-9.]+) deg ([0-9.]+)\' ([0-9.]+)"!', $coordinates, $matches)) {
return false;
}

return intval($matches[1]) + (intval($matches[2]) / 60) + (floatval($matches[3]) / 3600);
}
}
}
Loading