params['exclude'])) { $this->_exclude = explode(',', $this->params['exclude']); } if (isset($this->params['files']) && !is_array($this->params['files'])) { $this->_files = explode(',', $this->params['files']); } if (isset($this->params['paths'])) { $this->_paths = explode(',', $this->params['paths']); } else { $defaultPath = APP_PATH; $message = __d('cake_console', "What is the path you would like to extract?\n[Q]uit [D]one"); while (true) { $response = $this->in($message, null, $defaultPath); if (strtoupper($response) === 'Q') { $this->out(__d('cake_console', 'Extract Aborted')); $this->_stop(); } elseif (strtoupper($response) === 'D') { $this->out(); break; } elseif (is_dir($response)) { $this->_paths[] = $response; $defaultPath = 'D'; } else { $this->err(__d('cake_console', 'The directory path you supplied was not found. Please try again.')); } $this->out(); } } if (isset($this->params['output'])) { $this->_output = $this->params['output']; } else { $message = __d('cake_console', "What is the path you would like to output?\n[Q]uit", $this->_paths[0] . DS . 'locale'); while (true) { $response = $this->in($message, null, $this->_paths[0] . DS . 'locale'); if (strtoupper($response) === 'Q') { $this->out(__d('cake_console', 'Extract Aborted')); $this->_stop(); } elseif (is_dir($response)) { $this->_output = $response . DS; break; } else { $this->err(__d('cake_console', 'The directory path you supplied was not found. Please try again.')); } $this->out(); } } if (isset($this->params['merge'])) { $this->_merge = !(strtolower($this->params['merge']) === 'no'); } else { $this->out(); $response = $this->in(__d('cake_console', 'Would you like to merge all domains strings into the default.pot file?'), array('y', 'n'), 'n'); $this->_merge = strtolower($response) === 'y'; } if (empty($this->_files)) { $this->_searchFiles(); } $this->_extract(); } /** * Extract text * * @return void * @access protected */ protected function _extract() { $this->out(); $this->out(); $this->out(__d('cake_console', 'Extracting...')); $this->hr(); $this->out(__d('cake_console', 'Paths:')); foreach ($this->_paths as $path) { $this->out(' ' . $path); } $this->out(__d('cake_console', 'Output Directory: ') . $this->_output); $this->hr(); $this->_extractTokens(); $this->_buildFiles(); $this->_writeFiles(); $this->_paths = $this->_files = $this->_storage = array(); $this->_strings = $this->_tokens = array(); $this->out(); $this->out(__d('cake_console', 'Done.')); } /** * Get & configure the option parser * * @return void */ public function getOptionParser() { $parser = parent::getOptionParser(); return $parser->description(__d('cake_console', 'CakePHP Language String Extraction:')) ->addOption('app', array('help' => __d('cake_console', 'Directory where your application is located.'))) ->addOption('paths', array('help' => __d('cake_console', 'Comma separated list of paths.'))) ->addOption('merge', array( 'help' => __d('cake_console', 'Merge all domain strings into the default.po file.'), 'choices' => array('yes', 'no') )) ->addOption('output', array('help' => __d('cake_console', 'Full path to output directory.'))) ->addOption('files', array('help' => __d('cake_console', 'Comma separated list of files.'))) ->addOption('exclude', array( 'help' => __d('cake_console', 'Comma separated list of directories to exclude. Any path containing a path segment with the provided values will be skipped. E.g. test,vendors') )); } /** * Show help options * * @return void */ public function help() { $this->out(__d('cake_console', 'CakePHP Language String Extraction:')); $this->hr(); $this->out(__d('cake_console', 'The Extract script generates .pot file(s) with translations')); $this->out(__d('cake_console', 'By default the .pot file(s) will be place in the locale directory of -app')); $this->out(__d('cake_console', 'By default -app is ROOT/app')); $this->hr(); $this->out(__d('cake_console', 'Usage: cake i18n extract ...')); $this->out(); $this->out(__d('cake_console', 'Params:')); $this->out(__d('cake_console', ' -app [path...]: directory where your application is located')); $this->out(__d('cake_console', ' -root [path...]: path to install')); $this->out(__d('cake_console', ' -core [path...]: path to cake directory')); $this->out(__d('cake_console', ' -paths [comma separated list of paths]')); $this->out(__d('cake_console', ' -merge [yes|no]: Merge all domains strings into the default.pot file')); $this->out(__d('cake_console', ' -output [path...]: Full path to output directory')); $this->out(__d('cake_console', ' -files: [comma separated list of files]')); $this->out(); $this->out(__d('cake_console', 'Commands:')); $this->out(__d('cake_console', ' cake i18n extract help: Shows this help message.')); $this->out(); } /** * Extract tokens out of all files to be processed * * @return void * @access protected */ protected function _extractTokens() { foreach ($this->_files as $file) { $this->_file = $file; $this->out(__d('cake_console', 'Processing %s...', $file)); $code = file_get_contents($file); $allTokens = token_get_all($code); $this->_tokens = array(); foreach ($allTokens as $token) { if (!is_array($token) || ($token[0] != T_WHITESPACE && $token[0] != T_INLINE_HTML)) { $this->_tokens[] = $token; } } unset($allTokens); $this->_parse('__', array('singular')); $this->_parse('__n', array('singular', 'plural')); $this->_parse('__d', array('domain', 'singular')); $this->_parse('__c', array('singular')); $this->_parse('__dc', array('domain', 'singular')); $this->_parse('__dn', array('domain', 'singular', 'plural')); $this->_parse('__dcn', array('domain', 'singular', 'plural')); } } /** * Parse tokens * * @param string $functionName Function name that indicates translatable string (e.g: '__') * @param array $map Array containing what variables it will find (e.g: domain, singular, plural) * @return void * @access protected */ protected function _parse($functionName, $map) { $count = 0; $tokenCount = count($this->_tokens); while (($tokenCount - $count) > 1) { list($countToken, $firstParenthesis) = array($this->_tokens[$count], $this->_tokens[$count + 1]); if (!is_array($countToken)) { $count++; continue; } list($type, $string, $line) = $countToken; if (($type == T_STRING) && ($string == $functionName) && ($firstParenthesis == '(')) { $position = $count; $depth = 0; while ($depth == 0) { if ($this->_tokens[$position] == '(') { $depth++; } elseif ($this->_tokens[$position] == ')') { $depth--; } $position++; } $mapCount = count($map); $strings = $this->_getStrings($position, $mapCount); if ($mapCount == count($strings)) { extract(array_combine($map, $strings)); $domain = isset($domain) ? $domain : 'default'; $string = isset($plural) ? $singular . "\0" . $plural : $singular; $this->_strings[$domain][$string][$this->_file][] = $line; } else { $this->_markerError($this->_file, $line, $functionName, $count); } } $count++; } } /** * Build the translate template file contents out of obtained strings * * @return void * @access protected */ protected function _buildFiles() { foreach ($this->_strings as $domain => $strings) { foreach ($strings as $string => $files) { $occurrences = array(); foreach ($files as $file => $lines) { $occurrences[] = $file . ':' . implode(';', $lines); } $occurrences = implode("\n#: ", $occurrences); $header = '#: ' . str_replace($this->_paths, '', $occurrences) . "\n"; if (strpos($string, "\0") === false) { $sentence = "msgid \"{$string}\"\n"; $sentence .= "msgstr \"\"\n\n"; } else { list($singular, $plural) = explode("\0", $string); $sentence = "msgid \"{$singular}\"\n"; $sentence .= "msgid_plural \"{$plural}\"\n"; $sentence .= "msgstr[0] \"\"\n"; $sentence .= "msgstr[1] \"\"\n\n"; } $this->_store($domain, $header, $sentence); if ($domain != 'default' && $this->_merge) { $this->_store('default', $header, $sentence); } } } } /** * Prepare a file to be stored * * @return void * @access protected */ protected function _store($domain, $header, $sentence) { if (!isset($this->_storage[$domain])) { $this->_storage[$domain] = array(); } if (!isset($this->_storage[$domain][$sentence])) { $this->_storage[$domain][$sentence] = $header; } else { $this->_storage[$domain][$sentence] .= $header; } } /** * Write the files that need to be stored * * @return void * @access protected */ protected function _writeFiles() { $overwriteAll = false; foreach ($this->_storage as $domain => $sentences) { $output = $this->_writeHeader(); foreach ($sentences as $sentence => $header) { $output .= $header . $sentence; } $filename = $domain . '.pot'; $File = new File($this->_output . $filename); $response = ''; while ($overwriteAll === false && $File->exists() && strtoupper($response) !== 'Y') { $this->out(); $response = $this->in(__d('cake_console', 'Error: %s already exists in this location. Overwrite? [Y]es, [N]o, [A]ll', $filename), array('y', 'n', 'a'), 'y'); if (strtoupper($response) === 'N') { $response = ''; while ($response == '') { $response = $this->in(__d('cake_console', "What would you like to name this file?"), null, 'new_' . $filename); $File = new File($this->_output . $response); $filename = $response; } } elseif (strtoupper($response) === 'A') { $overwriteAll = true; } } $File->write($output); $File->close(); } } /** * Build the translation template header * * @return string Translation template header * @access protected */ protected function _writeHeader() { $output = "# LANGUAGE translation of CakePHP Application\n"; $output .= "# Copyright YEAR NAME \n"; $output .= "#\n"; $output .= "#, fuzzy\n"; $output .= "msgid \"\"\n"; $output .= "msgstr \"\"\n"; $output .= "\"Project-Id-Version: PROJECT VERSION\\n\"\n"; $output .= "\"POT-Creation-Date: " . date("Y-m-d H:iO") . "\\n\"\n"; $output .= "\"PO-Revision-Date: YYYY-mm-DD HH:MM+ZZZZ\\n\"\n"; $output .= "\"Last-Translator: NAME \\n\"\n"; $output .= "\"Language-Team: LANGUAGE \\n\"\n"; $output .= "\"MIME-Version: 1.0\\n\"\n"; $output .= "\"Content-Type: text/plain; charset=utf-8\\n\"\n"; $output .= "\"Content-Transfer-Encoding: 8bit\\n\"\n"; $output .= "\"Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\\n\"\n\n"; return $output; } /** * Get the strings from the position forward * * @param int $position Actual position on tokens array * @param int $target Number of strings to extract * @return array Strings extracted * @access protected */ protected function _getStrings(&$position, $target) { $strings = array(); while (count($strings) < $target && ($this->_tokens[$position] == ',' || $this->_tokens[$position][0] == T_CONSTANT_ENCAPSED_STRING)) { if ($this->_tokens[$position][0] == T_CONSTANT_ENCAPSED_STRING && $this->_tokens[$position+1] == '.') { $string = ''; while ($this->_tokens[$position][0] == T_CONSTANT_ENCAPSED_STRING || $this->_tokens[$position] == '.') { if ($this->_tokens[$position][0] == T_CONSTANT_ENCAPSED_STRING) { $string .= $this->_formatString($this->_tokens[$position][1]); } $position++; } $strings[] = $string; } else if ($this->_tokens[$position][0] == T_CONSTANT_ENCAPSED_STRING) { $strings[] = $this->_formatString($this->_tokens[$position][1]); } $position++; } return $strings; } /** * Format a string to be added as a translatable string * * @param string $string String to format * @return string Formatted string * @access protected */ protected function _formatString($string) { $quote = substr($string, 0, 1); $string = substr($string, 1, -1); if ($quote == '"') { $string = stripcslashes($string); } else { $string = strtr($string, array("\\'" => "'", "\\\\" => "\\")); } $string = str_replace("\r\n", "\n", $string); return addcslashes($string, "\0..\37\\\""); } /** * Indicate an invalid marker on a processed file * * @param string $file File where invalid marker resides * @param integer $line Line number * @param string $marker Marker found * @param integer $count Count * @return void * @access protected */ protected function _markerError($file, $line, $marker, $count) { $this->out(__d('cake_console', "Invalid marker content in %s:%s\n* %s(", $file, $line, $marker), true); $count += 2; $tokenCount = count($this->_tokens); $parenthesis = 1; while ((($tokenCount - $count) > 0) && $parenthesis) { if (is_array($this->_tokens[$count])) { $this->out($this->_tokens[$count][1], false); } else { $this->out($this->_tokens[$count], false); if ($this->_tokens[$count] == '(') { $parenthesis++; } if ($this->_tokens[$count] == ')') { $parenthesis--; } } $count++; } $this->out("\n", true); } /** * Search files that may contain translatable strings * * @return void * @access protected */ protected function _searchFiles() { $pattern = false; if (!empty($this->_exclude)) { $pattern = '/[\/\\\\]' . implode('|', $this->_exclude) . '[\/\\\\]/'; } foreach ($this->_paths as $path) { $Folder = new Folder($path); $files = $Folder->findRecursive('.*\.(php|ctp|thtml|inc|tpl)', true); if (!empty($pattern)) { foreach ($files as $i => $file) { if (preg_match($pattern, $file)) { unset($files[$i]); } } $files = array_values($files); } $this->_files = array_merge($this->_files, $files); } } }