mirror of
				https://github.com/alchemy-fr/Phraseanet.git
				synced 2025-10-24 18:33:13 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			270 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			270 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| /**
 | |
|  * This file is part of Phraseanet
 | |
|  *
 | |
|  * (c) 2005-2016 Alchemy
 | |
|  *
 | |
|  * For the full copyright and license information, please view the LICENSE
 | |
|  * file that was distributed with this source code.
 | |
|  */
 | |
| 
 | |
| namespace Alchemy\Phrasea\TaskManager\Job;
 | |
| 
 | |
| use Alchemy\Phrasea\Application;
 | |
| use Alchemy\Phrasea\Core\PhraseaTokens;
 | |
| use Alchemy\Phrasea\Metadata\TagFactory;
 | |
| use Alchemy\Phrasea\TaskManager\Editor\WriteMetadataEditor;
 | |
| use PHPExiftool\Driver\Metadata;
 | |
| use PHPExiftool\Driver\Value;
 | |
| use PHPExiftool\Driver\Tag;
 | |
| use PHPExiftool\Exception\ExceptionInterface as PHPExiftoolException;
 | |
| use PHPExiftool\Writer as ExifWriter;
 | |
| use PHPExiftool\Exception\TagUnknown;
 | |
| use PHPExiftool\Writer;
 | |
| 
 | |
| class WriteMetadataJob extends AbstractJob
 | |
| {
 | |
|     /**
 | |
|      * {@inheritdoc}
 | |
|      */
 | |
|     public function getName()
 | |
|     {
 | |
|         return $this->translator->trans('task::writemeta:ecriture des metadatas');
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * {@inheritdoc}
 | |
|      */
 | |
|     public function getJobId()
 | |
|     {
 | |
|         return 'WriteMetadata';
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * {@inheritdoc}
 | |
|      */
 | |
|     public function getDescription()
 | |
|     {
 | |
|         return $this->translator->trans("task::writemeta:(re)ecriture des metadatas dans les documents (et subdefs concernees)");
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * {@inheritdoc}
 | |
|      */
 | |
|     public function getEditor()
 | |
|     {
 | |
|         return new WriteMetadataEditor($this->translator);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * {@inheritdoc}
 | |
|      */
 | |
|     protected function doJob(JobData $jobData)
 | |
|     {
 | |
|         $settings = simplexml_load_string($jobData->getTask()->getSettings());
 | |
|         $clearDoc = (bool) (string) $settings->cleardoc;
 | |
|         $MWG = (bool) (string) $settings->mwg;
 | |
| 
 | |
|         foreach ($jobData->getApplication()->getDataboxes() as $databox) {
 | |
|             $connection = $databox->get_connection();
 | |
| 
 | |
|             $statement = $connection->prepare('SELECT record_id, coll_id, jeton FROM record WHERE (jeton & :token > 0)');
 | |
|             $statement->execute(['token' => PhraseaTokens::WRITE_META]);
 | |
|             $rs = $statement->fetchAll(\PDO::FETCH_ASSOC);
 | |
|             $statement->closeCursor();
 | |
| 
 | |
|             foreach ($rs as $row) {
 | |
|                 $record_id = $row['record_id'];
 | |
|                 $token = $row['jeton'];
 | |
| 
 | |
|                 $record = $databox->get_record($record_id);
 | |
|                 $type = $record->getType();
 | |
| 
 | |
|                 $subdefs = [];
 | |
|                 foreach ($record->get_subdefs() as $name => $subdef) {
 | |
|                     $write_document = (($token & PhraseaTokens::WRITE_META_DOC) && $name == 'document');
 | |
|                     $write_subdef = (($token & PhraseaTokens::WRITE_META_SUBDEF) && $this->isSubdefMetadataUpdateRequired($databox, $type, $name));
 | |
| 
 | |
|                     if (($write_document || $write_subdef) && $subdef->is_physically_present()) {
 | |
|                         $subdefs[$name] = $subdef->getRealPath();
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 $metadata = new Metadata\MetadataBag();
 | |
| 
 | |
|                 if ($record->getUuid()) {
 | |
|                     $metadata->add(
 | |
|                         new Metadata\Metadata(
 | |
|                             new Tag\XMPExif\ImageUniqueID(),
 | |
|                             new Value\Mono($record->getUuid())
 | |
|                         )
 | |
|                     );
 | |
|                     $metadata->add(
 | |
|                         new Metadata\Metadata(
 | |
|                             new Tag\ExifIFD\ImageUniqueID(),
 | |
|                             new Value\Mono($record->getUuid())
 | |
|                         )
 | |
|                     );
 | |
|                     $metadata->add(
 | |
|                         new Metadata\Metadata(
 | |
|                             new Tag\IPTC\UniqueDocumentID(),
 | |
|                             new Value\Mono($record->getUuid())
 | |
|                         )
 | |
|                     );
 | |
|                 }
 | |
| 
 | |
|                 $caption = $record->get_caption();
 | |
|                 foreach($databox->get_meta_structure() as $fieldStructure) {
 | |
| 
 | |
|                     $tagName = $fieldStructure->get_tag()->getTagname();
 | |
|                     $fieldName = $fieldStructure->get_name();
 | |
| 
 | |
|                     // skip fields with no src
 | |
|                     if($tagName == '' || $tagName == 'Phraseanet:no-source') {
 | |
|                         continue;
 | |
|                     }
 | |
| 
 | |
|                     // check exiftool known tags to skip Phraseanet:tf-*
 | |
|                     try {
 | |
|                         $tag = TagFactory::getFromRDFTagname($tagName);
 | |
|                         if(!$tag->isWritable()) {
 | |
|                             continue;
 | |
|                         }
 | |
|                     } catch (TagUnknown $e) {
 | |
|                         continue;
 | |
|                     }
 | |
| 
 | |
|                     try {
 | |
|                         $field = $caption->get_field($fieldName);
 | |
|                         $fieldValues = $field->get_values();
 | |
| 
 | |
|                         if ($fieldStructure->is_multi()) {
 | |
|                             $values = array();
 | |
|                             foreach ($fieldValues as $value) {
 | |
|                                 $values[] = $this->removeNulChar($value->getValue());
 | |
|                             }
 | |
| 
 | |
|                             $value = new Value\Multi($values);
 | |
|                         } else {
 | |
|                             $fieldValue = array_pop($fieldValues);
 | |
|                             $value = $this->removeNulChar($fieldValue->getValue());
 | |
| 
 | |
|                             // fix the dates edited into phraseanet
 | |
|                             if($fieldStructure->get_type() === $fieldStructure::TYPE_DATE) {
 | |
|                                 try {
 | |
|                                     $value = self::fixDate($value); // will return NULL if the date is not valid
 | |
|                                 }
 | |
|                                 catch (\Exception $e) {
 | |
|                                     $value = null;    // do NOT write back to iptc
 | |
|                                 }
 | |
|                             }
 | |
| 
 | |
|                             if($value !== null) {   // do not write invalid dates
 | |
|                                 $value = new Value\Mono($value);
 | |
|                             }
 | |
|                         }
 | |
|                     } catch (\Exception $e) {
 | |
|                         // the field is not set in the record, erase it
 | |
|                         if ($fieldStructure->is_multi()) {
 | |
|                             $value = new Value\Multi(array(''));
 | |
|                         } else {
 | |
|                             $value = new Value\Mono('');
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     if($value !== null) {   // do not write invalid data
 | |
|                         $metadata->add(
 | |
|                             new Metadata\Metadata($fieldStructure->get_tag(), $value)
 | |
|                         );
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 $writer = $this->getMetadataWriter($jobData->getApplication());
 | |
|                 $writer->reset();
 | |
| 
 | |
|                 if($MWG) {
 | |
|                     $writer->setModule(ExifWriter::MODULE_MWG, true);
 | |
|                 }
 | |
| 
 | |
|                 foreach ($subdefs as $name => $file) {
 | |
|                     $writer->erase($name != 'document' || $clearDoc, true);
 | |
|                     try {
 | |
|                         $writer->write($file, $metadata);
 | |
| 
 | |
|                         $this->log('info',sprintf('meta written for sbasid=%1$d - recordid=%2$d (%3$s)', $databox->get_sbas_id(), $record_id, $name));
 | |
|                     } catch (PHPExiftoolException $e) {
 | |
|                         $this->log('error',sprintf('meta NOT written for sbasid=%1$d - recordid=%2$d (%3$s) because "%s"', $databox->get_sbas_id(), $record_id, $name, $e->getMessage()));
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 $statement = $connection->prepare('UPDATE record SET jeton=jeton & ~:token WHERE record_id = :record_id');
 | |
|                 $statement->execute([
 | |
|                     'record_id' => $record_id,
 | |
|                     'token' => PhraseaTokens::WRITE_META,
 | |
|                 ]);
 | |
|                 $statement->closeCursor();
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Application $app
 | |
|      * @return Writer
 | |
|      */
 | |
|     private function getMetadataWriter(Application $app)
 | |
|     {
 | |
|         return $app['exiftool.writer'];
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param \databox $databox
 | |
|      * @param string $subdefType
 | |
|      * @param string $subdefName
 | |
|      * @return bool
 | |
|      */
 | |
|     private function isSubdefMetadataUpdateRequired(\databox $databox, $subdefType, $subdefName)
 | |
|     {
 | |
|         if ($databox->get_subdef_structure()->hasSubdef($subdefType, $subdefName)) {
 | |
|             return $databox->get_subdef_structure()->get_subdef($subdefType, $subdefName)->isMetadataUpdateRequired();
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     private function removeNulChar($value)
 | |
|     {
 | |
|         return str_replace("\0", "", $value);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * re-format a phraseanet date for iptc writing
 | |
|      * return NULL if the date is not valid
 | |
|      *
 | |
|      * @param string $value
 | |
|      * @return string|null
 | |
|      */
 | |
|     private static function fixDate($value)
 | |
|     {
 | |
|         $date = null;
 | |
|         try {
 | |
|             $a = explode(';', preg_replace('/\D+/', ';', trim($value)));
 | |
|             switch (count($a)) {
 | |
|                 case 3:     // yyyy;mm;dd
 | |
|                     $date = new \DateTime($a[0] . '-' . $a[1] . '-' . $a[2]);
 | |
|                     $date = $date->format('Y-m-d H:i:s');
 | |
|                     break;
 | |
|                 case 6:     // yyyy;mm;dd;hh;mm;ss
 | |
|                     $date = new \DateTime($a[0] . '-' . $a[1] . '-' . $a[2] . ' ' . $a[3] . ':' . $a[4] . ':' . $a[5]);
 | |
|                     $date = $date->format('Y-m-d H:i:s');
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
|         catch (\Exception $e) {
 | |
|             $date = null;
 | |
|         }
 | |
| 
 | |
|         return $date;
 | |
|     }
 | |
| }
 | 
