updated core to 7.67
This commit is contained in:
		| @@ -1,3 +1,29 @@ | |||||||
|  | Drupal 7.xx, xxxx-xx-xx (development version) | ||||||
|  | ----------------------- | ||||||
|  |  | ||||||
|  | Drupal 7.67, 2019-05-08 | ||||||
|  | ----------------------- | ||||||
|  | - Fixed security issues: | ||||||
|  |    - SA-CORE-2019-007 | ||||||
|  |  | ||||||
|  | Drupal 7.66, 2019-04-17 | ||||||
|  | ----------------------- | ||||||
|  | - Fixed security issues: | ||||||
|  |    - SA-CORE-2019-006 | ||||||
|  |  | ||||||
|  | Drupal 7.65, 2019-03-20 | ||||||
|  | ----------------------- | ||||||
|  | - Fixed security issues: | ||||||
|  |    - SA-CORE-2019-004 | ||||||
|  |  | ||||||
|  | Drupal 7.64, 2019-02-06 | ||||||
|  | ----------------------- | ||||||
|  | - [regression] Unset the 'host' header in drupal_http_request() during redirect | ||||||
|  | - Fixed: 7.x does not have Phar protection and Phar tests are failing on Drupal 7 | ||||||
|  | - Fixed: Notice: Undefined index: display_field in file_field_widget_value() (line 582 of /module/file/file.field.inc) | ||||||
|  | - Performance improvement: Registry rebuild should not parse the same file twice in the same request | ||||||
|  | - Fixed _registry_update() to clear caches after transaction is committed | ||||||
|  |  | ||||||
| Drupal 7.63, 2019-01-16 | Drupal 7.63, 2019-01-16 | ||||||
| ----------------------- | ----------------------- | ||||||
| - Fixed a fatal error for some Drush users introduced by SA-CORE-2019-002. | - Fixed a fatal error for some Drush users introduced by SA-CORE-2019-002. | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
| /** | /** | ||||||
|  * The current system version. |  * The current system version. | ||||||
|  */ |  */ | ||||||
| define('VERSION', '7.63'); | define('VERSION', '7.67'); | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Core API compatibility. |  * Core API compatibility. | ||||||
|   | |||||||
| @@ -1094,6 +1094,11 @@ function drupal_http_request($url, array $options = array()) { | |||||||
|       elseif ($options['max_redirects']) { |       elseif ($options['max_redirects']) { | ||||||
|         // Redirect to the new location. |         // Redirect to the new location. | ||||||
|         $options['max_redirects']--; |         $options['max_redirects']--; | ||||||
|  |  | ||||||
|  |         // We need to unset the 'Host' header | ||||||
|  |         // as we are redirecting to a new location. | ||||||
|  |         unset($options['headers']['Host']); | ||||||
|  |  | ||||||
|         $result = drupal_http_request($location, $options); |         $result = drupal_http_request($location, $options); | ||||||
|         $result->redirect_code = $code; |         $result->redirect_code = $code; | ||||||
|       } |       } | ||||||
|   | |||||||
| @@ -993,8 +993,15 @@ function file_build_uri($path) { | |||||||
|  * @return |  * @return | ||||||
|  *   The destination filepath, or FALSE if the file already exists |  *   The destination filepath, or FALSE if the file already exists | ||||||
|  *   and FILE_EXISTS_ERROR is specified. |  *   and FILE_EXISTS_ERROR is specified. | ||||||
|  |  * | ||||||
|  |  * @throws RuntimeException | ||||||
|  |  *   Thrown if the filename contains invalid UTF-8. | ||||||
|  */ |  */ | ||||||
| function file_destination($destination, $replace) { | function file_destination($destination, $replace) { | ||||||
|  |   $basename = drupal_basename($destination); | ||||||
|  |   if (!drupal_validate_utf8($basename)) { | ||||||
|  |     throw new RuntimeException(sprintf("Invalid filename '%s'", $basename)); | ||||||
|  |   } | ||||||
|   if (file_exists($destination)) { |   if (file_exists($destination)) { | ||||||
|     switch ($replace) { |     switch ($replace) { | ||||||
|       case FILE_EXISTS_REPLACE: |       case FILE_EXISTS_REPLACE: | ||||||
| @@ -1002,7 +1009,6 @@ function file_destination($destination, $replace) { | |||||||
|         break; |         break; | ||||||
|  |  | ||||||
|       case FILE_EXISTS_RENAME: |       case FILE_EXISTS_RENAME: | ||||||
|         $basename = drupal_basename($destination); |  | ||||||
|         $directory = drupal_dirname($destination); |         $directory = drupal_dirname($destination); | ||||||
|         $destination = file_create_filename($basename, $directory); |         $destination = file_create_filename($basename, $directory); | ||||||
|         break; |         break; | ||||||
| @@ -1218,11 +1224,20 @@ function file_unmunge_filename($filename) { | |||||||
|  * @return |  * @return | ||||||
|  *   File path consisting of $directory and a unique filename based off |  *   File path consisting of $directory and a unique filename based off | ||||||
|  *   of $basename. |  *   of $basename. | ||||||
|  |  * | ||||||
|  |  * @throws RuntimeException | ||||||
|  |  *   Thrown if the $basename is not valid UTF-8 or another error occurs | ||||||
|  |  *   stripping control characters. | ||||||
|  */ |  */ | ||||||
| function file_create_filename($basename, $directory) { | function file_create_filename($basename, $directory) { | ||||||
|  |   $original = $basename; | ||||||
|   // Strip control characters (ASCII value < 32). Though these are allowed in |   // Strip control characters (ASCII value < 32). Though these are allowed in | ||||||
|   // some filesystems, not many applications handle them well. |   // some filesystems, not many applications handle them well. | ||||||
|   $basename = preg_replace('/[\x00-\x1F]/u', '_', $basename); |   $basename = preg_replace('/[\x00-\x1F]/u', '_', $basename); | ||||||
|  |   if (preg_last_error() !== PREG_NO_ERROR) { | ||||||
|  |     throw new RuntimeException(sprintf("Invalid filename '%s'", $original)); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   if (substr(PHP_OS, 0, 3) == 'WIN') { |   if (substr(PHP_OS, 0, 3) == 'WIN') { | ||||||
|     // These characters are not allowed in Windows filenames |     // These characters are not allowed in Windows filenames | ||||||
|     $basename = str_replace(array(':', '*', '?', '"', '<', '>', '|'), '_', $basename); |     $basename = str_replace(array(':', '*', '?', '"', '<', '>', '|'), '_', $basename); | ||||||
| @@ -1563,7 +1578,13 @@ function file_save_upload($form_field_name, $validators = array(), $destination | |||||||
|   if (substr($destination, -1) != '/') { |   if (substr($destination, -1) != '/') { | ||||||
|     $destination .= '/'; |     $destination .= '/'; | ||||||
|   } |   } | ||||||
|  |   try { | ||||||
|     $file->destination = file_destination($destination . $file->filename, $replace); |     $file->destination = file_destination($destination . $file->filename, $replace); | ||||||
|  |   } | ||||||
|  |   catch (RuntimeException $e) { | ||||||
|  |     drupal_set_message(t('The file %source could not be uploaded because the name is invalid.', array('%source' => $form_field_name)), 'error'); | ||||||
|  |     return FALSE; | ||||||
|  |   } | ||||||
|   // If file_destination() returns FALSE then $replace == FILE_EXISTS_ERROR and |   // If file_destination() returns FALSE then $replace == FILE_EXISTS_ERROR and | ||||||
|   // there's an existing file so we need to bail. |   // there's an existing file so we need to bail. | ||||||
|   if ($file->destination === FALSE) { |   if ($file->destination === FALSE) { | ||||||
| @@ -2130,9 +2151,33 @@ function file_download_access($uri) { | |||||||
|  *   'filename', and 'name' members corresponding to the matching files. |  *   'filename', and 'name' members corresponding to the matching files. | ||||||
|  */ |  */ | ||||||
| function file_scan_directory($dir, $mask, $options = array(), $depth = 0) { | function file_scan_directory($dir, $mask, $options = array(), $depth = 0) { | ||||||
|  |   // Default nomask option. | ||||||
|  |   $nomask = '/(\.\.?|CVS)$/'; | ||||||
|  |  | ||||||
|  |   // Overrides the $nomask variable accordingly if $options['nomask'] is set. | ||||||
|  |   // | ||||||
|  |   // Allow directories specified in settings.php to be ignored. You can use this | ||||||
|  |   // to not check for files in common special-purpose directories. For example, | ||||||
|  |   // node_modules and bower_components. Ignoring irrelevant directories is a | ||||||
|  |   // performance boost. | ||||||
|  |   if (!isset($options['nomask'])) { | ||||||
|  |     $ignore_directories = variable_get( | ||||||
|  |       'file_scan_ignore_directories', | ||||||
|  |       array() | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     foreach ($ignore_directories as $index => $ignore_directory) { | ||||||
|  |       $ignore_directories[$index] = preg_quote($ignore_directory, '/'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (!empty($ignore_directories)) { | ||||||
|  |       $nomask = '/^(\.\.?)|CVS|' . implode('|', $ignore_directories) . '$/'; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // Merge in defaults. |   // Merge in defaults. | ||||||
|   $options += array( |   $options += array( | ||||||
|     'nomask' => '/(\.\.?|CVS)$/', |     'nomask' => $nomask, | ||||||
|     'callback' => 0, |     'callback' => 0, | ||||||
|     'recurse' => TRUE, |     'recurse' => TRUE, | ||||||
|     'key' => 'uri', |     'key' => 'uri', | ||||||
|   | |||||||
| @@ -18,7 +18,21 @@ function file_register_phar_wrapper() { | |||||||
|   include_once $directory . '/Helper.php'; |   include_once $directory . '/Helper.php'; | ||||||
|   include_once $directory . '/Manager.php'; |   include_once $directory . '/Manager.php'; | ||||||
|   include_once $directory . '/PharStreamWrapper.php'; |   include_once $directory . '/PharStreamWrapper.php'; | ||||||
|  |   include_once $directory . '/Collectable.php'; | ||||||
|  |   include_once $directory . '/Interceptor/ConjunctionInterceptor.php'; | ||||||
|  |   include_once $directory . '/Interceptor/PharMetaDataInterceptor.php'; | ||||||
|  |   include_once $directory . '/Phar/Container.php'; | ||||||
|  |   include_once $directory . '/Phar/DeserializationException.php'; | ||||||
|  |   include_once $directory . '/Phar/Manifest.php'; | ||||||
|  |   include_once $directory . '/Phar/Reader.php'; | ||||||
|  |   include_once $directory . '/Phar/ReaderException.php'; | ||||||
|  |   include_once $directory . '/Phar/Stub.php'; | ||||||
|  |   include_once $directory . '/Resolvable.php'; | ||||||
|  |   include_once $directory . '/Resolver/PharInvocation.php'; | ||||||
|  |   include_once $directory . '/Resolver/PharInvocationCollection.php'; | ||||||
|  |   include_once $directory . '/Resolver/PharInvocationResolver.php'; | ||||||
|   include_once DRUPAL_ROOT . '/misc/typo3/drupal-security/PharExtensionInterceptor.php'; |   include_once DRUPAL_ROOT . '/misc/typo3/drupal-security/PharExtensionInterceptor.php'; | ||||||
|  |   include_once DRUPAL_ROOT . '/misc/brumann/polyfill-unserialize/src/Unserialize.php'; | ||||||
|  |  | ||||||
|   // Set up a stream wrapper to handle insecurities due to PHP's built-in |   // Set up a stream wrapper to handle insecurities due to PHP's built-in | ||||||
|   // phar stream wrapper. |   // phar stream wrapper. | ||||||
|   | |||||||
| @@ -19,7 +19,6 @@ | |||||||
|  * Does the work for registry_update(). |  * Does the work for registry_update(). | ||||||
|  */ |  */ | ||||||
| function _registry_update() { | function _registry_update() { | ||||||
|  |  | ||||||
|   // The registry serves as a central autoloader for all classes, including |   // The registry serves as a central autoloader for all classes, including | ||||||
|   // the database query builders. However, the registry rebuild process |   // the database query builders. However, the registry rebuild process | ||||||
|   // requires write ability to the database, which means having access to the |   // requires write ability to the database, which means having access to the | ||||||
| @@ -33,6 +32,11 @@ function _registry_update() { | |||||||
|   require_once DRUPAL_ROOT . '/includes/database/select.inc'; |   require_once DRUPAL_ROOT . '/includes/database/select.inc'; | ||||||
|   require_once DRUPAL_ROOT . '/includes/database/' . $driver . '/query.inc'; |   require_once DRUPAL_ROOT . '/includes/database/' . $driver . '/query.inc'; | ||||||
|  |  | ||||||
|  |   // During the first registry rebuild in a request, we check all the files. | ||||||
|  |   // During subsequent rebuilds, we only add new files. It makes the rebuilding | ||||||
|  |   // process faster during installation of modules. | ||||||
|  |   static $check_existing_files = TRUE; | ||||||
|  |  | ||||||
|   // Get current list of modules and their files. |   // Get current list of modules and their files. | ||||||
|   $modules = db_query("SELECT * FROM {system} WHERE type = 'module'")->fetchAll(); |   $modules = db_query("SELECT * FROM {system} WHERE type = 'module'")->fetchAll(); | ||||||
|   // Get the list of files we are going to parse. |   // Get the list of files we are going to parse. | ||||||
| @@ -55,6 +59,9 @@ function _registry_update() { | |||||||
|     $files["$filename"] = array('module' => '', 'weight' => 0); |     $files["$filename"] = array('module' => '', 'weight' => 0); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   // Initialize an empty array for the unchanged files. | ||||||
|  |   $unchanged_files = array(); | ||||||
|  |  | ||||||
|   $transaction = db_transaction(); |   $transaction = db_transaction(); | ||||||
|   try { |   try { | ||||||
|     // Allow modules to manually modify the list of files before the registry |     // Allow modules to manually modify the list of files before the registry | ||||||
| @@ -63,11 +70,20 @@ function _registry_update() { | |||||||
|     // list can then be added to the list of files that the registry will parse, |     // list can then be added to the list of files that the registry will parse, | ||||||
|     // or modify attributes of a file. |     // or modify attributes of a file. | ||||||
|     drupal_alter('registry_files', $files, $modules); |     drupal_alter('registry_files', $files, $modules); | ||||||
|  |  | ||||||
|     foreach (registry_get_parsed_files() as $filename => $file) { |     foreach (registry_get_parsed_files() as $filename => $file) { | ||||||
|       // Add the hash for those files we have already parsed. |       // Add the hash for those files we have already parsed. | ||||||
|       if (isset($files[$filename])) { |       if (isset($files[$filename])) { | ||||||
|  |         if ($check_existing_files === TRUE) { | ||||||
|           $files[$filename]['hash'] = $file['hash']; |           $files[$filename]['hash'] = $file['hash']; | ||||||
|         } |         } | ||||||
|  |         else { | ||||||
|  |           // Ignore that file for this request, it has been parsed previously | ||||||
|  |           // and it is unlikely it has changed. | ||||||
|  |           unset($files[$filename]); | ||||||
|  |           $unchanged_files[$filename] = $file; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|       else { |       else { | ||||||
|         // Flush the registry of resources in files that are no longer on disc |         // Flush the registry of resources in files that are no longer on disc | ||||||
|         // or are in files that no installed modules require to be parsed. |         // or are in files that no installed modules require to be parsed. | ||||||
| @@ -79,8 +95,12 @@ function _registry_update() { | |||||||
|           ->execute(); |           ->execute(); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     $parsed_files = _registry_parse_files($files); |     $parsed_files = _registry_parse_files($files); | ||||||
|  |  | ||||||
|  |     // Add unchanged files to the files. | ||||||
|  |     $files += $unchanged_files; | ||||||
|  |  | ||||||
|     $unchanged_resources = array(); |     $unchanged_resources = array(); | ||||||
|     $lookup_cache = array(); |     $lookup_cache = array(); | ||||||
|     if ($cache = cache_get('lookup_cache', 'cache_bootstrap')) { |     if ($cache = cache_get('lookup_cache', 'cache_bootstrap')) { | ||||||
| @@ -89,12 +109,10 @@ function _registry_update() { | |||||||
|     foreach ($lookup_cache as $key => $file) { |     foreach ($lookup_cache as $key => $file) { | ||||||
|       // If the file for this cached resource is carried over unchanged from |       // If the file for this cached resource is carried over unchanged from | ||||||
|       // the last registry build, then we can safely re-cache it. |       // the last registry build, then we can safely re-cache it. | ||||||
|       if ($file && in_array($file, array_keys($files)) && !in_array($file, $parsed_files)) { |       if ($file && isset($files[$file]) && !in_array($file, $parsed_files, TRUE)) { | ||||||
|         $unchanged_resources[$key] = $file; |         $unchanged_resources[$key] = $file; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     module_implements('', FALSE, TRUE); |  | ||||||
|     _registry_check_code(REGISTRY_RESET_LOOKUP_CACHE); |  | ||||||
|   } |   } | ||||||
|   catch (Exception $e) { |   catch (Exception $e) { | ||||||
|     $transaction->rollback(); |     $transaction->rollback(); | ||||||
| @@ -102,6 +120,13 @@ function _registry_update() { | |||||||
|     throw $e; |     throw $e; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   module_implements('', FALSE, TRUE); | ||||||
|  |   _registry_check_code(REGISTRY_RESET_LOOKUP_CACHE); | ||||||
|  |  | ||||||
|  |   // During the next run in this request, don't bother re-checking existing | ||||||
|  |   // files. | ||||||
|  |   $check_existing_files = FALSE; | ||||||
|  |  | ||||||
|   // We have some unchanged resources, warm up the cache - no need to pay |   // We have some unchanged resources, warm up the cache - no need to pay | ||||||
|   // for looking them up again. |   // for looking them up again. | ||||||
|   if (count($unchanged_resources) > 0) { |   if (count($unchanged_resources) > 0) { | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								misc/brumann/polyfill-unserialize/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								misc/brumann/polyfill-unserialize/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | /vendor/ | ||||||
|  | /phpunit.xml | ||||||
|  | /.composer.lock | ||||||
|  |  | ||||||
							
								
								
									
										20
									
								
								misc/brumann/polyfill-unserialize/.travis.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								misc/brumann/polyfill-unserialize/.travis.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | language: php | ||||||
|  |  | ||||||
|  | sudo: false | ||||||
|  |  | ||||||
|  | php: | ||||||
|  |   - '5.3' | ||||||
|  |   - '5.4' | ||||||
|  |   - '5.5' | ||||||
|  |   - '5.6' | ||||||
|  |   - '7.0' | ||||||
|  |   - '7.1' | ||||||
|  |  | ||||||
|  | before_install: | ||||||
|  |   - phpenv config-rm xdebug.ini | ||||||
|  |   - composer self-update | ||||||
|  |  | ||||||
|  | install: | ||||||
|  |   - composer install | ||||||
|  |  | ||||||
|  | script: phpunit | ||||||
							
								
								
									
										21
									
								
								misc/brumann/polyfill-unserialize/LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								misc/brumann/polyfill-unserialize/LICENSE
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | MIT License | ||||||
|  |  | ||||||
|  | Copyright (c) 2016 Denis Brumann | ||||||
|  |  | ||||||
|  | Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
|  | of this software and associated documentation files (the "Software"), to deal | ||||||
|  | in the Software without restriction, including without limitation the rights | ||||||
|  | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||||
|  | copies of the Software, and to permit persons to whom the Software is | ||||||
|  | furnished to do so, subject to the following conditions: | ||||||
|  |  | ||||||
|  | The above copyright notice and this permission notice shall be included in all | ||||||
|  | copies or substantial portions of the Software. | ||||||
|  |  | ||||||
|  | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||||
|  | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||||
|  | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||||
|  | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||||
|  | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||||
|  | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||||
|  | SOFTWARE. | ||||||
							
								
								
									
										61
									
								
								misc/brumann/polyfill-unserialize/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								misc/brumann/polyfill-unserialize/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | |||||||
|  | Polyfill unserialize [](https://travis-ci.org/dbrumann/polyfill-unserialize) | ||||||
|  | === | ||||||
|  |  | ||||||
|  | Backports unserialize options introduced in PHP 7.0 to older PHP versions. | ||||||
|  | This was originally designed as a Proof of Concept for Symfony Issue [#21090](https://github.com/symfony/symfony/pull/21090). | ||||||
|  |  | ||||||
|  | You can use this package in projects that rely on PHP versions older than PHP 7.0. | ||||||
|  | In case you are using PHP 7.0+ the original `unserialize()` will be used instead. | ||||||
|  |  | ||||||
|  | From the [documentation](https://secure.php.net/manual/en/function.unserialize.php): | ||||||
|  |  | ||||||
|  | > Warning: Do not pass untrusted user input to unserialize(). Unserialization can | ||||||
|  | > result in code being loaded and executed due to object instantiation | ||||||
|  | > and autoloading, and a malicious user may be able to exploit this. | ||||||
|  |  | ||||||
|  | This warning holds true even when `allowed_classes` is used. | ||||||
|  |  | ||||||
|  | Requirements | ||||||
|  | ------------ | ||||||
|  |  | ||||||
|  |  - PHP 5.3+ | ||||||
|  |  | ||||||
|  | Installation | ||||||
|  | ------------ | ||||||
|  |  | ||||||
|  | You can install this package via composer: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | composer require brumann/polyfill-unserialize "^1.0" | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Known Issues | ||||||
|  | ------------ | ||||||
|  |  | ||||||
|  | There is a mismatch in behavior when `allowed_classes` in `$options` is not | ||||||
|  | of the correct type (array or boolean). PHP 7.1 will issue a warning, whereas | ||||||
|  | PHP 7.0 will not. I opted to copy the behavior of the former. | ||||||
|  |  | ||||||
|  | Tests | ||||||
|  | ----- | ||||||
|  |  | ||||||
|  | You can run the test suite using PHPUnit. It is intentionally not bundled as | ||||||
|  | dev dependency to make sure this package has the lowest restrictions on the | ||||||
|  | implementing system as possible. | ||||||
|  |  | ||||||
|  | Please read the [PHPUnit Manual](https://phpunit.de/manual/current/en/installation.html) | ||||||
|  | for information how to install it on your system. | ||||||
|  |  | ||||||
|  | You can run the test suite as follows: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | phpunit -c phpunit.xml.dist tests/ | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Contributing | ||||||
|  | ------------ | ||||||
|  |  | ||||||
|  | This package is considered feature complete. As such I will likely not update it | ||||||
|  | unless there are security issues. | ||||||
|  |  | ||||||
|  | Should you find any bugs or have questions, feel free to submit an Issue or a Pull Request. | ||||||
							
								
								
									
										26
									
								
								misc/brumann/polyfill-unserialize/composer.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								misc/brumann/polyfill-unserialize/composer.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | { | ||||||
|  |     "name": "brumann/polyfill-unserialize", | ||||||
|  |     "description": "Backports unserialize options introduced in PHP 7.0 to older PHP versions.", | ||||||
|  |     "type": "library", | ||||||
|  |     "license": "MIT", | ||||||
|  |     "authors": [ | ||||||
|  |         { | ||||||
|  |             "name": "Denis Brumann", | ||||||
|  |             "email": "denis.brumann@sensiolabs.de" | ||||||
|  |         } | ||||||
|  |     ], | ||||||
|  |     "autoload": { | ||||||
|  |         "psr-4": { | ||||||
|  |             "Brumann\\Polyfill\\": "src/" | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  |     "autoload-dev": { | ||||||
|  |         "psr-4": { | ||||||
|  |             "Tests\\Brumann\\Polyfill\\": "tests/" | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  |     "minimum-stability": "stable", | ||||||
|  |     "require": { | ||||||
|  |         "php": "^5.3|^7.0" | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										25
									
								
								misc/brumann/polyfill-unserialize/phpunit.xml.dist
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								misc/brumann/polyfill-unserialize/phpunit.xml.dist
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  |  | ||||||
|  | <phpunit | ||||||
|  |     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||||
|  |     xsi:noNamespaceSchemaLocation="http://schema.phpunit.de/4.1/phpunit.xsd" | ||||||
|  |     backupGlobals="false" | ||||||
|  |     colors="true" | ||||||
|  |     bootstrap="vendor/autoload.php" | ||||||
|  | > | ||||||
|  |     <php> | ||||||
|  |         <ini name="error_reporting" value="-1" /> | ||||||
|  |     </php> | ||||||
|  |  | ||||||
|  |     <testsuites> | ||||||
|  |         <testsuite name="Brumann\Polyfill Test Suite"> | ||||||
|  |             <directory>./tests/</directory> | ||||||
|  |         </testsuite> | ||||||
|  |     </testsuites> | ||||||
|  |  | ||||||
|  |     <filter> | ||||||
|  |         <whitelist> | ||||||
|  |             <directory>./src/</directory> | ||||||
|  |         </whitelist> | ||||||
|  |     </filter> | ||||||
|  | </phpunit> | ||||||
							
								
								
									
										58
									
								
								misc/brumann/polyfill-unserialize/src/Unserialize.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								misc/brumann/polyfill-unserialize/src/Unserialize.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | namespace Brumann\Polyfill; | ||||||
|  |  | ||||||
|  | final class Unserialize | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * @see https://secure.php.net/manual/en/function.unserialize.php | ||||||
|  |      * | ||||||
|  |      * @param string $serialized Serialized data | ||||||
|  |      * @param array $options Associative array containing options | ||||||
|  |      * | ||||||
|  |      * @return mixed | ||||||
|  |      */ | ||||||
|  |     public static function unserialize($serialized, array $options = array()) | ||||||
|  |     { | ||||||
|  |         if (PHP_VERSION_ID >= 70000) { | ||||||
|  |             return \unserialize($serialized, $options); | ||||||
|  |         } | ||||||
|  |         if (!array_key_exists('allowed_classes', $options)) { | ||||||
|  |             $options['allowed_classes'] = true; | ||||||
|  |         } | ||||||
|  |         $allowedClasses = $options['allowed_classes']; | ||||||
|  |         if (true === $allowedClasses) { | ||||||
|  |             return \unserialize($serialized); | ||||||
|  |         } | ||||||
|  |         if (false === $allowedClasses) { | ||||||
|  |             $allowedClasses = array(); | ||||||
|  |         } | ||||||
|  |         if (!is_array($allowedClasses)) { | ||||||
|  |             trigger_error( | ||||||
|  |                 'unserialize(): allowed_classes option should be array or boolean', | ||||||
|  |                 E_USER_WARNING | ||||||
|  |             ); | ||||||
|  |             $allowedClasses = array(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $sanitizedSerialized = preg_replace_callback( | ||||||
|  |             '/(^|;)O:\d+:"([^"]*)":(\d+):{/', | ||||||
|  |             function ($match) use ($allowedClasses) { | ||||||
|  |                 list($completeMatch, $leftBorder, $className, $objectSize) = $match; | ||||||
|  |                 if (in_array($className, $allowedClasses)) { | ||||||
|  |                     return $completeMatch; | ||||||
|  |                 } else { | ||||||
|  |                     return sprintf( | ||||||
|  |                         '%sO:22:"__PHP_Incomplete_Class":%d:{s:27:"__PHP_Incomplete_Class_Name";%s', | ||||||
|  |                         $leftBorder, | ||||||
|  |                         $objectSize + 1, // size of object + 1 for added string | ||||||
|  |                         \serialize($className) | ||||||
|  |                     ); | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             $serialized | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         return \unserialize($sanitizedSerialized); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										112
									
								
								misc/jquery-extend-3.4.0.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								misc/jquery-extend-3.4.0.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,112 @@ | |||||||
|  | /** | ||||||
|  |  * For jQuery versions less than 3.4.0, this replaces the jQuery.extend | ||||||
|  |  * function with the one from jQuery 3.4.0, slightly modified (documented | ||||||
|  |  * below) to be compatible with older jQuery versions and browsers. | ||||||
|  |  * | ||||||
|  |  * This provides the Object.prototype pollution vulnerability fix to Drupal | ||||||
|  |  * installations running older jQuery versions, including the versions shipped | ||||||
|  |  * with Drupal core and https://www.drupal.org/project/jquery_update. | ||||||
|  |  * | ||||||
|  |  * @see https://github.com/jquery/jquery/pull/4333 | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | (function (jQuery) { | ||||||
|  |  | ||||||
|  | // Do not override jQuery.extend() if the jQuery version is already >=3.4.0. | ||||||
|  | var versionParts = jQuery.fn.jquery.split('.'); | ||||||
|  | var majorVersion = parseInt(versionParts[0]); | ||||||
|  | var minorVersion = parseInt(versionParts[1]); | ||||||
|  | var patchVersion = parseInt(versionParts[2]); | ||||||
|  | var isPreReleaseVersion = (patchVersion.toString() !== versionParts[2]); | ||||||
|  | if ( | ||||||
|  |   (majorVersion > 3) || | ||||||
|  |   (majorVersion === 3 && minorVersion > 4) || | ||||||
|  |   (majorVersion === 3 && minorVersion === 4 && patchVersion > 0) || | ||||||
|  |   (majorVersion === 3 && minorVersion === 4 && patchVersion === 0 && !isPreReleaseVersion) | ||||||
|  | ) { | ||||||
|  |   return; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * This is almost verbatim copied from jQuery 3.4.0. | ||||||
|  |  * | ||||||
|  |  * Only two minor changes have been made: | ||||||
|  |  * - The call to isFunction() is changed to jQuery.isFunction(). | ||||||
|  |  * - The two calls to Array.isArray() is changed to jQuery.isArray(). | ||||||
|  |  * | ||||||
|  |  * The above two changes ensure compatibility with all older jQuery versions | ||||||
|  |  * (1.4.4 - 3.3.1) and older browser versions (e.g., IE8). | ||||||
|  |  */ | ||||||
|  | jQuery.extend = jQuery.fn.extend = function() { | ||||||
|  |   var options, name, src, copy, copyIsArray, clone, | ||||||
|  |     target = arguments[ 0 ] || {}, | ||||||
|  |     i = 1, | ||||||
|  |     length = arguments.length, | ||||||
|  |     deep = false; | ||||||
|  |  | ||||||
|  |   // Handle a deep copy situation | ||||||
|  |   if ( typeof target === "boolean" ) { | ||||||
|  |     deep = target; | ||||||
|  |  | ||||||
|  |     // Skip the boolean and the target | ||||||
|  |     target = arguments[ i ] || {}; | ||||||
|  |     i++; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Handle case when target is a string or something (possible in deep copy) | ||||||
|  |   if ( typeof target !== "object" && !jQuery.isFunction( target ) ) { | ||||||
|  |     target = {}; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Extend jQuery itself if only one argument is passed | ||||||
|  |   if ( i === length ) { | ||||||
|  |     target = this; | ||||||
|  |     i--; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   for ( ; i < length; i++ ) { | ||||||
|  |  | ||||||
|  |     // Only deal with non-null/undefined values | ||||||
|  |     if ( ( options = arguments[ i ] ) != null ) { | ||||||
|  |  | ||||||
|  |       // Extend the base object | ||||||
|  |       for ( name in options ) { | ||||||
|  |         copy = options[ name ]; | ||||||
|  |  | ||||||
|  |         // Prevent Object.prototype pollution | ||||||
|  |         // Prevent never-ending loop | ||||||
|  |         if ( name === "__proto__" || target === copy ) { | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Recurse if we're merging plain objects or arrays | ||||||
|  |         if ( deep && copy && ( jQuery.isPlainObject( copy ) || | ||||||
|  |           ( copyIsArray = jQuery.isArray( copy ) ) ) ) { | ||||||
|  |           src = target[ name ]; | ||||||
|  |  | ||||||
|  |           // Ensure proper type for the source value | ||||||
|  |           if ( copyIsArray && !jQuery.isArray( src ) ) { | ||||||
|  |             clone = []; | ||||||
|  |           } else if ( !copyIsArray && !jQuery.isPlainObject( src ) ) { | ||||||
|  |             clone = {}; | ||||||
|  |           } else { | ||||||
|  |             clone = src; | ||||||
|  |           } | ||||||
|  |           copyIsArray = false; | ||||||
|  |  | ||||||
|  |           // Never move original objects, clone them | ||||||
|  |           target[ name ] = jQuery.extend( deep, clone, copy ); | ||||||
|  |  | ||||||
|  |           // Don't bring in undefined values | ||||||
|  |         } else if ( copy !== undefined ) { | ||||||
|  |           target[ name ] = copy; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Return the modified object | ||||||
|  |   return target; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | })(jQuery); | ||||||
							
								
								
									
										3
									
								
								misc/typo3/phar-stream-wrapper/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								misc/typo3/phar-stream-wrapper/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | .idea | ||||||
|  | vendor/ | ||||||
|  | composer.lock | ||||||
| @@ -63,7 +63,7 @@ adjusted to according requirements. | |||||||
|  |  | ||||||
| ``` | ``` | ||||||
| $behavior = new \TYPO3\PharStreamWrapper\Behavior(); | $behavior = new \TYPO3\PharStreamWrapper\Behavior(); | ||||||
| Manager::initialize( | \TYPO3\PharStreamWrapper\Manager::initialize( | ||||||
|     $behavior->withAssertion(new PharExtensionInterceptor()) |     $behavior->withAssertion(new PharExtensionInterceptor()) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| @@ -90,7 +90,7 @@ if (in_array('phar', stream_get_wrappers())) { | |||||||
|   + `COMMAND_UNLINK` |   + `COMMAND_UNLINK` | ||||||
|   + `COMMAND_URL_STAT` |   + `COMMAND_URL_STAT` | ||||||
|  |  | ||||||
| ## Interceptor | ## Interceptors | ||||||
|  |  | ||||||
| The following interceptor is shipped with the package and ready to use in order | The following interceptor is shipped with the package and ready to use in order | ||||||
| to block any Phar invocation of files not having a `.phar` suffix. Besides that | to block any Phar invocation of files not having a `.phar` suffix. Besides that | ||||||
| @@ -137,9 +137,72 @@ class PharExtensionInterceptor implements Assertable | |||||||
| } | } | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | ### ConjunctionInterceptor | ||||||
|  |  | ||||||
|  | This interceptor combines multiple interceptors implementing `Assertable`. | ||||||
|  | It succeeds when all nested interceptors succeed as well (logical `AND`). | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | $behavior = new \TYPO3\PharStreamWrapper\Behavior(); | ||||||
|  | \TYPO3\PharStreamWrapper\Manager::initialize( | ||||||
|  |     $behavior->withAssertion(new ConjunctionInterceptor(array( | ||||||
|  |         new PharExtensionInterceptor(), | ||||||
|  |         new PharMetaDataInterceptor() | ||||||
|  |     ))) | ||||||
|  | ); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### PharExtensionInterceptor | ||||||
|  |  | ||||||
|  | This (basic) interceptor just checks whether the invoked Phar archive has | ||||||
|  | an according `.phar` file extension. Resolving symbolic links as well as | ||||||
|  | Phar internal alias resolving are considered as well. | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | $behavior = new \TYPO3\PharStreamWrapper\Behavior(); | ||||||
|  | \TYPO3\PharStreamWrapper\Manager::initialize( | ||||||
|  |     $behavior->withAssertion(new PharExtensionInterceptor()) | ||||||
|  | ); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### PharMetaDataInterceptor | ||||||
|  |  | ||||||
|  | This interceptor is actually checking serialized Phar meta-data against | ||||||
|  | PHP objects and would consider a Phar archive malicious in case not only | ||||||
|  | scalar values are found. A custom low-level `Phar\Reader` is used in order to | ||||||
|  | avoid using PHP's `Phar` object which would trigger the initial vulnerability. | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | $behavior = new \TYPO3\PharStreamWrapper\Behavior(); | ||||||
|  | \TYPO3\PharStreamWrapper\Manager::initialize( | ||||||
|  |     $behavior->withAssertion(new PharMetaDataInterceptor()) | ||||||
|  | ); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Reader | ||||||
|  |  | ||||||
|  | * `Phar\Reader::__construct(string $fileName)`: Creates low-level reader for Phar archive | ||||||
|  | * `Phar\Reader::resolveContainer(): Phar\Container`: Resolves model representing Phar archive | ||||||
|  | * `Phar\Container::getStub(): Phar\Stub`: Resolves (plain PHP) stub section of Phar archive | ||||||
|  | * `Phar\Container::getManifest(): Phar\Manifest`: Resolves parsed Phar archive manifest as | ||||||
|  |   documented at http://php.net/manual/en/phar.fileformat.manifestfile.php | ||||||
|  | * `Phar\Stub::getMappedAlias(): string`: Resolves internal Phar archive alias defined in stub | ||||||
|  |   using `Phar::mapPhar('alias.phar')` - actually the plain PHP source is analyzed here | ||||||
|  | * `Phar\Manifest::getAlias(): string` - Resolves internal Phar archive alias defined in manifest | ||||||
|  |   using `Phar::setAlias('alias.phar')` | ||||||
|  | * `Phar\Manifest::getMetaData(): string`: Resolves serialized Phar archive meta-data | ||||||
|  | * `Phar\Manifest::deserializeMetaData(): mixed`: Resolves deserialized Phar archive meta-data | ||||||
|  |   containing only scalar values - in case an object is determined, an according | ||||||
|  |   `Phar\DeserializationException` will be thrown | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | $reader = new Phar\Reader('example.phar'); | ||||||
|  | var_dump($reader->resolveContainer()->getManifest()->deserializeMetaData()); | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ## Helper | ## Helper | ||||||
|  |  | ||||||
| * `Helper::determineBaseFile(string $path)`: Determines base file that can be | * `Helper::determineBaseFile(string $path): string`: Determines base file that can be | ||||||
|   accessed using the regular file system. For instance the following path |   accessed using the regular file system. For instance the following path | ||||||
|   `phar:///home/user/bundle.phar/content.txt` would be resolved to |   `phar:///home/user/bundle.phar/content.txt` would be resolved to | ||||||
|   `/home/user/bundle.phar`. |   `/home/user/bundle.phar`. | ||||||
|   | |||||||
| @@ -6,9 +6,13 @@ | |||||||
|     "homepage": "https://typo3.org/", |     "homepage": "https://typo3.org/", | ||||||
|     "keywords": ["php", "phar", "stream-wrapper", "security"], |     "keywords": ["php", "phar", "stream-wrapper", "security"], | ||||||
|     "require": { |     "require": { | ||||||
|         "php": "^5.3.3|^7.0" |         "php": "^5.3.3|^7.0", | ||||||
|  |         "ext-fileinfo": "*", | ||||||
|  |         "ext-json": "*", | ||||||
|  |         "brumann/polyfill-unserialize": "^1.0" | ||||||
|     }, |     }, | ||||||
|     "require-dev": { |     "require-dev": { | ||||||
|  |         "ext-xdebug": "*", | ||||||
|         "phpunit/phpunit": "^4.8.36" |         "phpunit/phpunit": "^4.8.36" | ||||||
|     }, |     }, | ||||||
|     "autoload": { |     "autoload": { | ||||||
|   | |||||||
							
								
								
									
										37
									
								
								misc/typo3/phar-stream-wrapper/src/Collectable.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								misc/typo3/phar-stream-wrapper/src/Collectable.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | use TYPO3\PharStreamWrapper\Resolver\PharInvocation; | ||||||
|  |  | ||||||
|  | interface Collectable | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * @param PharInvocation $invocation | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     public function has(PharInvocation $invocation); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param PharInvocation $invocation | ||||||
|  |      * @param null $flags | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     public function collect(PharInvocation $invocation, $flags = null); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param callable $callback | ||||||
|  |      * @param bool $reverse | ||||||
|  |      * @return null|PharInvocation | ||||||
|  |      */ | ||||||
|  |     public function findByCallback($callback, $reverse = false); | ||||||
|  | } | ||||||
| @@ -11,6 +11,13 @@ namespace TYPO3\PharStreamWrapper; | |||||||
|  * The TYPO3 project - inspiring people to share! |  * The TYPO3 project - inspiring people to share! | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Helper provides low-level tools on file name resolving. However it does not | ||||||
|  |  * (and should not) maintain any runtime state information. In order to resolve | ||||||
|  |  * Phar archive paths according resolvers have to be used. | ||||||
|  |  * | ||||||
|  |  * @see \TYPO3\PharStreamWrapper\Resolvable::resolve() | ||||||
|  |  */ | ||||||
| class Helper | class Helper | ||||||
| { | { | ||||||
|     /* |     /* | ||||||
| @@ -54,6 +61,15 @@ class Helper | |||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $path | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     public static function hasPharPrefix($path) | ||||||
|  |     { | ||||||
|  |         return stripos($path, 'phar://') === 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @param string $path |      * @param string $path | ||||||
|      * @return string |      * @return string | ||||||
| @@ -61,7 +77,7 @@ class Helper | |||||||
|     public static function removePharPrefix($path) |     public static function removePharPrefix($path) | ||||||
|     { |     { | ||||||
|         $path = trim($path); |         $path = trim($path); | ||||||
|         if (stripos($path, 'phar://') !== 0) { |         if (!static::hasPharPrefix($path)) { | ||||||
|             return $path; |             return $path; | ||||||
|         } |         } | ||||||
|         return substr($path, 7); |         return substr($path, 7); | ||||||
| @@ -77,7 +93,7 @@ class Helper | |||||||
|     public static function normalizePath($path) |     public static function normalizePath($path) | ||||||
|     { |     { | ||||||
|         return rtrim( |         return rtrim( | ||||||
|             static::getCanonicalPath( |             static::normalizeWindowsPath( | ||||||
|                 static::removePharPrefix($path) |                 static::removePharPrefix($path) | ||||||
|             ), |             ), | ||||||
|             '/' |             '/' | ||||||
|   | |||||||
| @@ -0,0 +1,88 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Interceptor; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | use TYPO3\PharStreamWrapper\Assertable; | ||||||
|  | use TYPO3\PharStreamWrapper\Exception; | ||||||
|  |  | ||||||
|  | class ConjunctionInterceptor implements Assertable | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * @var Assertable[] | ||||||
|  |      */ | ||||||
|  |     private $assertions; | ||||||
|  |  | ||||||
|  |     public function __construct(array $assertions) | ||||||
|  |     { | ||||||
|  |         $this->assertAssertions($assertions); | ||||||
|  |         $this->assertions = $assertions; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Executes assertions based on all contained assertions. | ||||||
|  |      * | ||||||
|  |      * @param string $path | ||||||
|  |      * @param string $command | ||||||
|  |      * @return bool | ||||||
|  |      * @throws Exception | ||||||
|  |      */ | ||||||
|  |     public function assert($path, $command) | ||||||
|  |     { | ||||||
|  |         if ($this->invokeAssertions($path, $command)) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         throw new Exception( | ||||||
|  |             sprintf( | ||||||
|  |                 'Assertion failed in "%s"', | ||||||
|  |                 $path | ||||||
|  |             ), | ||||||
|  |             1539625084 | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param Assertable[] $assertions | ||||||
|  |      */ | ||||||
|  |     private function assertAssertions(array $assertions) | ||||||
|  |     { | ||||||
|  |         foreach ($assertions as $assertion) { | ||||||
|  |             if (!$assertion instanceof Assertable) { | ||||||
|  |                 throw new \InvalidArgumentException( | ||||||
|  |                     sprintf( | ||||||
|  |                         'Instance %s must implement Assertable', | ||||||
|  |                         get_class($assertion) | ||||||
|  |                     ), | ||||||
|  |                     1539624719 | ||||||
|  |                 ); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $path | ||||||
|  |      * @param string $command | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     private function invokeAssertions($path, $command) | ||||||
|  |     { | ||||||
|  |         try { | ||||||
|  |             foreach ($this->assertions as $assertion) { | ||||||
|  |                 if (!$assertion->assert($path, $command)) { | ||||||
|  |                     return false; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } catch (Exception $exception) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -12,8 +12,8 @@ namespace TYPO3\PharStreamWrapper\Interceptor; | |||||||
|  */ |  */ | ||||||
|  |  | ||||||
| use TYPO3\PharStreamWrapper\Assertable; | use TYPO3\PharStreamWrapper\Assertable; | ||||||
| use TYPO3\PharStreamWrapper\Helper; |  | ||||||
| use TYPO3\PharStreamWrapper\Exception; | use TYPO3\PharStreamWrapper\Exception; | ||||||
|  | use TYPO3\PharStreamWrapper\Manager; | ||||||
|  |  | ||||||
| class PharExtensionInterceptor implements Assertable | class PharExtensionInterceptor implements Assertable | ||||||
| { | { | ||||||
| @@ -45,11 +45,11 @@ class PharExtensionInterceptor implements Assertable | |||||||
|      */ |      */ | ||||||
|     private function baseFileContainsPharExtension($path) |     private function baseFileContainsPharExtension($path) | ||||||
|     { |     { | ||||||
|         $baseFile = Helper::determineBaseFile($path); |         $invocation = Manager::instance()->resolve($path); | ||||||
|         if ($baseFile === null) { |         if ($invocation === null) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         $fileExtension = pathinfo($baseFile, PATHINFO_EXTENSION); |         $fileExtension = pathinfo($invocation->getBaseName(), PATHINFO_EXTENSION); | ||||||
|         return strtolower($fileExtension) === 'phar'; |         return strtolower($fileExtension) === 'phar'; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,73 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Interceptor; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | use TYPO3\PharStreamWrapper\Assertable; | ||||||
|  | use TYPO3\PharStreamWrapper\Exception; | ||||||
|  | use TYPO3\PharStreamWrapper\Manager; | ||||||
|  | use TYPO3\PharStreamWrapper\Phar\DeserializationException; | ||||||
|  | use TYPO3\PharStreamWrapper\Phar\Reader; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @internal Experimental implementation of checking against serialized objects in Phar meta-data | ||||||
|  |  * @internal This functionality has not been 100% pentested... | ||||||
|  |  */ | ||||||
|  | class PharMetaDataInterceptor implements Assertable | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * Determines whether the according Phar archive contains | ||||||
|  |      * (potential insecure) serialized objects. | ||||||
|  |      * | ||||||
|  |      * @param string $path | ||||||
|  |      * @param string $command | ||||||
|  |      * @return bool | ||||||
|  |      * @throws Exception | ||||||
|  |      */ | ||||||
|  |     public function assert($path, $command) | ||||||
|  |     { | ||||||
|  |         if ($this->baseFileDoesNotHaveMetaDataIssues($path)) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         throw new Exception( | ||||||
|  |             sprintf( | ||||||
|  |                 'Problematic meta-data in "%s"', | ||||||
|  |                 $path | ||||||
|  |             ), | ||||||
|  |             1539632368 | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $path | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     private function baseFileDoesNotHaveMetaDataIssues($path) | ||||||
|  |     { | ||||||
|  |         $invocation = Manager::instance()->resolve($path); | ||||||
|  |         if ($invocation === null) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         // directly return in case invocation was checked before | ||||||
|  |         if ($invocation->getVariable(__CLASS__) === true) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         // otherwise analyze meta-data | ||||||
|  |         try { | ||||||
|  |             $reader = new Reader($invocation->getBaseName()); | ||||||
|  |             $reader->resolveContainer()->getManifest()->deserializeMetaData(); | ||||||
|  |             $invocation->setVariable(__CLASS__, true); | ||||||
|  |         } catch (DeserializationException $exception) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -11,7 +11,11 @@ namespace TYPO3\PharStreamWrapper; | |||||||
|  * The TYPO3 project - inspiring people to share! |  * The TYPO3 project - inspiring people to share! | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| class Manager implements Assertable | use TYPO3\PharStreamWrapper\Resolver\PharInvocation; | ||||||
|  | use TYPO3\PharStreamWrapper\Resolver\PharInvocationCollection; | ||||||
|  | use TYPO3\PharStreamWrapper\Resolver\PharInvocationResolver; | ||||||
|  |  | ||||||
|  | class Manager | ||||||
| { | { | ||||||
|     /** |     /** | ||||||
|      * @var self |      * @var self | ||||||
| @@ -23,14 +27,29 @@ class Manager implements Assertable | |||||||
|      */ |      */ | ||||||
|     private $behavior; |     private $behavior; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var Resolvable | ||||||
|  |      */ | ||||||
|  |     private $resolver; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var Collectable | ||||||
|  |      */ | ||||||
|  |     private $collection; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @param Behavior $behaviour |      * @param Behavior $behaviour | ||||||
|  |      * @param Resolvable $resolver | ||||||
|  |      * @param Collectable $collection | ||||||
|      * @return self |      * @return self | ||||||
|      */ |      */ | ||||||
|     public static function initialize(Behavior $behaviour) |     public static function initialize( | ||||||
|     { |         Behavior $behaviour, | ||||||
|  |         Resolvable $resolver = null, | ||||||
|  |         Collectable $collection = null | ||||||
|  |     ) { | ||||||
|         if (self::$instance === null) { |         if (self::$instance === null) { | ||||||
|             self::$instance = new self($behaviour); |             self::$instance = new self($behaviour, $resolver, $collection); | ||||||
|             return self::$instance; |             return self::$instance; | ||||||
|         } |         } | ||||||
|         throw new \LogicException( |         throw new \LogicException( | ||||||
| @@ -67,9 +86,22 @@ class Manager implements Assertable | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @param Behavior $behaviour |      * @param Behavior $behaviour | ||||||
|  |      * @param Resolvable $resolver | ||||||
|  |      * @param Collectable $collection | ||||||
|      */ |      */ | ||||||
|     private function __construct(Behavior $behaviour) |     private function __construct( | ||||||
|     { |         Behavior $behaviour, | ||||||
|  |         Resolvable $resolver = null, | ||||||
|  |         Collectable $collection = null | ||||||
|  |     ) { | ||||||
|  |         if ($collection === null) { | ||||||
|  |             $collection = new PharInvocationCollection(); | ||||||
|  |         } | ||||||
|  |         if ($resolver === null) { | ||||||
|  |             $resolver = new PharInvocationResolver(); | ||||||
|  |         } | ||||||
|  |         $this->collection = $collection; | ||||||
|  |         $this->resolver = $resolver; | ||||||
|         $this->behavior = $behaviour; |         $this->behavior = $behaviour; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -82,4 +114,22 @@ class Manager implements Assertable | |||||||
|     { |     { | ||||||
|         return $this->behavior->assert($path, $command); |         return $this->behavior->assert($path, $command); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $path | ||||||
|  |      * @param null|int $flags | ||||||
|  |      * @return null|PharInvocation | ||||||
|  |      */ | ||||||
|  |     public function resolve($path, $flags = null) | ||||||
|  |     { | ||||||
|  |         return $this->resolver->resolve($path, $flags); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return Collectable | ||||||
|  |      */ | ||||||
|  |     public function getCollection() | ||||||
|  |     { | ||||||
|  |         return $this->collection; | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										59
									
								
								misc/typo3/phar-stream-wrapper/src/Phar/Container.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								misc/typo3/phar-stream-wrapper/src/Phar/Container.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Phar; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | class Container | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * @var Stub | ||||||
|  |      */ | ||||||
|  |     private $stub; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var Manifest | ||||||
|  |      */ | ||||||
|  |     private $manifest; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param Stub $stub | ||||||
|  |      * @param Manifest $manifest | ||||||
|  |      */ | ||||||
|  |     public function __construct(Stub $stub, Manifest $manifest) | ||||||
|  |     { | ||||||
|  |         $this->stub = $stub; | ||||||
|  |         $this->manifest = $manifest; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return Stub | ||||||
|  |      */ | ||||||
|  |     public function getStub() | ||||||
|  |     { | ||||||
|  |         return $this->stub; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return Manifest | ||||||
|  |      */ | ||||||
|  |     public function getManifest() | ||||||
|  |     { | ||||||
|  |         return $this->manifest; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     public function getAlias() | ||||||
|  |     { | ||||||
|  |         return $this->manifest->getAlias() ?: $this->stub->getMappedAlias(); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,18 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Phar; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | use TYPO3\PharStreamWrapper\Exception; | ||||||
|  |  | ||||||
|  | class DeserializationException extends Exception | ||||||
|  | { | ||||||
|  | } | ||||||
							
								
								
									
										176
									
								
								misc/typo3/phar-stream-wrapper/src/Phar/Manifest.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								misc/typo3/phar-stream-wrapper/src/Phar/Manifest.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Phar; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | use Brumann\Polyfill\Unserialize; | ||||||
|  |  | ||||||
|  | class Manifest | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * @param string $content | ||||||
|  |      * @return self | ||||||
|  |      * @see http://php.net/manual/en/phar.fileformat.phar.php | ||||||
|  |      */ | ||||||
|  |     public static function fromContent($content) | ||||||
|  |     { | ||||||
|  |         $target = new static(); | ||||||
|  |         $target->manifestLength = Reader::resolveFourByteLittleEndian($content, 0); | ||||||
|  |         $target->amountOfFiles = Reader::resolveFourByteLittleEndian($content, 4); | ||||||
|  |         $target->flags = Reader::resolveFourByteLittleEndian($content, 10); | ||||||
|  |         $target->aliasLength = Reader::resolveFourByteLittleEndian($content, 14); | ||||||
|  |         $target->alias = substr($content, 18, $target->aliasLength); | ||||||
|  |         $target->metaDataLength = Reader::resolveFourByteLittleEndian($content, 18 + $target->aliasLength); | ||||||
|  |         $target->metaData = substr($content, 22 + $target->aliasLength, $target->metaDataLength); | ||||||
|  |  | ||||||
|  |         $apiVersionNibbles = Reader::resolveTwoByteBigEndian($content, 8); | ||||||
|  |         $target->apiVersion = implode('.', array( | ||||||
|  |             ($apiVersionNibbles & 0xf000) >> 12, | ||||||
|  |             ($apiVersionNibbles & 0x0f00) >> 8, | ||||||
|  |             ($apiVersionNibbles & 0x00f0) >> 4, | ||||||
|  |         )); | ||||||
|  |  | ||||||
|  |         return $target; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var int | ||||||
|  |      */ | ||||||
|  |     private $manifestLength; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var int | ||||||
|  |      */ | ||||||
|  |     private $amountOfFiles; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var string | ||||||
|  |      */ | ||||||
|  |     private $apiVersion; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var int | ||||||
|  |      */ | ||||||
|  |     private $flags; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var int | ||||||
|  |      */ | ||||||
|  |     private $aliasLength; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var string | ||||||
|  |      */ | ||||||
|  |     private $alias; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var int | ||||||
|  |      */ | ||||||
|  |     private $metaDataLength; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var string | ||||||
|  |      */ | ||||||
|  |     private $metaData; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Avoid direct instantiation. | ||||||
|  |      */ | ||||||
|  |     private function __construct() | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return int | ||||||
|  |      */ | ||||||
|  |     public function getManifestLength() | ||||||
|  |     { | ||||||
|  |         return $this->manifestLength; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return int | ||||||
|  |      */ | ||||||
|  |     public function getAmountOfFiles() | ||||||
|  |     { | ||||||
|  |         return $this->amountOfFiles; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     public function getApiVersion() | ||||||
|  |     { | ||||||
|  |         return $this->apiVersion; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return int | ||||||
|  |      */ | ||||||
|  |     public function getFlags() | ||||||
|  |     { | ||||||
|  |         return $this->flags; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return int | ||||||
|  |      */ | ||||||
|  |     public function getAliasLength() | ||||||
|  |     { | ||||||
|  |         return $this->aliasLength; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     public function getAlias() | ||||||
|  |     { | ||||||
|  |         return $this->alias; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return int | ||||||
|  |      */ | ||||||
|  |     public function getMetaDataLength() | ||||||
|  |     { | ||||||
|  |         return $this->metaDataLength; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     public function getMetaData() | ||||||
|  |     { | ||||||
|  |         return $this->metaData; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return mixed|null | ||||||
|  |      */ | ||||||
|  |     public function deserializeMetaData() | ||||||
|  |     { | ||||||
|  |         if (empty($this->metaData)) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $result = Unserialize::unserialize($this->metaData, array('allowed_classes' => false)); | ||||||
|  |  | ||||||
|  |         $serialized = json_encode($result); | ||||||
|  |         if (strpos($serialized, '__PHP_Incomplete_Class_Name') !== false) { | ||||||
|  |             throw new DeserializationException( | ||||||
|  |                 'Meta-data contains serialized object', | ||||||
|  |                 1539623382 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return $result; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										220
									
								
								misc/typo3/phar-stream-wrapper/src/Phar/Reader.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										220
									
								
								misc/typo3/phar-stream-wrapper/src/Phar/Reader.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,220 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Phar; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | class Reader | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * @var string | ||||||
|  |      */ | ||||||
|  |     private $fileName; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var string | ||||||
|  |      */ | ||||||
|  |     private $fileType; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $fileName | ||||||
|  |      */ | ||||||
|  |     public function __construct($fileName) | ||||||
|  |     { | ||||||
|  |         if (strpos($fileName, '://') !== false) { | ||||||
|  |             throw new ReaderException( | ||||||
|  |                 'File name must not contain stream prefix', | ||||||
|  |                 1539623708 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $this->fileName = $fileName; | ||||||
|  |         $this->fileType = $this->determineFileType(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return Container | ||||||
|  |      */ | ||||||
|  |     public function resolveContainer() | ||||||
|  |     { | ||||||
|  |         $data = $this->extractData($this->resolveStream() . $this->fileName); | ||||||
|  |  | ||||||
|  |         if ($data['stubContent'] === null) { | ||||||
|  |             throw new ReaderException( | ||||||
|  |                 'Cannot resolve stub', | ||||||
|  |                 1547807881 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         if ($data['manifestContent'] === null || $data['manifestLength'] === null) { | ||||||
|  |             throw new ReaderException( | ||||||
|  |                 'Cannot resolve manifest', | ||||||
|  |                 1547807882 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         if (strlen($data['manifestContent']) < $data['manifestLength']) { | ||||||
|  |             throw new ReaderException( | ||||||
|  |                 sprintf( | ||||||
|  |                     'Exected manifest length %d, got %d', | ||||||
|  |                     strlen($data['manifestContent']), | ||||||
|  |                     $data['manifestLength'] | ||||||
|  |                 ), | ||||||
|  |                 1547807883 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return new Container( | ||||||
|  |             Stub::fromContent($data['stubContent']), | ||||||
|  |             Manifest::fromContent($data['manifestContent']) | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $fileName e.g. '/path/file.phar' or 'compress.zlib:///path/file.phar' | ||||||
|  |      * @return array | ||||||
|  |      */ | ||||||
|  |     private function extractData($fileName) | ||||||
|  |     { | ||||||
|  |         $stubContent = null; | ||||||
|  |         $manifestContent = null; | ||||||
|  |         $manifestLength = null; | ||||||
|  |  | ||||||
|  |         $resource = fopen($fileName, 'r'); | ||||||
|  |         if (!is_resource($resource)) { | ||||||
|  |             throw new ReaderException( | ||||||
|  |                 sprintf('Resource %s could not be opened', $fileName), | ||||||
|  |                 1547902055 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         while (!feof($resource)) { | ||||||
|  |             $line = fgets($resource); | ||||||
|  |             // stop reading file when manifest can be extracted | ||||||
|  |             if ($manifestLength !== null && $manifestContent !== null && strlen($manifestContent) >= $manifestLength) { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             $manifestPosition = strpos($line, '__HALT_COMPILER();'); | ||||||
|  |  | ||||||
|  |             // first line contains start of manifest | ||||||
|  |             if ($stubContent === null && $manifestContent === null && $manifestPosition !== false) { | ||||||
|  |                 $stubContent = substr($line, 0, $manifestPosition - 1); | ||||||
|  |                 $manifestContent = preg_replace('#^.*__HALT_COMPILER\(\);(?>[ \n]\?>(?>\r\n|\n)?)?#', '', $line); | ||||||
|  |                 $manifestLength = $this->resolveManifestLength($manifestContent); | ||||||
|  |             // line contains start of stub | ||||||
|  |             } elseif ($stubContent === null) { | ||||||
|  |                 $stubContent = $line; | ||||||
|  |             // line contains start of manifest | ||||||
|  |             } elseif ($manifestContent === null && $manifestPosition !== false) { | ||||||
|  |                 $manifestContent = preg_replace('#^.*__HALT_COMPILER\(\);(?>[ \n]\?>(?>\r\n|\n)?)?#', '', $line); | ||||||
|  |                 $manifestLength = $this->resolveManifestLength($manifestContent); | ||||||
|  |             // manifest has been started (thus is cannot be stub anymore), add content | ||||||
|  |             } elseif ($manifestContent !== null) { | ||||||
|  |                 $manifestContent .= $line; | ||||||
|  |                 $manifestLength = $this->resolveManifestLength($manifestContent); | ||||||
|  |             // stub has been started (thus cannot be manifest here, yet), add content | ||||||
|  |             } elseif ($stubContent !== null) { | ||||||
|  |                 $stubContent .= $line; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         fclose($resource); | ||||||
|  |  | ||||||
|  |         return array( | ||||||
|  |             'stubContent' => $stubContent, | ||||||
|  |             'manifestContent' => $manifestContent, | ||||||
|  |             'manifestLength' => $manifestLength, | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Resolves stream in order to handle compressed Phar archives. | ||||||
|  |      * | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     private function resolveStream() | ||||||
|  |     { | ||||||
|  |         if ($this->fileType === 'application/x-gzip') { | ||||||
|  |             return 'compress.zlib://'; | ||||||
|  |         } elseif ($this->fileType === 'application/x-bzip2') { | ||||||
|  |             return 'compress.bzip2://'; | ||||||
|  |         } | ||||||
|  |         return ''; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     private function determineFileType() | ||||||
|  |     { | ||||||
|  |         $fileInfo = new \finfo(); | ||||||
|  |         return $fileInfo->file($this->fileName, FILEINFO_MIME_TYPE); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $content | ||||||
|  |      * @return int|null | ||||||
|  |      */ | ||||||
|  |     private function resolveManifestLength($content) | ||||||
|  |     { | ||||||
|  |         if (strlen($content) < 4) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return static::resolveFourByteLittleEndian($content, 0); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $content | ||||||
|  |      * @param int $start | ||||||
|  |      * @return int | ||||||
|  |      */ | ||||||
|  |     public static function resolveFourByteLittleEndian($content, $start) | ||||||
|  |     { | ||||||
|  |         $payload = substr($content, $start, 4); | ||||||
|  |         if (!is_string($payload)) { | ||||||
|  |             throw new ReaderException( | ||||||
|  |                 sprintf('Cannot resolve value at offset %d', $start), | ||||||
|  |                 1539614260 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $value = unpack('V', $payload); | ||||||
|  |         if (!isset($value[1])) { | ||||||
|  |             throw new ReaderException( | ||||||
|  |                 sprintf('Cannot resolve value at offset %d', $start), | ||||||
|  |                 1539614261 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         return $value[1]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $content | ||||||
|  |      * @param int $start | ||||||
|  |      * @return int | ||||||
|  |      */ | ||||||
|  |     public static function resolveTwoByteBigEndian($content, $start) | ||||||
|  |     { | ||||||
|  |         $payload = substr($content, $start, 2); | ||||||
|  |         if (!is_string($payload)) { | ||||||
|  |             throw new ReaderException( | ||||||
|  |                 sprintf('Cannot resolve value at offset %d', $start), | ||||||
|  |                 1539614263 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $value = unpack('n', $payload); | ||||||
|  |         if (!isset($value[1])) { | ||||||
|  |             throw new ReaderException( | ||||||
|  |                 sprintf('Cannot resolve value at offset %d', $start), | ||||||
|  |                 1539614264 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         return $value[1]; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										18
									
								
								misc/typo3/phar-stream-wrapper/src/Phar/ReaderException.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								misc/typo3/phar-stream-wrapper/src/Phar/ReaderException.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Phar; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | use TYPO3\PharStreamWrapper\Exception; | ||||||
|  |  | ||||||
|  | class ReaderException extends Exception | ||||||
|  | { | ||||||
|  | } | ||||||
							
								
								
									
										65
									
								
								misc/typo3/phar-stream-wrapper/src/Phar/Stub.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								misc/typo3/phar-stream-wrapper/src/Phar/Stub.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Phar; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @internal Experimental implementation of Phar archive internals | ||||||
|  |  */ | ||||||
|  | class Stub | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * @param string $content | ||||||
|  |      * @return self | ||||||
|  |      */ | ||||||
|  |     public static function fromContent($content) | ||||||
|  |     { | ||||||
|  |         $target = new static(); | ||||||
|  |         $target->content = $content; | ||||||
|  |  | ||||||
|  |         if ( | ||||||
|  |             stripos($content, 'Phar::mapPhar(') !== false | ||||||
|  |             && preg_match('#Phar\:\:mapPhar\(([^)]+)\)#', $content, $matches) | ||||||
|  |         ) { | ||||||
|  |             // remove spaces, single & double quotes | ||||||
|  |             // @todo `'my' . 'alias' . '.phar'` is not evaluated here | ||||||
|  |             $target->mappedAlias = trim($matches[1], ' \'"'); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return $target; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var string | ||||||
|  |      */ | ||||||
|  |     private $content; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var string | ||||||
|  |      */ | ||||||
|  |     private $mappedAlias = ''; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     public function getContent() | ||||||
|  |     { | ||||||
|  |         return $this->content; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     public function getMappedAlias() | ||||||
|  |     { | ||||||
|  |         return $this->mappedAlias; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -11,6 +11,8 @@ namespace TYPO3\PharStreamWrapper; | |||||||
|  * The TYPO3 project - inspiring people to share! |  * The TYPO3 project - inspiring people to share! | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
|  | use TYPO3\PharStreamWrapper\Resolver\PharInvocation; | ||||||
|  |  | ||||||
| class PharStreamWrapper | class PharStreamWrapper | ||||||
| { | { | ||||||
|     /** |     /** | ||||||
| @@ -29,6 +31,11 @@ class PharStreamWrapper | |||||||
|      */ |      */ | ||||||
|     protected $internalResource; |     protected $internalResource; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var PharInvocation | ||||||
|  |      */ | ||||||
|  |     protected $invocation; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @return bool |      * @return bool | ||||||
|      */ |      */ | ||||||
| @@ -409,7 +416,8 @@ class PharStreamWrapper | |||||||
|      */ |      */ | ||||||
|     protected function assert($path, $command) |     protected function assert($path, $command) | ||||||
|     { |     { | ||||||
|         if ($this->resolveAssertable()->assert($path, $command) === true) { |         if (Manager::instance()->assert($path, $command) === true) { | ||||||
|  |             $this->collectInvocation($path); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -424,7 +432,33 @@ class PharStreamWrapper | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @return Assertable |      * @param string $path | ||||||
|  |      */ | ||||||
|  |     protected function collectInvocation($path) | ||||||
|  |     { | ||||||
|  |         if (isset($this->invocation)) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $manager = Manager::instance(); | ||||||
|  |         $this->invocation = $manager->resolve($path); | ||||||
|  |         if ($this->invocation === null) { | ||||||
|  |             throw new Exception( | ||||||
|  |                 'Expected invocation could not be resolved', | ||||||
|  |                 1556389591 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         // confirm, previous interceptor(s) validated invocation | ||||||
|  |         $this->invocation->confirm(); | ||||||
|  |         $collection = $manager->getCollection(); | ||||||
|  |         if (!$collection->has($this->invocation)) { | ||||||
|  |             $collection->collect($this->invocation); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return Manager|Assertable | ||||||
|  |      * @deprecated Use Manager::instance() directly | ||||||
|      */ |      */ | ||||||
|     protected function resolveAssertable() |     protected function resolveAssertable() | ||||||
|     { |     { | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								misc/typo3/phar-stream-wrapper/src/Resolvable.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								misc/typo3/phar-stream-wrapper/src/Resolvable.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | use TYPO3\PharStreamWrapper\Resolver\PharInvocation; | ||||||
|  |  | ||||||
|  | interface Resolvable | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * @param string $path | ||||||
|  |      * @param null|int $flags | ||||||
|  |      * @return null|PharInvocation | ||||||
|  |      */ | ||||||
|  |     public function resolve($path, $flags = null); | ||||||
|  | } | ||||||
							
								
								
									
										125
									
								
								misc/typo3/phar-stream-wrapper/src/Resolver/PharInvocation.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								misc/typo3/phar-stream-wrapper/src/Resolver/PharInvocation.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,125 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Resolver; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | use TYPO3\PharStreamWrapper\Exception; | ||||||
|  |  | ||||||
|  | class PharInvocation | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * @var string | ||||||
|  |      */ | ||||||
|  |     private $baseName; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var string | ||||||
|  |      */ | ||||||
|  |     private $alias; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var bool | ||||||
|  |      * @see \TYPO3\PharStreamWrapper\PharStreamWrapper::collectInvocation() | ||||||
|  |      */ | ||||||
|  |     private $confirmed = false; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Arbitrary variables to be used by interceptors as registry | ||||||
|  |      * (e.g. in order to avoid duplicate processing and assertions) | ||||||
|  |      * | ||||||
|  |      * @var array | ||||||
|  |      */ | ||||||
|  |     private $variables; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $baseName | ||||||
|  |      * @param string $alias | ||||||
|  |      */ | ||||||
|  |     public function __construct($baseName, $alias = '') | ||||||
|  |     { | ||||||
|  |         if ($baseName === '') { | ||||||
|  |             throw new Exception( | ||||||
|  |                 'Base-name cannot be empty', | ||||||
|  |                 1551283689 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         $this->baseName = $baseName; | ||||||
|  |         $this->alias = $alias; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     public function __toString() | ||||||
|  |     { | ||||||
|  |         return $this->baseName; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     public function getBaseName() | ||||||
|  |     { | ||||||
|  |         return $this->baseName; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return null|string | ||||||
|  |      */ | ||||||
|  |     public function getAlias() | ||||||
|  |     { | ||||||
|  |         return $this->alias; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     public function isConfirmed() | ||||||
|  |     { | ||||||
|  |         return $this->confirmed; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function confirm() | ||||||
|  |     { | ||||||
|  |         $this->confirmed = true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $name | ||||||
|  |      * @return mixed|null | ||||||
|  |      */ | ||||||
|  |     public function getVariable($name) | ||||||
|  |     { | ||||||
|  |         if (!isset($this->variables[$name])) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return $this->variables[$name]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $name | ||||||
|  |      * @param mixed $value | ||||||
|  |      */ | ||||||
|  |     public function setVariable($name, $value) | ||||||
|  |     { | ||||||
|  |         $this->variables[$name] = $value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param PharInvocation $other | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     public function equals(PharInvocation $other) | ||||||
|  |     { | ||||||
|  |         return $other->baseName === $this->baseName | ||||||
|  |             && $other->alias === $this->alias; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,156 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Resolver; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | use TYPO3\PharStreamWrapper\Collectable; | ||||||
|  |  | ||||||
|  | class PharInvocationCollection implements Collectable | ||||||
|  | { | ||||||
|  |     const UNIQUE_INVOCATION = 1; | ||||||
|  |     const UNIQUE_BASE_NAME = 2; | ||||||
|  |     const DUPLICATE_ALIAS_WARNING = 32; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var PharInvocation[] | ||||||
|  |      */ | ||||||
|  |     private $invocations = array(); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param PharInvocation $invocation | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     public function has(PharInvocation $invocation) | ||||||
|  |     { | ||||||
|  |         return in_array($invocation, $this->invocations, true); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param PharInvocation $invocation | ||||||
|  |      * @param null|int $flags | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     public function collect(PharInvocation $invocation, $flags = null) | ||||||
|  |     { | ||||||
|  |         if ($flags === null) { | ||||||
|  |             $flags = static::UNIQUE_INVOCATION | static::DUPLICATE_ALIAS_WARNING; | ||||||
|  |         } | ||||||
|  |         if ($invocation->getBaseName() === '' | ||||||
|  |             || $invocation->getAlias() === '' | ||||||
|  |             || !$this->assertUniqueBaseName($invocation, $flags) | ||||||
|  |             || !$this->assertUniqueInvocation($invocation, $flags) | ||||||
|  |         ) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         if ($flags & static::DUPLICATE_ALIAS_WARNING) { | ||||||
|  |             $this->triggerDuplicateAliasWarning($invocation); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $this->invocations[] = $invocation; | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param callable $callback | ||||||
|  |      * @param bool $reverse | ||||||
|  |      * @return null|PharInvocation | ||||||
|  |      */ | ||||||
|  |     public function findByCallback($callback, $reverse = false) | ||||||
|  |     { | ||||||
|  |         foreach ($this->getInvocations($reverse) as $invocation) { | ||||||
|  |             if (call_user_func($callback, $invocation) === true) { | ||||||
|  |                 return $invocation; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Asserts that base-name is unique. This disallows having multiple invocations for | ||||||
|  |      * same base-name but having different alias names. | ||||||
|  |      * | ||||||
|  |      * @param PharInvocation $invocation | ||||||
|  |      * @param int $flags | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     private function assertUniqueBaseName(PharInvocation $invocation, $flags) | ||||||
|  |     { | ||||||
|  |         if (!($flags & static::UNIQUE_BASE_NAME)) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         return $this->findByCallback( | ||||||
|  |                 function (PharInvocation $candidate) use ($invocation) { | ||||||
|  |                     return $candidate->getBaseName() === $invocation->getBaseName(); | ||||||
|  |                 } | ||||||
|  |             ) === null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Asserts that combination of base-name and alias is unique. This allows having multiple | ||||||
|  |      * invocations for same base-name but having different alias names (for whatever reason). | ||||||
|  |      * | ||||||
|  |      * @param PharInvocation $invocation | ||||||
|  |      * @param int $flags | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     private function assertUniqueInvocation(PharInvocation $invocation, $flags) | ||||||
|  |     { | ||||||
|  |         if (!($flags & static::UNIQUE_INVOCATION)) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         return $this->findByCallback( | ||||||
|  |                 function (PharInvocation $candidate) use ($invocation) { | ||||||
|  |                     return $candidate->equals($invocation); | ||||||
|  |                 } | ||||||
|  |             ) === null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Triggers warning for invocations with same alias and same confirmation state. | ||||||
|  |      * | ||||||
|  |      * @param PharInvocation $invocation | ||||||
|  |      * @see \TYPO3\PharStreamWrapper\PharStreamWrapper::collectInvocation() | ||||||
|  |      */ | ||||||
|  |     private function triggerDuplicateAliasWarning(PharInvocation $invocation) | ||||||
|  |     { | ||||||
|  |         $sameAliasInvocation = $this->findByCallback( | ||||||
|  |             function (PharInvocation $candidate) use ($invocation) { | ||||||
|  |                 return $candidate->isConfirmed() === $invocation->isConfirmed() | ||||||
|  |                     && $candidate->getAlias() === $invocation->getAlias(); | ||||||
|  |             }, | ||||||
|  |             true | ||||||
|  |         ); | ||||||
|  |         if ($sameAliasInvocation === null) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         trigger_error( | ||||||
|  |             sprintf( | ||||||
|  |                 'Alias %s cannot be used by %s, already used by %s', | ||||||
|  |                 $invocation->getAlias(), | ||||||
|  |                 $invocation->getBaseName(), | ||||||
|  |                 $sameAliasInvocation->getBaseName() | ||||||
|  |             ), | ||||||
|  |             E_USER_WARNING | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param bool $reverse | ||||||
|  |      * @return PharInvocation[] | ||||||
|  |      */ | ||||||
|  |     private function getInvocations($reverse = false) | ||||||
|  |     { | ||||||
|  |         if ($reverse) { | ||||||
|  |             return array_reverse($this->invocations); | ||||||
|  |         } | ||||||
|  |         return $this->invocations; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,241 @@ | |||||||
|  | <?php | ||||||
|  | namespace TYPO3\PharStreamWrapper\Resolver; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * This file is part of the TYPO3 project. | ||||||
|  |  * | ||||||
|  |  * It is free software; you can redistribute it and/or modify it under the terms | ||||||
|  |  * of the MIT License (MIT). For the full copyright and license information, | ||||||
|  |  * please read the LICENSE file that was distributed with this source code. | ||||||
|  |  * | ||||||
|  |  * The TYPO3 project - inspiring people to share! | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | use TYPO3\PharStreamWrapper\Helper; | ||||||
|  | use TYPO3\PharStreamWrapper\Manager; | ||||||
|  | use TYPO3\PharStreamWrapper\Phar\Reader; | ||||||
|  | use TYPO3\PharStreamWrapper\Resolvable; | ||||||
|  |  | ||||||
|  | class PharInvocationResolver implements Resolvable | ||||||
|  | { | ||||||
|  |     const RESOLVE_REALPATH = 1; | ||||||
|  |     const RESOLVE_ALIAS = 2; | ||||||
|  |     const ASSERT_INTERNAL_INVOCATION = 32; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @var string[] | ||||||
|  |      */ | ||||||
|  |     private $invocationFunctionNames = array( | ||||||
|  |         'include', | ||||||
|  |         'include_once', | ||||||
|  |         'require', | ||||||
|  |         'require_once' | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Contains resolved base names in order to reduce file IO. | ||||||
|  |      * | ||||||
|  |      * @var string[] | ||||||
|  |      */ | ||||||
|  |     private $baseNames = array(); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Resolves PharInvocation value object (baseName and optional alias). | ||||||
|  |      * | ||||||
|  |      * Phar aliases are intended to be used only inside Phar archives, however | ||||||
|  |      * PharStreamWrapper needs this information exposed outside of Phar as well | ||||||
|  |      * It is possible that same alias is used for different $baseName values. | ||||||
|  |      * That's why PharInvocationCollection behaves like a stack when resolving | ||||||
|  |      * base-name for a given alias. On the other hand it is not possible that | ||||||
|  |      * one $baseName is referring to multiple aliases. | ||||||
|  |      * @see https://secure.php.net/manual/en/phar.setalias.php | ||||||
|  |      * @see https://secure.php.net/manual/en/phar.mapphar.php | ||||||
|  |      * | ||||||
|  |      * @param string $path | ||||||
|  |      * @param int|null $flags | ||||||
|  |      * @return null|PharInvocation | ||||||
|  |      */ | ||||||
|  |     public function resolve($path, $flags = null) | ||||||
|  |     { | ||||||
|  |         $hasPharPrefix = Helper::hasPharPrefix($path); | ||||||
|  |         if ($flags === null) { | ||||||
|  |             $flags = static::RESOLVE_REALPATH | static::RESOLVE_ALIAS | static::ASSERT_INTERNAL_INVOCATION; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if ($hasPharPrefix && $flags & static::RESOLVE_ALIAS) { | ||||||
|  |             $invocation = $this->findByAlias($path); | ||||||
|  |             if ($invocation !== null) { | ||||||
|  |                 return $invocation; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $baseName = $this->resolveBaseName($path, $flags); | ||||||
|  |         if ($baseName === null) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if ($flags & static::RESOLVE_REALPATH) { | ||||||
|  |             $baseName = $this->baseNames[$baseName]; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return $this->retrieveInvocation($baseName, $flags); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Retrieves PharInvocation, either existing in collection or created on demand | ||||||
|  |      * with resolving a potential alias name used in the according Phar archive. | ||||||
|  |      * | ||||||
|  |      * @param string $baseName | ||||||
|  |      * @param int $flags | ||||||
|  |      * @return PharInvocation | ||||||
|  |      */ | ||||||
|  |     private function retrieveInvocation($baseName, $flags) | ||||||
|  |     { | ||||||
|  |         $invocation = $this->findByBaseName($baseName); | ||||||
|  |         if ($invocation !== null) { | ||||||
|  |             return $invocation; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if ($flags & static::RESOLVE_ALIAS) { | ||||||
|  |             $reader = new Reader($baseName); | ||||||
|  |             $alias = $reader->resolveContainer()->getAlias(); | ||||||
|  |         } else { | ||||||
|  |             $alias = ''; | ||||||
|  |         } | ||||||
|  |         // add unconfirmed(!) new invocation to collection | ||||||
|  |         $invocation = new PharInvocation($baseName, $alias); | ||||||
|  |         Manager::instance()->getCollection()->collect($invocation); | ||||||
|  |         return $invocation; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $path | ||||||
|  |      * @param int $flags | ||||||
|  |      * @return null|string | ||||||
|  |      */ | ||||||
|  |     private function resolveBaseName($path, $flags) | ||||||
|  |     { | ||||||
|  |         $baseName = $this->findInBaseNames($path); | ||||||
|  |         if ($baseName !== null) { | ||||||
|  |             return $baseName; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $baseName = Helper::determineBaseFile($path); | ||||||
|  |         if ($baseName !== null) { | ||||||
|  |             $this->addBaseName($baseName); | ||||||
|  |             return $baseName; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $possibleAlias = $this->resolvePossibleAlias($path); | ||||||
|  |         if (!($flags & static::RESOLVE_ALIAS) || $possibleAlias === null) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $trace = debug_backtrace(); | ||||||
|  |         foreach ($trace as $item) { | ||||||
|  |             if (!isset($item['function']) || !isset($item['args'][0]) | ||||||
|  |                 || !in_array($item['function'], $this->invocationFunctionNames, true)) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |             $currentPath = $item['args'][0]; | ||||||
|  |             if (Helper::hasPharPrefix($currentPath)) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |             $currentBaseName = Helper::determineBaseFile($currentPath); | ||||||
|  |             if ($currentBaseName === null) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |             // ensure the possible alias name (how we have been called initially) matches | ||||||
|  |             // the resolved alias name that was retrieved by the current possible base name | ||||||
|  |             $reader = new Reader($currentBaseName); | ||||||
|  |             $currentAlias = $reader->resolveContainer()->getAlias(); | ||||||
|  |             if ($currentAlias !== $possibleAlias) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |             $this->addBaseName($currentBaseName); | ||||||
|  |             return $currentBaseName; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $path | ||||||
|  |      * @return null|string | ||||||
|  |      */ | ||||||
|  |     private function resolvePossibleAlias($path) | ||||||
|  |     { | ||||||
|  |         $normalizedPath = Helper::normalizePath($path); | ||||||
|  |         return strstr($normalizedPath, '/', true) ?: null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $baseName | ||||||
|  |      * @return null|PharInvocation | ||||||
|  |      */ | ||||||
|  |     private function findByBaseName($baseName) | ||||||
|  |     { | ||||||
|  |         return Manager::instance()->getCollection()->findByCallback( | ||||||
|  |             function (PharInvocation $candidate) use ($baseName) { | ||||||
|  |                 return $candidate->getBaseName() === $baseName; | ||||||
|  |             }, | ||||||
|  |             true | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $path | ||||||
|  |      * @return null|string | ||||||
|  |      */ | ||||||
|  |     private function findInBaseNames($path) | ||||||
|  |     { | ||||||
|  |         // return directly if the resolved base name was submitted | ||||||
|  |         if (in_array($path, $this->baseNames, true)) { | ||||||
|  |             return $path; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $parts = explode('/', Helper::normalizePath($path)); | ||||||
|  |  | ||||||
|  |         while (count($parts)) { | ||||||
|  |             $currentPath = implode('/', $parts); | ||||||
|  |             if (isset($this->baseNames[$currentPath])) { | ||||||
|  |                 return $currentPath; | ||||||
|  |             } | ||||||
|  |             array_pop($parts); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param string $baseName | ||||||
|  |      */ | ||||||
|  |     private function addBaseName($baseName) | ||||||
|  |     { | ||||||
|  |         if (isset($this->baseNames[$baseName])) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         $this->baseNames[$baseName] = realpath($baseName); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Finds confirmed(!) invocations by alias. | ||||||
|  |      * | ||||||
|  |      * @param string $path | ||||||
|  |      * @return null|PharInvocation | ||||||
|  |      * @see \TYPO3\PharStreamWrapper\PharStreamWrapper::collectInvocation() | ||||||
|  |      */ | ||||||
|  |     private function findByAlias($path) | ||||||
|  |     { | ||||||
|  |         $possibleAlias = $this->resolvePossibleAlias($path); | ||||||
|  |         if ($possibleAlias === null) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return Manager::instance()->getCollection()->findByCallback( | ||||||
|  |             function (PharInvocation $candidate) use ($possibleAlias) { | ||||||
|  |                 return $candidate->isConfirmed() && $candidate->getAlias() === $possibleAlias; | ||||||
|  |             }, | ||||||
|  |             true | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -7,7 +7,7 @@ files[] = aggregator.test | |||||||
| configure = admin/config/services/aggregator/settings | configure = admin/config/services/aggregator/settings | ||||||
| stylesheets[all][] = aggregator.css | stylesheets[all][] = aggregator.css | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| files[] = block.test | files[] = block.test | ||||||
| configure = admin/structure/block | configure = admin/structure/block | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -13,7 +13,7 @@ regions[footer] = Footer | |||||||
| regions[highlighted] = Highlighted | regions[highlighted] = Highlighted | ||||||
| regions[help] = Help | regions[help] = Help | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| files[] = blog.test | files[] = blog.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ files[] = book.test | |||||||
| configure = admin/content/book/settings | configure = admin/content/book/settings | ||||||
| stylesheets[all][] = book.css | stylesheets[all][] = book.css | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| files[] = color.test | files[] = color.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ files[] = comment.test | |||||||
| configure = admin/content/comment | configure = admin/content/comment | ||||||
| stylesheets[all][] = comment.css | stylesheets[all][] = comment.css | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| files[] = contact.test | files[] = contact.test | ||||||
| configure = admin/structure/contact | configure = admin/structure/contact | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| files[] = contextual.test | files[] = contextual.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ files[] = dashboard.test | |||||||
| dependencies[] = block | dependencies[] = block | ||||||
| configure = admin/dashboard/customize | configure = admin/dashboard/customize | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| files[] = dblog.test | files[] = dblog.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ dependencies[] = field_sql_storage | |||||||
| required = TRUE | required = TRUE | ||||||
| stylesheets[all][] = theme/field.css | stylesheets[all][] = theme/field.css | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ dependencies[] = field | |||||||
| files[] = field_sql_storage.test | files[] = field_sql_storage.test | ||||||
| required = TRUE | required = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ dependencies[] = field | |||||||
| dependencies[] = options | dependencies[] = options | ||||||
| files[] = tests/list.test | files[] = tests/list.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ package = Testing | |||||||
| version = VERSION | version = VERSION | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| dependencies[] = field | dependencies[] = field | ||||||
| files[] = number.test | files[] = number.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -69,7 +69,7 @@ class NumberFieldTestCase extends DrupalWebTestCase { | |||||||
|     preg_match('|test-entity/manage/(\d+)/edit|', $this->url, $match); |     preg_match('|test-entity/manage/(\d+)/edit|', $this->url, $match); | ||||||
|     $id = $match[1]; |     $id = $match[1]; | ||||||
|     $this->assertRaw(t('test_entity @id has been created.', array('@id' => $id)), 'Entity was created'); |     $this->assertRaw(t('test_entity @id has been created.', array('@id' => $id)), 'Entity was created'); | ||||||
|     $this->assertRaw(round($value, 2), 'Value is displayed.'); |     $this->assertRaw($value, 'Value is displayed.'); | ||||||
|  |  | ||||||
|     // Try to create entries with more than one decimal separator; assert fail. |     // Try to create entries with more than one decimal separator; assert fail. | ||||||
|     $wrong_entries = array( |     $wrong_entries = array( | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| dependencies[] = field | dependencies[] = field | ||||||
| files[] = options.test | files[] = options.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ dependencies[] = field | |||||||
| files[] = text.test | files[] = text.test | ||||||
| required = TRUE | required = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ files[] = field_test.entity.inc | |||||||
| version = VERSION | version = VERSION | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| dependencies[] = field | dependencies[] = field | ||||||
| files[] = field_ui.test | files[] = field_ui.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -599,7 +599,7 @@ function file_field_widget_value($element, $input = FALSE, $form_state) { | |||||||
|     // If the display field is present make sure its unchecked value is saved. |     // If the display field is present make sure its unchecked value is saved. | ||||||
|     $field = field_widget_field($element, $form_state); |     $field = field_widget_field($element, $form_state); | ||||||
|     if (empty($input['display'])) { |     if (empty($input['display'])) { | ||||||
|       $input['display'] = $field['settings']['display_field'] ? 0 : 1; |       $input['display'] = !empty($field['settings']['display_field']) ? 0 : 1; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| dependencies[] = field | dependencies[] = field | ||||||
| files[] = tests/file.test | files[] = tests/file.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -1875,3 +1875,60 @@ class FileFieldAnonymousSubmission extends FileFieldTestCase { | |||||||
|   } |   } | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Tests the file_scan_directory() function. | ||||||
|  |  */ | ||||||
|  | class FileScanDirectory extends FileFieldTestCase { | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * @var string | ||||||
|  |    */ | ||||||
|  |   protected $path; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * {@inheritdoc} | ||||||
|  |    */ | ||||||
|  |   public static function getInfo() { | ||||||
|  |     return array( | ||||||
|  |       'name' => 'File ScanDirectory', | ||||||
|  |       'description' => 'Tests the file_scan_directory() function.', | ||||||
|  |       'group' => 'File', | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * {@inheritdoc} | ||||||
|  |    */ | ||||||
|  |   function setUp() { | ||||||
|  |     parent::setUp(); | ||||||
|  |  | ||||||
|  |     $this->path = 'modules/file/tests/fixtures/file_scan_ignore'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Tests file_scan_directory() obeys 'file_scan_ignore_directories' setting. | ||||||
|  |    * If nomask is not passed as argument, it should use the default settings. | ||||||
|  |    * If nomask is passed as argument, it should obey this rule. | ||||||
|  |    */ | ||||||
|  |   public function testNoMask() { | ||||||
|  |     $files = file_scan_directory($this->path, '/\.txt$/'); | ||||||
|  |     $this->assertEqual(3, count($files), '3 text files found when not ignoring directories.'); | ||||||
|  |  | ||||||
|  |     global $conf; | ||||||
|  |     $conf['file_scan_ignore_directories'] = array('frontend_framework'); | ||||||
|  |  | ||||||
|  |     $files = file_scan_directory($this->path, '/\.txt$/'); | ||||||
|  |     $this->assertEqual(1, count($files), '1 text files found when ignoring directories called "frontend_framework".'); | ||||||
|  |  | ||||||
|  |     // Make that directories specified by default still work when a new nomask is provided. | ||||||
|  |     $files = file_scan_directory($this->path, '/\.txt$/', array('nomask' => '/^c.txt/')); | ||||||
|  |     $this->assertEqual(2, count($files), '2 text files found when an "nomask" option is passed in.'); | ||||||
|  |  | ||||||
|  |     // Ensure that the directories in file_scan_ignore_directories are escaped using preg_quote. | ||||||
|  |     $conf['file_scan_ignore_directories'] = array('frontend.*'); | ||||||
|  |     $files = file_scan_directory($this->path, '/\.txt$/'); | ||||||
|  |     $this->assertEqual(3, count($files), '2 text files found when ignoring a directory that is not there.'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
							
								
								
									
										0
									
								
								modules/file/tests/fixtures/file_scan_ignore/a.txt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								modules/file/tests/fixtures/file_scan_ignore/a.txt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								modules/file/tests/fixtures/file_scan_ignore/frontend_framework/b.txt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								modules/file/tests/fixtures/file_scan_ignore/frontend_framework/b.txt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								modules/file/tests/fixtures/file_scan_ignore/frontend_framework/c.txt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								modules/file/tests/fixtures/file_scan_ignore/frontend_framework/c.txt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -7,7 +7,7 @@ files[] = filter.test | |||||||
| required = TRUE | required = TRUE | ||||||
| configure = admin/config/content/formats | configure = admin/config/content/formats | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ files[] = forum.test | |||||||
| configure = admin/structure/forum | configure = admin/structure/forum | ||||||
| stylesheets[all][] = forum.css | stylesheets[all][] = forum.css | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| files[] = help.test | files[] = help.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ dependencies[] = file | |||||||
| files[] = image.test | files[] = image.test | ||||||
| configure = admin/config/media/image-styles | configure = admin/config/media/image-styles | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| files[] = image_module_test.module | files[] = image_module_test.module | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| files[] = locale.test | files[] = locale.test | ||||||
| configure = admin/config/regional/language | configure = admin/config/regional/language | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ package = Testing | |||||||
| version = VERSION | version = VERSION | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| files[] = menu.test | files[] = menu.test | ||||||
| configure = admin/structure/menu | configure = admin/structure/menu | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ required = TRUE | |||||||
| configure = admin/structure/types | configure = admin/structure/types | ||||||
| stylesheets[all][] = node.css | stylesheets[all][] = node.css | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ package = Core | |||||||
| core = 7.x | core = 7.x | ||||||
| files[] = openid.test | files[] = openid.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| dependencies[] = openid | dependencies[] = openid | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ package = Core | |||||||
| version = VERSION | version = VERSION | ||||||
| core = 7.x | core = 7.x | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| files[] = path.test | files[] = path.test | ||||||
| configure = admin/config/search/path | configure = admin/config/search/path | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| files[] = php.test | files[] = php.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| files[] = poll.test | files[] = poll.test | ||||||
| stylesheets[all][] = poll.css | stylesheets[all][] = poll.css | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ configure = admin/config/people/profile | |||||||
| ; See user_system_info_alter(). | ; See user_system_info_alter(). | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| files[] = rdf.test | files[] = rdf.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| hidden = TRUE | hidden = TRUE | ||||||
| dependencies[] = blog | dependencies[] = blog | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ files[] = search.test | |||||||
| configure = admin/config/search/settings | configure = admin/config/search/settings | ||||||
| stylesheets[all][] = search.css | stylesheets[all][] = search.css | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ core = 7.x | |||||||
| files[] = shortcut.test | files[] = shortcut.test | ||||||
| configure = admin/config/user-interface/shortcut | configure = admin/config/user-interface/shortcut | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -3012,7 +3012,7 @@ class DrupalWebTestCase extends DrupalTestCase { | |||||||
|     if (!$message) { |     if (!$message) { | ||||||
|       $message = t('Raw "@raw" found', array('@raw' => $raw)); |       $message = t('Raw "@raw" found', array('@raw' => $raw)); | ||||||
|     } |     } | ||||||
|     return $this->assert(strpos($this->drupalGetContent(), $raw) !== FALSE, $message, $group); |     return $this->assert(strpos($this->drupalGetContent(), (string) $raw) !== FALSE, $message, $group); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
| @@ -3032,7 +3032,7 @@ class DrupalWebTestCase extends DrupalTestCase { | |||||||
|     if (!$message) { |     if (!$message) { | ||||||
|       $message = t('Raw "@raw" not found', array('@raw' => $raw)); |       $message = t('Raw "@raw" not found', array('@raw' => $raw)); | ||||||
|     } |     } | ||||||
|     return $this->assert(strpos($this->drupalGetContent(), $raw) === FALSE, $message, $group); |     return $this->assert(strpos($this->drupalGetContent(), (string) $raw) === FALSE, $message, $group); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|   | |||||||
| @@ -57,7 +57,7 @@ files[] = tests/upgrade/update.trigger.test | |||||||
| files[] = tests/upgrade/update.field.test | files[] = tests/upgrade/update.field.test | ||||||
| files[] = tests/upgrade/update.user.test | files[] = tests/upgrade/update.user.test | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ package = Testing | |||||||
| version = VERSION | version = VERSION | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ version = VERSION | |||||||
| core = 7.x | core = 7.x | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ package = Testing | |||||||
| version = VERSION | version = VERSION | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ package = Testing | |||||||
| version = VERSION | version = VERSION | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
| @@ -729,16 +729,12 @@ class BootstrapMiscTestCase extends DrupalUnitTestCase { | |||||||
|    * Tests that the drupal_check_memory_limit() function works as expected. |    * Tests that the drupal_check_memory_limit() function works as expected. | ||||||
|    */ |    */ | ||||||
|   function testCheckMemoryLimit() { |   function testCheckMemoryLimit() { | ||||||
|     $memory_limit = ini_get('memory_limit'); |  | ||||||
|     // Test that a very reasonable amount of memory is available. |     // Test that a very reasonable amount of memory is available. | ||||||
|     $this->assertTrue(drupal_check_memory_limit('30MB'), '30MB of memory tested available.'); |     $this->assertTrue(drupal_check_memory_limit('30MB'), '30MB of memory tested available.'); | ||||||
|  |  | ||||||
|     // Get the available memory and multiply it by two to make it unreasonably |     // Test an unlimited memory limit. | ||||||
|     // high. |  | ||||||
|     $twice_avail_memory = ($memory_limit * 2) . 'MB'; |  | ||||||
|  |  | ||||||
|     // The function should always return true if the memory limit is set to -1. |     // The function should always return true if the memory limit is set to -1. | ||||||
|     $this->assertTrue(drupal_check_memory_limit($twice_avail_memory, -1), 'drupal_check_memory_limit() returns TRUE when a limit of -1 (none) is supplied'); |     $this->assertTrue(drupal_check_memory_limit('9999999999YB', -1), 'drupal_check_memory_limit() returns TRUE when a limit of -1 (none) is supplied'); | ||||||
|  |  | ||||||
|     // Test that even though we have 30MB of memory available - the function |     // Test that even though we have 30MB of memory available - the function | ||||||
|     // returns FALSE when given an upper limit for how much memory can be used. |     // returns FALSE when given an upper limit for how much memory can be used. | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ stylesheets[all][] = common_test.css | |||||||
| stylesheets[print][] = common_test.print.css | stylesheets[print][] = common_test.print.css | ||||||
| hidden = TRUE | hidden = TRUE | ||||||
|  |  | ||||||
| ; Information added by Drupal.org packaging script on 2019-01-16 | ; Information added by Drupal.org packaging script on 2019-05-08 | ||||||
| version = "7.63" | version = "7.67" | ||||||
| project = "drupal" | project = "drupal" | ||||||
| datestamp = "1547681965" | datestamp = "1557336079" | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user