<?php loco_require_lib('compiled/gettext.php'); /** * String extraction from source code. */ class Loco_gettext_Extraction { /** * @var Loco_package_Bundle */ private $bundle; /** * @var LocoExtracted */ private $extracted; /** * Extra strings to be pushed into domains * @var array */ private $extras = []; /** * List of files skipped due to memory limit * @var Loco_fs_FileList|null */ private $skipped; /** * Size in bytes of largest file encountered * @var int */ private $maxbytes = 0; /** * Initialize extractor for a given bundle */ public function __construct( Loco_package_Bundle $bundle ){ loco_check_extension('ctype'); if( ! loco_check_extension('tokenizer') ){ throw new Loco_error_Exception('String extraction not available without required extension'); } $this->bundle = $bundle; $this->extracted = new LocoExtracted; $this->extracted->setDomain('default'); $default = $bundle->getDefaultProject(); if( $default instanceof Loco_package_Project ){ $domain = $default->getDomain()->getName(); // wildcard stands in for empty text domain, meaning unspecified or dynamic domains will be included. // note that strings intended to be in "default" domain must specify explicitly, or be included here too. if( '*' === $domain ){ $domain = ''; $this->extracted->setDomain(''); } // pull bundle's default metadata. these are translations that may not be encountered in files $extras = []; $header = $bundle->getHeaderInfo(); foreach( $bundle->getMetaTranslatable() as $prop => $notes ){ $text = $header->__get($prop); if( is_string($text) && '' !== $text ){ $extras[] = ['source'=>$text, 'notes'=>$notes ]; } } if( $extras ){ $this->extras[$domain] = $extras; } } } /** * @return self */ public function addProject( Loco_package_Project $project ){ $base = $this->bundle->getDirectoryPath(); $domain = (string) $project->getDomain(); // skip files larger than configured maximum $opts = Loco_data_Settings::get(); $max = wp_convert_hr_to_bytes( $opts->max_php_size ); // *attempt* to raise memory limit to WP_MAX_MEMORY_LIMIT if( function_exists('wp_raise_memory_limit') ){ wp_raise_memory_limit('loco'); } /* @var Loco_fs_File $file */ foreach( $project->findSourceFiles() as $file ){ $type = $opts->ext2type( $file->extension() ); $fileref = $file->getRelativePath($base); try { $extr = loco_wp_extractor( $type, $file->fullExtension() ); if( 'php' === $type || 'twig' === $type) { // skip large files for PHP, because token_get_all is hungry if( 0 !== $max ){ $size = $file->size(); $this->maxbytes = max( $this->maxbytes, $size ); if( $size > $max ){ $list = $this->skipped or $list = ( $this->skipped = new Loco_fs_FileList() ); $list->add( $file ); continue; } } // extract headers from theme files (templates and patterns) if( $project->getBundle()->isTheme() ){ $extr->headerize( [ 'Template Name' => ['notes'=>'Name of the template'], ], $domain ); if( preg_match('!^patterns/!', $fileref) ){ $extr->headerize([ 'Title' => ['context'=>'Pattern title'], 'Description' => ['context'=>'Pattern description'], ], $domain ); } } } // normally missing domains are treated as "default", but we'll make an exception for theme.json. else if( 'json' === $type && $project->getBundle()->isTheme() ){ $extr->setDomain($domain); } $this->extracted->extractSource( $extr, $file->getContents(), $fileref ); } catch( Exception $e ){ Loco_error_AdminNotices::debug('Error extracting '.$fileref.': '.$e->getMessage() ); } } return $this; } private function includeBlock( Loco_fs_File $file, $domain ) { $def = json_decode( $file->getContents(), true ); if( ! is_array($def) || ! array_key_exists('$schema',$def) ){ return; } // adding dummy line number for well-formed file reference, not currently pulling line number. $ref = $file->getRelativePath( $this->bundle->getDirectoryPath() ).':1'; foreach(['title','description','keywords'] as $key ){ if( ! array_key_exists($key,$def) ) { continue; } $msgctxt = 'block '.rtrim($key,'s'); foreach( (array) $def[$key] as $msgid ){ if( is_string($msgid) && '' !== $msgid ){ $str = new Loco_gettext_String($msgid,$msgctxt); $str->addFileReferences($ref); $this->addString($str,$domain); } } } } /** * Add metadata strings deferred from construction. Note this will alter domain counts * @return self */ public function includeMeta(){ foreach( $this->extras as $domain => $extras ){ foreach( $extras as $entry ){ $this->extracted->pushEntry($entry,$domain); } } $this->extras = []; return $this; } /** * Add a custom source string constructed from `new Loco_gettext_String(msgid,[msgctxt])` * @param Loco_gettext_String $string * @param string $domain Optional text domain, if not current bundle's default * @return void */ public function addString( Loco_gettext_String $string, $domain = '' ){ if( ! $domain ) { $default = $this->bundle->getDefaultProject(); $domain = (string) ( $default ? $default->getDomain() : $this->extracted->getDomain() ); } $index = $this->extracted->pushEntry( $string->exportSingular(), $domain ); if( $string->hasPlural() ){ $this->extracted->pushPlural( $string->exportPlural(), $index ); } } /** * Get number of unique strings across all domains extracted (excluding additional metadata) * @return array { default: x, myDomain: y } */ public function getDomainCounts(){ return $this->extracted->getDomainCounts(); } /** * Pull extracted data into POT, filtering out any unwanted domains * @param string $domain * @return Loco_gettext_Data */ public function getTemplate( $domain ){ do_action('loco_extracted_template', $this, $domain ); $data = new Loco_gettext_Data( $this->extracted->filter($domain) ); return $data->templatize( $domain ); } /** * Get total number of strings extracted from all domains, excluding additional metadata * @return int */ public function getTotal(){ return $this->extracted->count(); } /** * Get list of files skipped, or null if none were skipped * @return Loco_fs_FileList|null */ public function getSkipped(){ return $this->skipped; } /** * Get size in bytes of largest file encountered, even if skipped. * This is the value required of the max_php_size plugin setting to extract all files * @return int */ public function getMaxPhpSize(){ return $this->maxbytes; } }