From 5ad1751e1129dda9518c110b9365d702826024b8 Mon Sep 17 00:00:00 2001 From: Alex Pott <alex.a.pott@googlemail.com> Date: Tue, 2 Apr 2024 13:53:30 +0100 Subject: [PATCH] Revert "Issue #3432882 by kim.pepper, Satane, andypost, alexpott, longwave: Removed deprecated code in File module (core/module/file)" This reverts commit 08c0efdfb38bd99c12f68132641007335e063988. --- core/.phpstan-baseline.php | 5 + .../Controller/CKEditor5ImageController.php | 2 +- core/modules/file/file.api.php | 4 +- core/modules/file/file.module | 374 +++++++++++++++++- core/modules/file/src/Element/ManagedFile.php | 10 +- .../src/Plugin/Field/FieldType/FileItem.php | 3 +- .../rest/resource/FileUploadResource.php | 63 ++- .../file/src/Upload/FileUploadHandler.php | 139 ++++++- .../file/src/Upload/FormUploadedFile.php | 42 ++ .../file/src/Upload/UploadedFileInterface.php | 44 +++ .../file/src/Validation/FileValidator.php | 48 ++- .../src/Functional/SaveUploadFormTest.php | 2 +- .../tests/src/Kernel/FileSaveUploadTest.php | 70 ++++ .../src/Kernel/FileUploadHandlerTest.php | 41 +- .../tests/src/Kernel/LegacyFileModuleTest.php | 50 +++ .../tests/src/Kernel/LegacyFileThemeTest.php | 48 +++ .../tests/src/Kernel/LegacyValidateTest.php | 59 +++ .../tests/src/Kernel/LegacyValidatorTest.php | 255 ++++++++++++ .../Upload/LegacyFileUploadHandlerTest.php | 39 ++ .../Kernel/Validation/FileValidatorTest.php | 16 +- 20 files changed, 1281 insertions(+), 33 deletions(-) create mode 100644 core/modules/file/tests/src/Kernel/FileSaveUploadTest.php create mode 100644 core/modules/file/tests/src/Kernel/LegacyFileModuleTest.php create mode 100644 core/modules/file/tests/src/Kernel/LegacyFileThemeTest.php create mode 100644 core/modules/file/tests/src/Kernel/LegacyValidateTest.php create mode 100644 core/modules/file/tests/src/Kernel/LegacyValidatorTest.php create mode 100644 core/modules/file/tests/src/Kernel/Upload/LegacyFileUploadHandlerTest.php diff --git a/core/.phpstan-baseline.php b/core/.phpstan-baseline.php index 65d4b5c62420..96be90402611 100644 --- a/core/.phpstan-baseline.php +++ b/core/.phpstan-baseline.php @@ -826,6 +826,11 @@ 'count' => 1, 'path' => __DIR__ . '/modules/file/file.module', ]; +$ignoreErrors[] = [ + 'message' => '#^Variable \\$message might not be defined\\.$#', + 'count' => 1, + 'path' => __DIR__ . '/modules/file/file.module', +]; $ignoreErrors[] = [ 'message' => '#^Variable \\$rows in empty\\(\\) always exists and is not falsy\\.$#', 'count' => 1, diff --git a/core/modules/ckeditor5/src/Controller/CKEditor5ImageController.php b/core/modules/ckeditor5/src/Controller/CKEditor5ImageController.php index 733d60866b3c..43f4d9b44cfc 100644 --- a/core/modules/ckeditor5/src/Controller/CKEditor5ImageController.php +++ b/core/modules/ckeditor5/src/Controller/CKEditor5ImageController.php @@ -149,7 +149,7 @@ public function upload(Request $request): Response { try { $uploadedFile = new FormUploadedFile($upload); - $uploadResult = $this->fileUploadHandler->handleFileUpload($uploadedFile, $validators, $destination, FileSystemInterface::EXISTS_RENAME); + $uploadResult = $this->fileUploadHandler->handleFileUpload($uploadedFile, $validators, $destination, FileSystemInterface::EXISTS_RENAME, FALSE); if ($uploadResult->hasViolations()) { throw new UnprocessableEntityHttpException((string) $uploadResult->getViolations()); } diff --git a/core/modules/file/file.api.php b/core/modules/file/file.api.php index f0e2eff6430f..fa86532f7c48 100644 --- a/core/modules/file/file.api.php +++ b/core/modules/file/file.api.php @@ -27,8 +27,8 @@ * '#type' => 'file', * '#title' => $this->t('Upload file'), * '#upload_validators' => [ - * 'FileExtension' => [ - * 'extensions' => 'png gif jpg', + * 'file_validate_extensions' => [ + * 'png gif jpg', * ], * ], * ]; diff --git a/core/modules/file/file.module b/core/modules/file/file.module index 184221455d32..525c8b15d5c4 100644 --- a/core/modules/file/file.module +++ b/core/modules/file/file.module @@ -68,6 +68,289 @@ function file_field_widget_info_alter(array &$info) { $info['uri']['field_types'][] = 'file_uri'; } +/** + * Checks that a file meets the criteria specified by the validators. + * + * After executing the validator callbacks specified hook_file_validate() will + * also be called to allow other modules to report errors about the file. + * + * @param \Drupal\file\FileInterface $file + * A file entity. + * @param array $validators + * (optional) An associative array of callback functions used to validate + * the file. The keys are function names and the values arrays of callback + * parameters which will be passed in after the file entity. The functions + * should return an array of error messages; an empty array indicates that + * the file passed validation. The callback functions will be called in the + * order specified in the array, then the hook hook_file_validate() + * will be invoked so other modules can validate the new file. + * + * @return array + * An array containing validation error messages. + * + * @deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the + * 'file.validator' service instead. + * + * @see https://www.drupal.org/node/3363700 + */ +function file_validate(FileInterface $file, $validators = []) { + @trigger_error(__FUNCTION__ . '() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $violations = \Drupal::service('file.validator')->validate($file, $validators); + $errors = []; + foreach ($violations as $violation) { + $errors[] = $violation->getMessage(); + } + return $errors; +} + +/** + * Checks for files with names longer than can be stored in the database. + * + * @param \Drupal\file\FileInterface $file + * A file entity. + * + * @return array + * An empty array if the file name length is smaller than the limit or an + * array containing an error message if it's not or is empty. + * + * @deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the + * 'file.validator' service instead. + * + * @see https://www.drupal.org/node/3363700 + */ +function file_validate_name_length(FileInterface $file) { + @trigger_error(__FUNCTION__ . '() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $errors = []; + + if (!$file->getFilename()) { + $errors[] = t("The file's name is empty. Enter a name for the file."); + } + if (strlen($file->getFilename()) > 240) { + $errors[] = t("The file's name exceeds the 240 characters limit. Rename the file and try again."); + } + return $errors; +} + +/** + * Checks that the filename ends with an allowed extension. + * + * @param \Drupal\file\FileInterface $file + * A file entity. + * @param string $extensions + * A string with a space separated list of allowed extensions. + * + * @return array + * An empty array if the file extension is allowed or an array containing an + * error message if it's not. + * + * @deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the + * 'file.validator' service instead. + * + * @see https://www.drupal.org/node/3363700 + * @see hook_file_validate() + */ +function file_validate_extensions(FileInterface $file, $extensions) { + @trigger_error(__FUNCTION__ . '() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $errors = []; + + $regex = '/\.(' . preg_replace('/ +/', '|', preg_quote($extensions)) . ')$/i'; + // Filename may differ from the basename, for instance in case files migrated + // from D7 file entities. Because of that new files are saved temporarily with + // a generated file name, without the original extension, we will use the + // generated filename property for extension validation only in case of + // temporary files; and use the file system file name in case of permanent + // files. + $subject = $file->isTemporary() ? $file->getFilename() : $file->getFileUri(); + if (!preg_match($regex, $subject)) { + $errors[] = t('Only files with the following extensions are allowed: %files-allowed.', ['%files-allowed' => $extensions]); + } + return $errors; +} + +/** + * Checks that the file's size is below certain limits. + * + * @param \Drupal\file\FileInterface $file + * A file entity. + * @param int $file_limit + * (optional) The maximum file size in bytes. Zero (the default) indicates + * that no limit should be enforced. + * @param int $user_limit + * (optional) The maximum number of bytes the user is allowed. Zero (the + * default) indicates that no limit should be enforced. + * + * @return array + * An empty array if the file size is below limits or an array containing an + * error message if it's not. + * + * @deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the + * 'file.validator' service instead. + * + * @see https://www.drupal.org/node/3363700 + * @see hook_file_validate() + */ +function file_validate_size(FileInterface $file, $file_limit = 0, $user_limit = 0) { + @trigger_error(__FUNCTION__ . '() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $user = \Drupal::currentUser(); + $errors = []; + + if ($file_limit && $file->getSize() > $file_limit) { + $errors[] = t('The file is %filesize exceeding the maximum file size of %maxsize.', [ + '%filesize' => ByteSizeMarkup::create($file->getSize()), + '%maxsize' => ByteSizeMarkup::create($file_limit), + ]); + } + + // Save a query by only calling spaceUsed() when a limit is provided. + if ($user_limit && (\Drupal::entityTypeManager()->getStorage('file')->spaceUsed($user->id()) + $file->getSize()) > $user_limit) { + $errors[] = t('The file is %filesize which would exceed your disk quota of %quota.', [ + '%filesize' => ByteSizeMarkup::create($file->getSize()), + '%quota' => ByteSizeMarkup::create($user_limit), + ]); + } + + return $errors; +} + +/** + * Checks that the file is recognized as a valid image. + * + * @param \Drupal\file\FileInterface $file + * A file entity. + * + * @return array + * An empty array if the file is a valid image or an array containing an error + * message if it's not. + * + * @deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the + * 'file.validator' service instead. + * + * @see https://www.drupal.org/node/3363700 + * @see hook_file_validate() + */ +function file_validate_is_image(FileInterface $file) { + @trigger_error(__FUNCTION__ . '() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $errors = []; + + $image_factory = \Drupal::service('image.factory'); + $image = $image_factory->get($file->getFileUri()); + if (!$image->isValid()) { + $supported_extensions = $image_factory->getSupportedExtensions(); + $errors[] = t('The image file is invalid or the image type is not allowed. Allowed types: %types', ['%types' => implode(', ', $supported_extensions)]); + } + + return $errors; +} + +/** + * Verifies that image dimensions are within the specified maximum and minimum. + * + * Non-image files will be ignored. If an image toolkit is available the image + * will be scaled to fit within the desired maximum dimensions. + * + * @param \Drupal\file\FileInterface $file + * A file entity. This function may resize the file affecting its size. + * @param string|int $maximum_dimensions + * (optional) A string in the form WIDTHxHEIGHT; for example, '640x480' or + * '85x85'. If an image toolkit is installed, the image will be resized down + * to these dimensions. A value of zero (the default) indicates no restriction + * on size, so no resizing will be attempted. + * @param string|int $minimum_dimensions + * (optional) A string in the form WIDTHxHEIGHT. This will check that the + * image meets a minimum size. A value of zero (the default) indicates that + * there is no restriction on size. + * + * @return array + * An empty array if the file meets the specified dimensions, was resized + * successfully to meet those requirements or is not an image. If the image + * does not meet the requirements or an attempt to resize it fails, an array + * containing the error message will be returned. + * + * @deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the + * 'file.validator' service instead. + * + * @see https://www.drupal.org/node/3363700 + * @see hook_file_validate() + */ +function file_validate_image_resolution(FileInterface $file, $maximum_dimensions = 0, $minimum_dimensions = 0) { + @trigger_error(__FUNCTION__ . '() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $errors = []; + + // Check first that the file is an image. + $image_factory = \Drupal::service('image.factory'); + $image = $image_factory->get($file->getFileUri()); + + if ($image->isValid()) { + $scaling = FALSE; + if ($maximum_dimensions) { + // Check that it is smaller than the given dimensions. + [$width, $height] = explode('x', $maximum_dimensions); + if ($image->getWidth() > $width || $image->getHeight() > $height) { + // Try to resize the image to fit the dimensions. + if ($image->scale($width, $height)) { + $scaling = TRUE; + $image->save(); + // Update the file size now that the image has been resized. + $file->setSize($image->getFileSize()); + if (!empty($width) && !empty($height)) { + $message = t('The image was resized to fit within the maximum allowed dimensions of %dimensions pixels. The new dimensions of the resized image are %new_widthx%new_height pixels.', + [ + '%dimensions' => $maximum_dimensions, + '%new_width' => $image->getWidth(), + '%new_height' => $image->getHeight(), + ]); + } + elseif (empty($width)) { + $message = t('The image was resized to fit within the maximum allowed height of %height pixels. The new dimensions of the resized image are %new_widthx%new_height pixels.', + [ + '%height' => $height, + '%new_width' => $image->getWidth(), + '%new_height' => $image->getHeight(), + ]); + } + elseif (empty($height)) { + $message = t('The image was resized to fit within the maximum allowed width of %width pixels. The new dimensions of the resized image are %new_widthx%new_height pixels.', + [ + '%width' => $width, + '%new_width' => $image->getWidth(), + '%new_height' => $image->getHeight(), + ]); + } + \Drupal::messenger()->addStatus($message); + } + else { + $errors[] = t('The image exceeds the maximum allowed dimensions and an attempt to resize it failed.'); + } + } + } + + if ($minimum_dimensions) { + // Check that it is larger than the given dimensions. + [$width, $height] = explode('x', $minimum_dimensions); + if ($image->getWidth() < $width || $image->getHeight() < $height) { + if ($scaling) { + $errors[] = t('The resized image is too small. The minimum dimensions are %dimensions pixels and after resizing, the image size will be %widthx%height pixels.', + [ + '%dimensions' => $minimum_dimensions, + '%width' => $image->getWidth(), + '%height' => $image->getHeight(), + ]); + } + else { + $errors[] = t('The image is too small. The minimum dimensions are %dimensions pixels and the image size is %widthx%height pixels.', + [ + '%dimensions' => $minimum_dimensions, + '%width' => $image->getWidth(), + '%height' => $image->getHeight(), + ]); + } + } + } + } + + return $errors; +} + /** * Examines a file entity and returns appropriate content headers for download. * @@ -306,7 +589,7 @@ function _file_save_upload_from_form(array $element, FormStateInterface $form_st * If the array is empty, it will be set up to call file_validate_extensions() * with a safe list of extensions, as follows: "jpg jpeg gif png txt doc * xls pdf ppt pps odt ods odp". To allow all extensions, you must explicitly - * set this array to ['FileExtension' => []]. (Beware: this is not + * set this array to ['file_validate_extensions' => '']. (Beware: this is not * safe and should only be allowed for trusted users, if at all.) * @param string|false $destination * (optional) A string containing the URI that the file should be copied to. @@ -378,7 +661,7 @@ function file_save_upload($form_field_name, $validators = [], $destination = FAL continue; } $form_uploaded_file = new FormUploadedFile($uploaded_file); - $result = $file_upload_handler->handleFileUpload($form_uploaded_file, $validators, $destination, $replace); + $result = $file_upload_handler->handleFileUpload($form_uploaded_file, $validators, $destination, $replace, FALSE); if ($result->hasViolations()) { $errors = []; foreach ($result->getViolations() as $violation) { @@ -442,6 +725,33 @@ function file_save_upload($form_field_name, $validators = [], $destination = FAL return isset($delta) ? $files[$delta] : $files; } +/** + * Determines the preferred upload progress implementation. + * + * @return string|false + * A string indicating which upload progress system is available. Either "apc" + * or "uploadprogress". If neither are available, returns FALSE. + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * extension_loaded('uploadprogress') instead. + * + * @see https://www.drupal.org/node/3397577 + */ +function file_progress_implementation() { + @trigger_error(__FUNCTION__ . '() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use extension_loaded(\'uploadprogress\') instead. See https://www.drupal.org/node/3397577', E_USER_DEPRECATED); + static $implementation; + if (!isset($implementation)) { + $implementation = FALSE; + + // We prefer the PECL extension uploadprogress because it supports multiple + // simultaneous uploads. APCu only supports one at a time. + if (extension_loaded('uploadprogress')) { + $implementation = 'uploadprogress'; + } + } + return $implementation; +} + /** * Implements hook_ENTITY_TYPE_predelete() for file entities. */ @@ -929,18 +1239,32 @@ function template_preprocess_file_upload_help(&$variables) { $descriptions[] = \Drupal::translation()->formatPlural($cardinality, 'One file only.', 'Maximum @count files.'); } } - + if (isset($upload_validators['file_validate_size'])) { + @trigger_error('\'file_validate_size\' is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'FileSizeLimit\' constraint instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $descriptions[] = t('@size limit.', ['@size' => ByteSizeMarkup::create($upload_validators['file_validate_size'][0])]); + } if (isset($upload_validators['FileSizeLimit'])) { $descriptions[] = t('@size limit.', ['@size' => ByteSizeMarkup::create($upload_validators['FileSizeLimit']['fileLimit'])]); } + if (isset($upload_validators['file_validate_extensions'])) { + @trigger_error('\'file_validate_extensions\' is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'FileExtension\' constraint instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $descriptions[] = t('Allowed types: @extensions.', ['@extensions' => $upload_validators['file_validate_extensions'][0]]); + } if (isset($upload_validators['FileExtension'])) { $descriptions[] = t('Allowed types: @extensions.', ['@extensions' => $upload_validators['FileExtension']['extensions']]); } - if (isset($upload_validators['FileImageDimensions'])) { - $max = $upload_validators['FileImageDimensions']['maxDimensions']; - $min = $upload_validators['FileImageDimensions']['minDimensions']; + if (isset($upload_validators['file_validate_image_resolution']) || isset($upload_validators['FileImageDimensions'])) { + if (isset($upload_validators['file_validate_image_resolution'])) { + @trigger_error('\'file_validate_image_resolution\' is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'FileImageDimensions\' constraint instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $max = $upload_validators['file_validate_image_resolution'][0]; + $min = $upload_validators['file_validate_image_resolution'][1]; + } + else { + $max = $upload_validators['FileImageDimensions']['maxDimensions']; + $min = $upload_validators['FileImageDimensions']['minDimensions']; + } if ($min && $max && $min == $max) { $descriptions[] = t('Images must be exactly <strong>@size</strong> pixels.', ['@size' => $max]); } @@ -961,6 +1285,44 @@ function template_preprocess_file_upload_help(&$variables) { $variables['descriptions'] = $descriptions; } +/** + * Gets a class for the icon for a MIME type. + * + * @param string $mime_type + * A MIME type. + * + * @return string + * A class associated with the file. + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\IconMimeTypes::getIconClass() instead. + * + * @see https://www.drupal.org/node/3411269 + */ +function file_icon_class($mime_type) { + @trigger_error(__FUNCTION__ . '() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use \Drupal\file\IconMimeTypes::getIconClass() instead. See https://www.drupal.org/node/3411269', E_USER_DEPRECATED); + return IconMimeTypes::getIconClass($mime_type); +} + +/** + * Determines the generic icon MIME package based on a file's MIME type. + * + * @param string $mime_type + * A MIME type. + * + * @return string|false + * The generic icon MIME package expected for this file. + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\IconMimeTypes::getGenericMimeType() instead. + * + * @see https://www.drupal.org/node/3411269 + */ +function file_icon_map($mime_type) { + @trigger_error(__FUNCTION__ . '() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use \Drupal\file\IconMimeTypes::getGenericMimeType() instead. See https://www.drupal.org/node/3411269', E_USER_DEPRECATED); + return IconMimeTypes::getGenericMimeType($mime_type); +} + /** * Retrieves a list of references to a file. * diff --git a/core/modules/file/src/Element/ManagedFile.php b/core/modules/file/src/Element/ManagedFile.php index 2d076bfa6819..5cc5c43805a7 100644 --- a/core/modules/file/src/Element/ManagedFile.php +++ b/core/modules/file/src/Element/ManagedFile.php @@ -358,8 +358,14 @@ public static function processManagedFile(&$element, FormStateInterface $form_st } // Add the extension list to the page as JavaScript settings. - if (isset($element['#upload_validators']['FileExtension']['extensions'])) { - $allowed_extensions = $element['#upload_validators']['FileExtension']['extensions']; + if (isset($element['#upload_validators']['file_validate_extensions'][0]) || isset($element['#upload_validators']['FileExtension']['extensions'])) { + if (isset($element['#upload_validators']['file_validate_extensions'][0])) { + @trigger_error('\'file_validate_extensions\' is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'FileExtension\' constraint instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $allowed_extensions = $element['#upload_validators']['file_validate_extensions'][0]; + } + else { + $allowed_extensions = $element['#upload_validators']['FileExtension']['extensions']; + } $extension_list = implode(',', array_filter(explode(' ', $allowed_extensions))); $element['upload']['#attached']['drupalSettings']['file']['elements']['#' . $id] = $extension_list; } diff --git a/core/modules/file/src/Plugin/Field/FieldType/FileItem.php b/core/modules/file/src/Plugin/Field/FieldType/FileItem.php index 604960c0c263..d9fe46ec600c 100644 --- a/core/modules/file/src/Plugin/Field/FieldType/FileItem.php +++ b/core/modules/file/src/Plugin/Field/FieldType/FileItem.php @@ -238,8 +238,7 @@ public static function validateDirectory($element, FormStateInterface $form_stat * * This doubles as a convenience clean-up function and a validation routine. * Commas are allowed by the end-user, but ultimately the value will be stored - * as a space-separated list for compatibility with the 'FileExtension' - * constraint. + * as a space-separated list for compatibility with file_validate_extensions(). */ public static function validateExtensions($element, FormStateInterface $form_state) { if (!empty($element['#value'])) { diff --git a/core/modules/file/src/Plugin/rest/resource/FileUploadResource.php b/core/modules/file/src/Plugin/rest/resource/FileUploadResource.php index 2d010922c460..1206acc77a29 100644 --- a/core/modules/file/src/Plugin/rest/resource/FileUploadResource.php +++ b/core/modules/file/src/Plugin/rest/resource/FileUploadResource.php @@ -63,6 +63,32 @@ class FileUploadResource extends ResourceBase { validate as resourceValidate; } + /** + * The regex used to extract the filename from the content disposition header. + * + * @var string + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\Upload\ContentDispositionFilenameParser::REQUEST_HEADER_FILENAME_REGEX + * instead. + * + * @see https://www.drupal.org/node/3380380 + */ + const REQUEST_HEADER_FILENAME_REGEX = '@\bfilename(?<star>\*?)=\"(?<filename>.+)\"@'; + + /** + * The amount of bytes to read in each iteration when streaming file data. + * + * @var int + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\Upload\InputStreamFileWriterInterface::DEFAULT_BYTES_TO_READ + * instead. + * + * @see https://www.drupal.org/node/3380607 + */ + const BYTES_TO_READ = 8192; + /** * The file system service. * @@ -167,12 +193,12 @@ class FileUploadResource extends ResourceBase { * The system file configuration. * @param \Symfony\Contracts\EventDispatcher\EventDispatcherInterface $event_dispatcher * The event dispatcher service. - * @param \Drupal\file\Validation\FileValidatorInterface $file_validator + * @param \Drupal\file\Validation\FileValidatorInterface|null $file_validator * The file validator service. - * @param \Drupal\file\Upload\InputStreamFileWriterInterface $input_stream_file_writer + * @param \Drupal\file\Upload\InputStreamFileWriterInterface|null $input_stream_file_writer * The input stream file writer. */ - public function __construct(array $configuration, $plugin_id, $plugin_definition, $serializer_formats, LoggerInterface $logger, FileSystemInterface $file_system, EntityTypeManagerInterface $entity_type_manager, EntityFieldManagerInterface $entity_field_manager, AccountInterface $current_user, $mime_type_guesser, Token $token, LockBackendInterface $lock, Config $system_file_config, EventDispatcherInterface $event_dispatcher, FileValidatorInterface $file_validator, InputStreamFileWriterInterface $input_stream_file_writer) { + public function __construct(array $configuration, $plugin_id, $plugin_definition, $serializer_formats, LoggerInterface $logger, FileSystemInterface $file_system, EntityTypeManagerInterface $entity_type_manager, EntityFieldManagerInterface $entity_field_manager, AccountInterface $current_user, $mime_type_guesser, Token $token, LockBackendInterface $lock, Config $system_file_config, EventDispatcherInterface $event_dispatcher, FileValidatorInterface $file_validator = NULL, InputStreamFileWriterInterface $input_stream_file_writer = NULL) { parent::__construct($configuration, $plugin_id, $plugin_definition, $serializer_formats, $logger); $this->fileSystem = $file_system; $this->entityTypeManager = $entity_type_manager; @@ -183,7 +209,15 @@ public function __construct(array $configuration, $plugin_id, $plugin_definition $this->lock = $lock; $this->systemFileConfig = $system_file_config; $this->eventDispatcher = $event_dispatcher; + if (!$file_validator) { + @trigger_error('Calling ' . __METHOD__ . '() without the $file_validator argument is deprecated in drupal:10.2.0 and is required in drupal:11.0.0. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $file_validator = \Drupal::service('file.validator'); + } $this->fileValidator = $file_validator; + if (!$input_stream_file_writer) { + @trigger_error('Calling ' . __METHOD__ . '() without the $input_stream_file_writer argument is deprecated in drupal:10.3.0 and is required in drupal:11.0.0. See https://www.drupal.org/node/3380607', E_USER_DEPRECATED); + $input_stream_file_writer = \Drupal::service('file.input_stream_file_writer'); + } $this->inputStreamFileWriter = $input_stream_file_writer; } @@ -362,6 +396,29 @@ protected function streamUploadData(): string { return $temp_file_path; } + /** + * Validates and extracts the filename from the Content-Disposition header. + * + * @param \Symfony\Component\HttpFoundation\Request $request + * The request object. + * + * @return string + * The filename extracted from the header. + * + * @throws \Symfony\Component\HttpKernel\Exception\BadRequestHttpException + * Thrown when the 'Content-Disposition' request header is invalid. + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\Upload\ContentDispositionFilenameParser::parseFilename() + * instead. + * + * @see https://www.drupal.org/node/3380380 + */ + protected function validateAndParseContentDispositionHeader(Request $request) { + @trigger_error('Calling ' . __METHOD__ . '() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use \Drupal\file\Upload\ContentDispositionFilenameParser::parseFilename() instead. See https://www.drupal.org/node/3380380', E_USER_DEPRECATED); + return ContentDispositionFilenameParser::parseFilename($request); + } + /** * Validates and loads a field definition instance. * diff --git a/core/modules/file/src/Upload/FileUploadHandler.php b/core/modules/file/src/Upload/FileUploadHandler.php index 402017e75926..6313960fdfac 100644 --- a/core/modules/file/src/Upload/FileUploadHandler.php +++ b/core/modules/file/src/Upload/FileUploadHandler.php @@ -14,9 +14,18 @@ use Drupal\Core\Session\AccountInterface; use Drupal\Core\StreamWrapper\StreamWrapperManagerInterface; use Drupal\file\Entity\File; +use Drupal\file\FileInterface; use Drupal\file\FileRepositoryInterface; use Drupal\file\Validation\FileValidatorInterface; use Symfony\Component\EventDispatcher\EventDispatcherInterface; +use Symfony\Component\HttpFoundation\File\Exception\CannotWriteFileException; +use Symfony\Component\HttpFoundation\File\Exception\ExtensionFileException; +use Symfony\Component\HttpFoundation\File\Exception\FileException; +use Symfony\Component\HttpFoundation\File\Exception\FormSizeFileException; +use Symfony\Component\HttpFoundation\File\Exception\IniSizeFileException; +use Symfony\Component\HttpFoundation\File\Exception\NoFileException; +use Symfony\Component\HttpFoundation\File\Exception\NoTmpDirFileException; +use Symfony\Component\HttpFoundation\File\Exception\PartialFileException; use Symfony\Component\HttpFoundation\RequestStack; use Symfony\Component\Mime\MimeTypeGuesserInterface; @@ -110,11 +119,11 @@ class FileUploadHandler { * The current user. * @param \Symfony\Component\HttpFoundation\RequestStack $requestStack * The request stack. - * @param \Drupal\file\FileRepositoryInterface $fileRepository + * @param \Drupal\file\FileRepositoryInterface|null $fileRepository * The file repository. - * @param \Drupal\file\Validation\FileValidatorInterface $file_validator + * @param \Drupal\file\Validation\FileValidatorInterface|null $file_validator * The file validator. - * @param \Drupal\Core\Lock\LockBackendInterface $lock + * @param \Drupal\Core\Lock\LockBackendInterface|null $lock * The lock. */ public function __construct( @@ -125,9 +134,9 @@ public function __construct( MimeTypeGuesserInterface $mimeTypeGuesser, AccountInterface $currentUser, RequestStack $requestStack, - FileRepositoryInterface $fileRepository, - FileValidatorInterface $file_validator, - protected LockBackendInterface $lock, + FileRepositoryInterface $fileRepository = NULL, + FileValidatorInterface $file_validator = NULL, + protected ?LockBackendInterface $lock = NULL, ) { $this->fileSystem = $fileSystem; $this->entityTypeManager = $entityTypeManager; @@ -136,8 +145,20 @@ public function __construct( $this->mimeTypeGuesser = $mimeTypeGuesser; $this->currentUser = $currentUser; $this->requestStack = $requestStack; + if ($fileRepository === NULL) { + @trigger_error('Calling ' . __METHOD__ . ' without the $fileRepository argument is deprecated in drupal:10.1.0 and will be required in drupal:11.0.0. See https://www.drupal.org/node/3346839', E_USER_DEPRECATED); + $fileRepository = \Drupal::service('file.repository'); + } $this->fileRepository = $fileRepository; + if (!$file_validator) { + @trigger_error('Calling ' . __METHOD__ . '() without the $file_validator argument is deprecated in drupal:10.2.0 and is required in drupal:11.0.0. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + $file_validator = \Drupal::service('file.validator'); + } $this->fileValidator = $file_validator; + if (!$this->lock) { + @trigger_error('Calling ' . __METHOD__ . '() without the $lock argument is deprecated in drupal:10.3.0 and is required in drupal:11.0.0. See https://www.drupal.org/node/3389017', E_USER_DEPRECATED); + $this->lock = \Drupal::service('lock'); + } } /** @@ -155,6 +176,8 @@ public function __construct( * - FileSystemInterface::EXISTS_RENAME - Append _{incrementing number} * until the filename is unique. * - FileSystemInterface::EXISTS_ERROR - Throw an exception. + * @param bool $throw + * (optional) Whether to throw an exception if the file is invalid. * * @return \Drupal\file\Upload\FileUploadResult * The created file entity. @@ -170,8 +193,46 @@ public function __construct( * @throws \Drupal\Core\Lock\LockAcquiringException * Thrown when a lock cannot be acquired. */ - public function handleFileUpload(UploadedFileInterface $uploadedFile, array $validators = [], string $destination = 'temporary://', int $replace = FileSystemInterface::EXISTS_REPLACE): FileUploadResult { + public function handleFileUpload(UploadedFileInterface $uploadedFile, array $validators = [], string $destination = 'temporary://', int $replace = FileSystemInterface::EXISTS_REPLACE, bool $throw = TRUE): FileUploadResult { $originalName = $uploadedFile->getClientOriginalName(); + // @phpstan-ignore-next-line + if ($throw && !$uploadedFile->isValid()) { + @trigger_error('Calling ' . __METHOD__ . '() with the $throw argument as TRUE is deprecated in drupal:10.3.0 and will be removed in drupal:11.0.0. Use \Drupal\file\Upload\FileUploadResult::getViolations() instead. See https://www.drupal.org/node/3375456', E_USER_DEPRECATED); + // @phpstan-ignore-next-line + switch ($uploadedFile->getError()) { + case \UPLOAD_ERR_INI_SIZE: + // @phpstan-ignore-next-line + throw new IniSizeFileException($uploadedFile->getErrorMessage()); + + case \UPLOAD_ERR_FORM_SIZE: + // @phpstan-ignore-next-line + throw new FormSizeFileException($uploadedFile->getErrorMessage()); + + case \UPLOAD_ERR_PARTIAL: + // @phpstan-ignore-next-line + throw new PartialFileException($uploadedFile->getErrorMessage()); + + case \UPLOAD_ERR_NO_FILE: + // @phpstan-ignore-next-line + throw new NoFileException($uploadedFile->getErrorMessage()); + + case \UPLOAD_ERR_CANT_WRITE: + // @phpstan-ignore-next-line + throw new CannotWriteFileException($uploadedFile->getErrorMessage()); + + case \UPLOAD_ERR_NO_TMP_DIR: + // @phpstan-ignore-next-line + throw new NoTmpDirFileException($uploadedFile->getErrorMessage()); + + case \UPLOAD_ERR_EXTENSION: + // @phpstan-ignore-next-line + throw new ExtensionFileException($uploadedFile->getErrorMessage()); + + } + // @phpstan-ignore-next-line + throw new FileException($uploadedFile->getErrorMessage()); + } + $extensions = $this->handleExtensionValidation($validators); // Assert that the destination contains a valid stream. @@ -234,6 +295,20 @@ public function handleFileUpload(UploadedFileInterface $uploadedFile, array $val return $result; } + if ($throw) { + $errors = []; + foreach ($violations as $violation) { + $errors[] = $violation->getMessage(); + } + if (!empty($errors)) { + throw new FileValidationException( + 'File validation failed', + $filename, + $errors + ); + } + } + $file->setFileUri($destinationFilename); if (!$this->moveUploadedFile($uploadedFile, $file->getFileUri())) { @@ -268,6 +343,18 @@ public function handleFileUpload(UploadedFileInterface $uploadedFile, array $val // We can now validate the file object itself before it's saved. $violations = $file->validate(); + if ($throw) { + foreach ($violations as $violation) { + $errors[] = $violation->getMessage(); + } + if (!empty($errors)) { + throw new FileValidationException( + 'File validation failed', + $filename, + $errors + ); + } + } if (count($violations) > 0) { $result->addViolations($violations); @@ -329,6 +416,25 @@ protected function moveUploadedFile(UploadedFileInterface $uploadedFile, string * The space delimited list of allowed file extensions. */ protected function handleExtensionValidation(array &$validators): string { + // Handle legacy extension validation. + if (isset($validators['file_validate_extensions'])) { + @trigger_error( + '\'file_validate_extensions\' is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'FileExtension\' constraint instead. See https://www.drupal.org/node/3363700', + E_USER_DEPRECATED + ); + // Empty string means all extensions are allowed so we should remove the + // validator. + if (\is_string($validators['file_validate_extensions']) && empty($validators['file_validate_extensions'])) { + unset($validators['file_validate_extensions']); + return ''; + } + // The deprecated 'file_validate_extensions' has configuration, so that + // should be used. + $validators['FileExtension']['extensions'] = $validators['file_validate_extensions'][0]; + unset($validators['file_validate_extensions']); + return $validators['FileExtension']['extensions']; + } + // No validator was provided, so add one using the default list. // Build a default non-munged safe list for // \Drupal\system\EventSubscriber\SecurityFileUploadEventSubscriber::sanitizeName(). @@ -349,6 +455,25 @@ protected function handleExtensionValidation(array &$validators): string { return $validators['FileExtension']['extensions']; } + /** + * Loads the first File entity found with the specified URI. + * + * @param string $uri + * The file URI. + * + * @return \Drupal\file\FileInterface|null + * The first file with the matched URI if found, NULL otherwise. + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. + * Use \Drupal\file\FileRepositoryInterface::loadByUri(). + * + * @see https://www.drupal.org/node/3409326 + */ + protected function loadByUri(string $uri): ?FileInterface { + @trigger_error('FileUploadHandler::loadByUri() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use \Drupal\file\FileRepositoryInterface::loadByUri(). See https://www.drupal.org/node/3409326', E_USER_DEPRECATED); + return $this->fileRepository->loadByUri($uri); + } + /** * Generates a lock ID based on the file URI. */ diff --git a/core/modules/file/src/Upload/FormUploadedFile.php b/core/modules/file/src/Upload/FormUploadedFile.php index b26a7e01eb20..7ae45b7721ee 100644 --- a/core/modules/file/src/Upload/FormUploadedFile.php +++ b/core/modules/file/src/Upload/FormUploadedFile.php @@ -33,6 +33,48 @@ public function getClientOriginalName(): string { return $this->uploadedFile->getClientOriginalName(); } + /** + * {@inheritdoc} + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\Validation\UploadedFileValidatorInterface::validate() + * instead. + * + * @see https://www.drupal.org/node/3375456 + */ + public function isValid(): bool { + @trigger_error(__METHOD__ . '() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use \Drupal\file\Validation\UploadedFileValidatorInterface::validate() instead. See https://www.drupal.org/node/3375456', E_USER_DEPRECATED); + return $this->uploadedFile->isValid(); + } + + /** + * {@inheritdoc} + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\Validation\UploadedFileValidatorInterface::validate() + * instead. + * + * @see https://www.drupal.org/node/3375456 + */ + public function getErrorMessage(): string { + @trigger_error(__METHOD__ . '() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use \Drupal\file\Validation\UploadedFileValidatorInterface::validate() instead. See https://www.drupal.org/node/3375456', E_USER_DEPRECATED); + return $this->uploadedFile->getErrorMessage(); + } + + /** + * {@inheritdoc} + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\Validation\UploadedFileValidatorInterface::validate() + * instead. + * + * @see https://www.drupal.org/node/3375456 + */ + public function getError(): int { + @trigger_error(__METHOD__ . '() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use \Drupal\file\Validation\UploadedFileValidatorInterface::validate() instead. See https://www.drupal.org/node/3375456', E_USER_DEPRECATED); + return $this->uploadedFile->getError(); + } + /** * {@inheritdoc} */ diff --git a/core/modules/file/src/Upload/UploadedFileInterface.php b/core/modules/file/src/Upload/UploadedFileInterface.php index dabb67970b50..c601e0e9a604 100644 --- a/core/modules/file/src/Upload/UploadedFileInterface.php +++ b/core/modules/file/src/Upload/UploadedFileInterface.php @@ -18,6 +18,50 @@ interface UploadedFileInterface { */ public function getClientOriginalName(): string; + /** + * Returns whether the file was uploaded successfully. + * + * @return bool + * TRUE if the file has been uploaded with HTTP and no error occurred. + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\Validation\UploadedFileValidatorInterface::validate() + * instead. + * @see https://www.drupal.org/node/3375456 + */ + public function isValid(): bool; + + /** + * Returns an informative upload error message. + * + * @return string + * The error message regarding a failed upload. + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\Validation\UploadedFileValidatorInterface::validate() + * instead. + * + * @see https://www.drupal.org/node/3375456 + */ + public function getErrorMessage(): string; + + /** + * Returns the upload error code. + * + * If the upload was successful, the constant UPLOAD_ERR_OK is returned. + * Otherwise, one of the other UPLOAD_ERR_XXX constants is returned. + * + * @return int + * The upload error code. + * + * @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use + * \Drupal\file\Validation\UploadedFileValidatorInterface::validate() + * instead. + * + * @see https://www.drupal.org/node/3375456 + */ + public function getError(): int; + /** * Gets file size. * diff --git a/core/modules/file/src/Validation/FileValidator.php b/core/modules/file/src/Validation/FileValidator.php index c649f219be59..4e517e531ad8 100644 --- a/core/modules/file/src/Validation/FileValidator.php +++ b/core/modules/file/src/Validation/FileValidator.php @@ -2,9 +2,13 @@ namespace Drupal\file\Validation; +use Drupal\Component\Plugin\Exception\PluginNotFoundException; use Drupal\Core\Extension\ModuleHandlerInterface; use Drupal\Core\Validation\ConstraintManager; +use Drupal\Core\Validation\DrupalTranslator; use Drupal\file\FileInterface; +use Symfony\Component\Validator\ConstraintViolation; +use Symfony\Component\Validator\ConstraintViolationList; use Symfony\Component\Validator\ConstraintViolationListInterface; use Symfony\Component\Validator\Validator\ValidatorInterface; use Symfony\Contracts\EventDispatcher\EventDispatcherInterface; @@ -38,16 +42,52 @@ public function __construct( */ public function validate(FileInterface $file, array $validators): ConstraintViolationListInterface { $constraints = []; + $errors = []; foreach ($validators as $validator => $options) { - // Create the constraint. - // Options are an associative array of constraint properties and values. - $constraints[] = $this->constraintManager->create($validator, $options); + if (function_exists($validator)) { + @trigger_error('Support for file validation function ' . $validator . '() is deprecated in drupal:10.2.0 and will be removed in drupal:11.0.0. Use Symfony Constraints instead. See https://www.drupal.org/node/3363700', E_USER_DEPRECATED); + if (!is_array($options)) { + $options = [$options]; + } + array_unshift($options, $file); + // Call the validation function. + // Options are a list of function args. + $errors = array_merge($errors, call_user_func_array($validator, $options)); + } + else { + // Create the constraint. + // Options are an associative array of constraint properties and values. + try { + $constraints[] = $this->constraintManager->create($validator, $options); + } + catch (PluginNotFoundException) { + @trigger_error(sprintf('Passing invalid constraint plugin ID "%s" in the list of $validators to Drupal\file\Validation\FileValidator::validate() is deprecated in drupal:10.2.0 and will throw an exception in drupal:11.0.0. See https://www.drupal.org/node/3363700', $validator), E_USER_DEPRECATED); + } + } + } + + // Call legacy hook implementations. + $errors = array_merge($errors, $this->moduleHandler->invokeAllDeprecated('Use file validation events instead. See https://www.drupal.org/node/3363700', 'file_validate', [$file])); + + $violations = new ConstraintViolationList(); + + // Convert legacy errors to violations. + $translator = new DrupalTranslator(); + foreach ($errors as $error) { + $violation = new ConstraintViolation($translator->trans($error), + $error, + [], + $file, + '', + NULL + ); + $violations->add($violation); } // Get the typed data. $fileTypedData = $file->getTypedData(); - $violations = $this->validator->validate($fileTypedData, $constraints); + $violations->addAll($this->validator->validate($fileTypedData, $constraints)); $this->eventDispatcher->dispatch(new FileValidationEvent($file, $violations)); diff --git a/core/modules/file/tests/src/Functional/SaveUploadFormTest.php b/core/modules/file/tests/src/Functional/SaveUploadFormTest.php index 63a8641d851b..612396bdfbb4 100644 --- a/core/modules/file/tests/src/Functional/SaveUploadFormTest.php +++ b/core/modules/file/tests/src/Functional/SaveUploadFormTest.php @@ -372,7 +372,7 @@ public function testHandleFileMunge() { // Check that the correct hooks were called. $this->assertFileHooksCalled(['validate', 'insert']); - // Ensure that setting $validators['FileExtension'] = ['extensions' => NULL] + // Ensure that setting $validators['file_validate_extensions'] = [''] // rejects all files. // Reset the hook counters. file_test_reset(); diff --git a/core/modules/file/tests/src/Kernel/FileSaveUploadTest.php b/core/modules/file/tests/src/Kernel/FileSaveUploadTest.php new file mode 100644 index 000000000000..d512a0f74272 --- /dev/null +++ b/core/modules/file/tests/src/Kernel/FileSaveUploadTest.php @@ -0,0 +1,70 @@ +<?php + +namespace Drupal\Tests\file\Kernel; + +use Drupal\Core\Messenger\MessengerInterface; +use Drupal\KernelTests\KernelTestBase; +use Symfony\Component\HttpFoundation\File\UploadedFile; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpFoundation\RequestStack; + +/** + * Tests file_save_upload(). + * + * @group file + * @group legacy + */ +class FileSaveUploadTest extends KernelTestBase { + + /** + * {@inheritdoc} + */ + protected static $modules = [ + 'file', + 'file_test', + 'file_validator_test', + 'user', + ]; + + /** + * {@inheritdoc} + */ + protected function setUp(): void { + \file_put_contents('test.bbb', 'test'); + + parent::setUp(); + $request = new Request(); + $request->files->set('files', [ + 'file' => new UploadedFile( + path: 'test.bbb', + originalName: 'test.bbb', + mimeType: 'text/plain', + error: \UPLOAD_ERR_OK, + test: TRUE + ), + ]); + + $requestStack = new RequestStack(); + $requestStack->push($request); + + $this->container->set('request_stack', $requestStack); + } + + /** + * Tests file_save_upload() with empty extensions. + */ + public function testFileSaveUploadEmptyExtensions(): void { + // Allow all extensions. + $validators = ['file_validate_extensions' => '']; + $this->expectDeprecation('\'file_validate_extensions\' is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'FileExtension\' constraint instead. See https://www.drupal.org/node/3363700'); + $files = file_save_upload('file', $validators); + $this->assertCount(1, $files); + $file = $files[0]; + // @todo work out why move_uploaded_file() is failing. + $this->assertFalse($file); + $messages = \Drupal::messenger()->messagesByType(MessengerInterface::TYPE_ERROR); + $this->assertNotEmpty($messages); + $this->assertEquals('File upload error. Could not move uploaded file.', $messages[0]); + } + +} diff --git a/core/modules/file/tests/src/Kernel/FileUploadHandlerTest.php b/core/modules/file/tests/src/Kernel/FileUploadHandlerTest.php index edd56b6a349a..5410552c9c29 100644 --- a/core/modules/file/tests/src/Kernel/FileUploadHandlerTest.php +++ b/core/modules/file/tests/src/Kernel/FileUploadHandlerTest.php @@ -7,6 +7,7 @@ use Drupal\file\Upload\FileUploadHandler; use Drupal\file\Upload\UploadedFileInterface; use Drupal\KernelTests\KernelTestBase; +use Symfony\Component\Mime\MimeTypeGuesserInterface; /** * Tests the file upload handler. @@ -33,6 +34,44 @@ protected function setUp(): void { $this->fileUploadHandler = $this->container->get('file.upload_handler'); } + /** + * Tests the legacy extension support. + * + * @group legacy + */ + public function testLegacyExtensions(): void { + $filename = $this->randomMachineName() . '.txt'; + $uploadedFile = $this->createMock(UploadedFileInterface::class); + $uploadedFile->expects($this->once()) + ->method('getClientOriginalName') + ->willReturn($filename); + $uploadedFile->expects($this->once())->method('isValid')->willReturn(TRUE); + + // Throw an exception in mimeTypeGuesser to return early from the method. + $mimeTypeGuesser = $this->createMock(MimeTypeGuesserInterface::class); + $mimeTypeGuesser->expects($this->once())->method('guessMimeType') + ->willThrowException(new \RuntimeException('Expected exception')); + + $fileUploadHandler = new FileUploadHandler( + fileSystem: $this->container->get('file_system'), + entityTypeManager: $this->container->get('entity_type.manager'), + streamWrapperManager: $this->container->get('stream_wrapper_manager'), + eventDispatcher: $this->container->get('event_dispatcher'), + mimeTypeGuesser: $mimeTypeGuesser, + currentUser: $this->container->get('current_user'), + requestStack: $this->container->get('request_stack'), + fileRepository: $this->container->get('file.repository'), + file_validator: $this->container->get('file.validator'), + ); + + $this->expectException(\Exception::class); + $this->expectDeprecation('\'file_validate_extensions\' is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'FileExtension\' constraint instead. See https://www.drupal.org/node/3363700'); + $fileUploadHandler->handleFileUpload($uploadedFile, ['file_validate_extensions' => ['txt']]); + + $subscriber = $this->container->get('file_validation_sanitization_subscriber'); + $this->assertEquals(['txt'], $subscriber->getAllowedExtensions()); + } + /** * Test the lock acquire exception. */ @@ -61,7 +100,7 @@ public function testLockAcquireException(): void { $this->expectException(LockAcquiringException::class); $this->expectExceptionMessage(sprintf('File "temporary://%s" is already locked for writing.', $file_name)); - $fileUploadHandler->handleFileUpload(uploadedFile: $file_info); + $fileUploadHandler->handleFileUpload(uploadedFile: $file_info, throw: FALSE); } } diff --git a/core/modules/file/tests/src/Kernel/LegacyFileModuleTest.php b/core/modules/file/tests/src/Kernel/LegacyFileModuleTest.php new file mode 100644 index 000000000000..1239a570c801 --- /dev/null +++ b/core/modules/file/tests/src/Kernel/LegacyFileModuleTest.php @@ -0,0 +1,50 @@ +<?php + +declare(strict_types=1); + +namespace Drupal\Tests\file\Kernel; + +use Drupal\KernelTests\KernelTestBase; + +// cspell:ignore msword + +/** + * Tests file module deprecations. + * + * @group legacy + * @group file + */ +class LegacyFileModuleTest extends KernelTestBase { + + /** + * {@inheritdoc} + */ + protected static $modules = ['file']; + + /** + * @covers ::file_progress_implementation + */ + public function testFileProgressDeprecation() { + $this->expectDeprecation('file_progress_implementation() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use extension_loaded(\'uploadprogress\') instead. See https://www.drupal.org/node/3397577'); + $this->assertFalse(\file_progress_implementation()); + } + + /** + * @covers ::file_icon_map + */ + public function testFileIconMapDeprecation(): void { + $this->expectDeprecation('file_icon_map() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use \Drupal\file\IconMimeTypes::getGenericMimeType() instead. See https://www.drupal.org/node/3411269'); + $mimeType = \file_icon_map('application/msword'); + $this->assertEquals('x-office-document', $mimeType); + } + + /** + * @covers ::file_icon_class + */ + public function testFileIconClassDeprecation(): void { + $this->expectDeprecation('file_icon_class() is deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use \Drupal\file\IconMimeTypes::getIconClass() instead. See https://www.drupal.org/node/3411269'); + $iconClass = \file_icon_class('image/jpeg'); + $this->assertEquals('image', $iconClass); + } + +} diff --git a/core/modules/file/tests/src/Kernel/LegacyFileThemeTest.php b/core/modules/file/tests/src/Kernel/LegacyFileThemeTest.php new file mode 100644 index 000000000000..257a85df8e57 --- /dev/null +++ b/core/modules/file/tests/src/Kernel/LegacyFileThemeTest.php @@ -0,0 +1,48 @@ +<?php + +namespace Drupal\Tests\file\Kernel; + +use Drupal\KernelTests\KernelTestBase; + +/** + * Tests legacy deprecated functions in file.module. + * + * @group file + * @group legacy + */ +class LegacyFileThemeTest extends KernelTestBase { + + /** + * {@inheritdoc} + */ + protected static $modules = ['file']; + + /** + * @covers ::template_preprocess_file_upload_help + */ + public function testTemplatePreprocessFileUploadHelp(): void { + $variables['description'] = 'foo'; + $variables['cardinality'] = 1; + $variables['upload_validators'] = [ + 'file_validate_size' => [1000], + 'file_validate_extensions' => ['txt'], + 'file_validate_image_resolution' => ['100x100', '50x50'], + ]; + + $this->expectDeprecation('\'file_validate_size\' is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'FileSizeLimit\' constraint instead. See https://www.drupal.org/node/3363700'); + $this->expectDeprecation('\'file_validate_extensions\' is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'FileExtension\' constraint instead. See https://www.drupal.org/node/3363700'); + $this->expectDeprecation('\'file_validate_image_resolution\' is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'FileImageDimensions\' constraint instead. See https://www.drupal.org/node/3363700'); + + template_preprocess_file_upload_help($variables); + + $this->assertCount(5, $variables['descriptions']); + + $descriptions = $variables['descriptions']; + $this->assertEquals('foo', $descriptions[0]); + $this->assertEquals('One file only.', $descriptions[1]); + $this->assertEquals('1000 bytes limit.', $descriptions[2]); + $this->assertEquals('Allowed types: txt.', $descriptions[3]); + $this->assertEquals('Images must be larger than 50x50 pixels. Images larger than 100x100 pixels will be resized.', $descriptions[4]); + } + +} diff --git a/core/modules/file/tests/src/Kernel/LegacyValidateTest.php b/core/modules/file/tests/src/Kernel/LegacyValidateTest.php new file mode 100644 index 000000000000..1317803603c7 --- /dev/null +++ b/core/modules/file/tests/src/Kernel/LegacyValidateTest.php @@ -0,0 +1,59 @@ +<?php + +namespace Drupal\Tests\file\Kernel; + +/** + * Tests the file_validate() function. + * + * @group file + * @group legacy + */ +class LegacyValidateTest extends FileManagedUnitTestBase { + + /** + * Tests that the validators passed into are checked. + */ + public function testCallerValidation() { + $file = $this->createFile(); + + // Empty validators. + $this->expectDeprecation('file_validate() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700'); + $this->assertEquals([], file_validate($file, []), 'Validating an empty array works successfully.'); + $this->assertFileHooksCalled([]); + + // Use the file_test.module's test validator to ensure that passing tests + // return correctly. + file_test_reset(); + file_test_set_return('validate', []); + $passing = ['file_test_validator' => [[]]]; + $this->assertEquals([], file_validate($file, $passing), 'Validating passes.'); + $this->assertFileHooksCalled([]); + + // Now test for failures in validators passed in and by hook_validate. + file_test_reset(); + $failing = ['file_test_validator' => [['Failed', 'Badly']]]; + $this->assertEquals(['Failed', 'Badly'], file_validate($file, $failing), 'Validating returns errors.'); + $this->assertFileHooksCalled([]); + } + + /** + * Tests hard-coded security check in file_validate(). + */ + public function testInsecureExtensions() { + $file = $this->createFile('test.php', 'Invalid PHP'); + + // Test that file_validate() will check for insecure extensions by default. + $this->expectDeprecation('file_validate() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700'); + $errors = file_validate($file, []); + $this->assertEquals('For security reasons, your upload has been rejected.', $errors[0]); + $this->assertFileHooksCalled([]); + file_test_reset(); + + // Test that the 'allow_insecure_uploads' is respected. + $this->config('system.file')->set('allow_insecure_uploads', TRUE)->save(); + $errors = file_validate($file, []); + $this->assertEmpty($errors); + $this->assertFileHooksCalled([]); + } + +} diff --git a/core/modules/file/tests/src/Kernel/LegacyValidatorTest.php b/core/modules/file/tests/src/Kernel/LegacyValidatorTest.php new file mode 100644 index 000000000000..20c27fe2d8b0 --- /dev/null +++ b/core/modules/file/tests/src/Kernel/LegacyValidatorTest.php @@ -0,0 +1,255 @@ +<?php + +namespace Drupal\Tests\file\Kernel; + +use Drupal\file\Entity\File; + +/** + * Tests the functions used to validate uploaded files. + * + * @group file + * @group legacy + */ +class LegacyValidatorTest extends FileManagedUnitTestBase { + + /** + * An image file. + * + * @var \Drupal\file\FileInterface + */ + protected $image; + + /** + * A file which is not an image. + * + * @var \Drupal\file\Entity\File + */ + protected $nonImage; + + /** + * {@inheritdoc} + */ + protected function setUp(): void { + parent::setUp(); + + $this->image = File::create(); + $this->image->setFileUri('core/misc/druplicon.png'); + /** @var \Drupal\Core\File\FileSystemInterface $file_system */ + $file_system = \Drupal::service('file_system'); + $this->image->setFilename($file_system->basename($this->image->getFileUri())); + $this->image->setSize(@filesize($this->image->getFileUri())); + + $this->nonImage = File::create(); + $this->nonImage->setFileUri('core/assets/vendor/jquery/jquery.min.js'); + $this->nonImage->setFilename($file_system->basename($this->nonImage->getFileUri())); + } + + /** + * Tests the file_validate_extensions() function. + */ + public function testFileValidateExtensions() { + $file = File::create(['filename' => 'asdf.txt']); + $this->expectDeprecation('file_validate_extensions() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700'); + $errors = file_validate_extensions($file, 'asdf txt pork'); + $this->assertCount(0, $errors, 'Valid extension accepted.'); + + $file->setFilename('asdf.txt'); + $errors = file_validate_extensions($file, 'exe png'); + $this->assertCount(1, $errors, 'Invalid extension blocked.'); + } + + /** + * Tests the file_validate_extensions() function. + * + * @param array $file_properties + * The properties of the file being validated. + * @param string[] $extensions + * An array of the allowed file extensions. + * @param string[] $expected_errors + * The expected error messages as string. + * + * @dataProvider providerTestFileValidateExtensionsOnUri + */ + public function testFileValidateExtensionsOnUri(array $file_properties, array $extensions, array $expected_errors) { + $file = File::create($file_properties); + $this->expectDeprecation('file_validate_extensions() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700'); + $actual_errors = file_validate_extensions($file, implode(' ', $extensions)); + $actual_errors_as_string = array_map(function ($error_message) { + return (string) $error_message; + }, $actual_errors); + $this->assertEquals($expected_errors, $actual_errors_as_string); + } + + /** + * Data provider for ::testFileValidateExtensionsOnUri. + * + * @return array[][] + * The test cases. + */ + public static function providerTestFileValidateExtensionsOnUri(): array { + $temporary_txt_file_properties = [ + 'filename' => 'asdf.txt', + 'uri' => 'temporary://asdf', + 'status' => 0, + ]; + $permanent_txt_file_properties = [ + 'filename' => 'asdf.txt', + 'uri' => 'public://asdf_0.txt', + 'status' => 1, + ]; + $permanent_png_file_properties = [ + 'filename' => 'The Druplicon', + 'uri' => 'public://druplicon.png', + 'status' => 1, + ]; + return [ + 'Temporary txt validated with "asdf", "txt", "pork"' => [ + 'File properties' => $temporary_txt_file_properties, + 'Allowed_extensions' => ['asdf', 'txt', 'pork'], + 'Expected errors' => [], + ], + 'Temporary txt validated with "exe" and "png"' => [ + 'File properties' => $temporary_txt_file_properties, + 'Allowed_extensions' => ['exe', 'png'], + 'Expected errors' => [ + 'Only files with the following extensions are allowed: <em class="placeholder">exe png</em>.', + ], + ], + 'Permanent txt validated with "asdf", "txt", "pork"' => [ + 'File properties' => $permanent_txt_file_properties, + 'Allowed_extensions' => ['asdf', 'txt', 'pork'], + 'Expected errors' => [], + ], + 'Permanent txt validated with "exe" and "png"' => [ + 'File properties' => $permanent_txt_file_properties, + 'Allowed_extensions' => ['exe', 'png'], + 'Expected errors' => [ + 'Only files with the following extensions are allowed: <em class="placeholder">exe png</em>.', + ], + ], + 'Permanent png validated with "png", "gif", "jpg", "jpeg"' => [ + 'File properties' => $permanent_png_file_properties, + 'Allowed_extensions' => ['png', 'gif', 'jpg', 'jpeg'], + 'Expected errors' => [], + ], + 'Permanent png validated with "exe" and "txt"' => [ + 'File properties' => $permanent_png_file_properties, + 'Allowed_extensions' => ['exe', 'txt'], + 'Expected errors' => [ + 'Only files with the following extensions are allowed: <em class="placeholder">exe txt</em>.', + ], + ], + ]; + } + + /** + * This ensures a specific file is actually an image. + */ + public function testFileValidateIsImage() { + $this->assertFileExists($this->image->getFileUri()); + $this->expectDeprecation('file_validate_is_image() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700'); + $errors = file_validate_is_image($this->image); + $this->assertCount(0, $errors, 'No error reported for our image file.'); + + $this->assertFileExists($this->nonImage->getFileUri()); + $errors = file_validate_is_image($this->nonImage); + $this->assertCount(1, $errors, 'An error reported for our non-image file.'); + } + + /** + * This ensures the dimensions of a specific file is within bounds. + * + * The image will be resized if it's too large. + */ + public function testFileValidateImageResolution() { + // Non-images. + $this->expectDeprecation('file_validate_image_resolution() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700'); + $errors = file_validate_image_resolution($this->nonImage); + $this->assertCount(0, $errors, 'Should not get any errors for a non-image file.'); + $errors = file_validate_image_resolution($this->nonImage, '50x50', '100x100'); + $this->assertCount(0, $errors, 'Do not check the dimensions on non files.'); + + // Minimum size. + $errors = file_validate_image_resolution($this->image); + $this->assertCount(0, $errors, 'No errors for an image when there is no minimum or maximum dimensions.'); + $errors = file_validate_image_resolution($this->image, 0, '200x1'); + $this->assertCount(1, $errors, 'Got an error for an image that was not wide enough.'); + $errors = file_validate_image_resolution($this->image, 0, '1x200'); + $this->assertCount(1, $errors, 'Got an error for an image that was not tall enough.'); + $errors = file_validate_image_resolution($this->image, 0, '200x200'); + $this->assertCount(1, $errors, 'Small images report an error.'); + + // Maximum size. + if ($this->container->get('image.factory')->getToolkitId()) { + // Copy the image so that the original doesn't get resized. + copy('core/misc/druplicon.png', 'temporary://druplicon.png'); + $this->image->setFileUri('temporary://druplicon.png'); + + $errors = file_validate_image_resolution($this->image, '10x5'); + $this->assertCount(0, $errors, 'No errors should be reported when an oversized image can be scaled down.'); + + $image = $this->container->get('image.factory')->get($this->image->getFileUri()); + // Verify that the image was scaled to the correct width and height. + $this->assertLessThanOrEqual(10, $image->getWidth()); + $this->assertLessThanOrEqual(5, $image->getHeight()); + // Verify that the file size has been updated after resizing. + $this->assertEquals($this->image->getSize(), $image->getFileSize()); + + // Once again, now with negative width and height to force an error. + copy('core/misc/druplicon.png', 'temporary://druplicon.png'); + $this->image->setFileUri('temporary://druplicon.png'); + $errors = file_validate_image_resolution($this->image, '-10x-5'); + $this->assertCount(1, $errors, 'An error reported for an oversized image that can not be scaled down.'); + + \Drupal::service('file_system')->unlink('temporary://druplicon.png'); + } + else { + // TODO: should check that the error is returned if no toolkit is available. + $errors = file_validate_image_resolution($this->image, '5x10'); + $this->assertCount(1, $errors, 'Oversize images that cannot be scaled get an error.'); + } + } + + /** + * This will ensure the filename length is valid. + */ + public function testFileValidateNameLength() { + // Create a new file entity. + $file = File::create(); + + // Add a filename with an allowed length and test it. + $file->setFilename(str_repeat('x', 240)); + $this->assertEquals(240, strlen($file->getFilename())); + $this->expectDeprecation('file_validate_name_length() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700'); + $errors = file_validate_name_length($file); + $this->assertCount(0, $errors, 'No errors reported for 240 length filename.'); + + // Add a filename with a length too long and test it. + $file->setFilename(str_repeat('x', 241)); + $errors = file_validate_name_length($file); + $this->assertCount(1, $errors, 'An error reported for 241 length filename.'); + + // Add a filename with an empty string and test it. + $file->setFilename(''); + $errors = file_validate_name_length($file); + $this->assertCount(1, $errors, 'An error reported for 0 length filename.'); + } + + /** + * Tests file_validate_size(). + */ + public function testFileValidateSize() { + // Create a file with a size of 1000 bytes, and quotas of only 1 byte. + $file = File::create(['filesize' => 1000]); + $this->expectDeprecation('file_validate_size() is deprecated in drupal:10.2.0 and is removed from drupal:11.0.0. Use the \'file.validator\' service instead. See https://www.drupal.org/node/3363700'); + $errors = file_validate_size($file, 0, 0); + $this->assertCount(0, $errors, 'No limits means no errors.'); + $errors = file_validate_size($file, 1, 0); + $this->assertCount(1, $errors, 'Error for the file being over the limit.'); + $errors = file_validate_size($file, 0, 1); + $this->assertCount(1, $errors, 'Error for the user being over their limit.'); + $errors = file_validate_size($file, 1, 1); + $this->assertCount(2, $errors, 'Errors for both the file and their limit.'); + } + +} diff --git a/core/modules/file/tests/src/Kernel/Upload/LegacyFileUploadHandlerTest.php b/core/modules/file/tests/src/Kernel/Upload/LegacyFileUploadHandlerTest.php new file mode 100644 index 000000000000..3ca6d2b623f8 --- /dev/null +++ b/core/modules/file/tests/src/Kernel/Upload/LegacyFileUploadHandlerTest.php @@ -0,0 +1,39 @@ +<?php + +namespace Drupal\Tests\file\Kernel\Upload; + +use Drupal\file\Upload\UploadedFileInterface; +use Drupal\KernelTests\KernelTestBase; +use Symfony\Component\HttpFoundation\File\Exception\FileException; + +/** + * Provides tests for legacy file upload handler code. + * + * @group file + * @group legacy + * @coversDefaultClass \Drupal\file\Upload\FileUploadHandler + */ +class LegacyFileUploadHandlerTest extends KernelTestBase { + + /** + * {@inheritdoc} + */ + protected static $modules = ['file']; + + /** + * @covers ::handleFileUpload + */ + public function testThrow(): void { + $fileUploadHandler = $this->container->get('file.upload_handler'); + + $uploadedFile = $this->createMock(UploadedFileInterface::class); + $uploadedFile->expects($this->once()) + ->method('isValid') + ->willReturn(FALSE); + + $this->expectDeprecation('Calling Drupal\file\Upload\FileUploadHandler::handleFileUpload() with the $throw argument as TRUE is deprecated in drupal:10.3.0 and will be removed in drupal:11.0.0. Use \Drupal\file\Upload\FileUploadResult::getViolations() instead. See https://www.drupal.org/node/3375456'); + $this->expectException(FileException::class); + $result = $fileUploadHandler->handleFileUpload(uploadedFile: $uploadedFile, throw: TRUE); + } + +} diff --git a/core/modules/file/tests/src/Kernel/Validation/FileValidatorTest.php b/core/modules/file/tests/src/Kernel/Validation/FileValidatorTest.php index 4effd1ac0c5d..a990cae8cd24 100644 --- a/core/modules/file/tests/src/Kernel/Validation/FileValidatorTest.php +++ b/core/modules/file/tests/src/Kernel/Validation/FileValidatorTest.php @@ -24,32 +24,40 @@ class FileValidatorTest extends FileValidatorTestBase { /** * Tests the validator. + * + * @group legacy */ public function testValidate(): void { - // Use plugin IDs to test they work. + // Use a mix of legacy functions and plugin IDs to test both work. // Each Constraint has its own tests under // core/modules/file/tests/src/Kernel/Plugin/Validation/Constraint. + // Also check that arbitrary strings can be used. $validators = [ + 'file_validate_name_length' => [], 'FileNameLength' => [], + 'foo' => [], ]; file_test_reset(); + $this->expectDeprecation('Support for file validation function file_validate_name_length() is deprecated in drupal:10.2.0 and will be removed in drupal:11.0.0. Use Symfony Constraints instead. See https://www.drupal.org/node/3363700'); $violations = $this->validator->validate($this->file, $validators); $this->assertCount(0, $violations); $this->assertCount(1, file_test_get_calls('validate')); + $this->expectDeprecation('Passing invalid constraint plugin ID "foo" in the list of $validators to Drupal\file\Validation\FileValidator::validate() is deprecated in drupal:10.2.0 and will throw an exception in drupal:11.0.0. See https://www.drupal.org/node/3363700'); file_test_reset(); $this->file->set('filename', ''); $violations = $this->validator->validate($this->file, $validators); - $this->assertCount(1, $violations); - $this->assertEquals($violations[0]->getMessage(), $violations[0]->getMessage(), 'Message names are equal'); + $this->assertCount(2, $violations); + $this->assertEquals($violations[0]->getMessage(), $violations[1]->getMessage(), 'Message names are equal'); $this->assertCount(1, file_test_get_calls('validate')); file_test_reset(); $this->file->set('filename', $this->randomMachineName(241)); $violations = $this->validator->validate($this->file, $validators); - $this->assertCount(1, $violations); + $this->assertCount(2, $violations); $this->assertEquals("The file's name exceeds the 240 characters limit. Rename the file and try again.", $violations[0]->getMessage()); + $this->assertEquals("The file's name exceeds the 240 characters limit. Rename the file and try again.", $violations[1]->getMessage()); $this->assertCount(1, file_test_get_calls('validate')); } -- GitLab