View source
<?php
namespace Drupal\filefield_sources\Plugin\FilefieldSource;
use Drupal\Core\Form\FormStateInterface;
use Drupal\filefield_sources\FilefieldSourceInterface;
use Symfony\Component\Routing\Route;
use Drupal\Core\Field\WidgetInterface;
use Symfony\Component\HttpFoundation\JsonResponse;
use Drupal\Component\Utility\UrlHelper;
use Drupal\Core\Site\Settings;
use Drupal\Component\Utility\Unicode;
use Drupal\Core\File\FileSystem;
use Drupal\Core\File\FileSystemInterface;
class Remote implements FilefieldSourceInterface {
public static function value(array &$element, &$input, FormStateInterface $form_state) {
if (isset($input['filefield_remote']['url']) && strlen($input['filefield_remote']['url']) > 0 && UrlHelper::isValid($input['filefield_remote']['url']) && $input['filefield_remote']['url'] != FILEFIELD_SOURCE_REMOTE_HINT_TEXT) {
$field = \Drupal::entityTypeManager()
->getStorage('field_config')
->load($element['#entity_type'] . '.' . $element['#bundle'] . '.' . $element['#field_name']);
$url = $input['filefield_remote']['url'];
$temporary_directory = 'temporary://';
if (!\Drupal::service('file_system')
->prepareDirectory($temporary_directory, FileSystemInterface::MODIFY_PERMISSIONS)) {
\Drupal::logger('filefield_sources')
->log(E_NOTICE, 'The directory %directory is not writable, because it does not have the correct permissions set.', [
'%directory' => \Drupal::service('file_system')
->realpath($temporary_directory),
]);
\Drupal::messenger()
->addError(t('The file could not be transferred because the temporary directory is not writable.'), 'error');
return;
}
$directory = $element['#upload_location'];
$mode = Settings::get('file_chmod_directory', FileSystem::CHMOD_DIRECTORY);
if (!\Drupal::service('file_system')
->chmod($directory, $mode) && !\Drupal::service('file_system')
->prepareDirectory($directory, FileSystemInterface::CREATE_DIRECTORY)) {
\Drupal::logger('filefield_sources')
->log(E_NOTICE, 'File %file could not be copied, because the destination directory %destination is not configured correctly.', [
'%file' => $url,
'%destination' => \Drupal::service('file_system')
->realpath($directory),
]);
\Drupal::messenger()
->addError(t('The specified file %file could not be copied, because the destination directory is not properly configured. This may be caused by a problem with file or directory permissions. More information is available in the system log.', [
'%file' => $url,
]), 'error');
return;
}
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_HEADER, TRUE);
curl_setopt($ch, CURLOPT_NOBODY, TRUE);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
curl_setopt($ch, CURLOPT_HEADERFUNCTION, [
get_called_class(),
'parseHeader',
]);
@curl_setopt($ch, CURLOPT_FOLLOWLOCATION, TRUE);
curl_exec($ch);
$info = curl_getinfo($ch);
if ($info['http_code'] != 200) {
curl_setopt($ch, CURLOPT_HTTPGET, TRUE);
$file_contents = curl_exec($ch);
$info = curl_getinfo($ch);
}
curl_close($ch);
if ($info['http_code'] != 200) {
switch ($info['http_code']) {
case 403:
$form_state
->setError($element, t('The remote file could not be transferred because access to the file was denied.'));
break;
case 404:
$form_state
->setError($element, t('The remote file could not be transferred because it was not found.'));
break;
default:
$form_state
->setError($element, t('The remote file could not be transferred due to an HTTP error (@code).', [
'@code' => $info['http_code'],
]));
}
return;
}
$url = $info['url'];
$url_info = parse_url($url);
$filesystem = \Drupal::service('file_system');
$filename = static::filename();
if (empty($filename)) {
$filename = rawurldecode($filesystem
->basename($url_info['path']));
}
$filename = \Drupal::transliteration()
->transliterate($filename);
$pathinfo = pathinfo($filename);
if (empty($pathinfo['extension']) && ($extension = static::mimeExtension())) {
$filename = $filename . '.' . $extension;
$pathinfo = pathinfo($filename);
}
$filename = filefield_sources_clean_filename($filename, $field
->getSetting('file_extensions'));
$filepath = \Drupal::service('file_system')
->createFilename($filename, $temporary_directory);
if (empty($pathinfo['extension'])) {
$form_state
->setError($element, t('The remote URL must be a file and have an extension.'));
return;
}
$extensions = $field
->getSetting('file_extensions');
$regex = '/\\.(' . preg_replace('/[ +]/', '|', preg_quote($extensions)) . ')$/i';
if (!empty($extensions) && !preg_match($regex, $filename)) {
$form_state
->setError($element, t('Only files with the following extensions are allowed: %files-allowed.', [
'%files-allowed' => $extensions,
]));
return;
}
if (!empty($element['#upload_validators']['file_validate_size'][0])) {
$max_size = $element['#upload_validators']['file_validate_size'][0];
$file_size = $info['download_content_length'];
if ($file_size > $max_size) {
$form_state
->setError($element, t('The remote file is %filesize exceeding the maximum file size of %maxsize.', [
'%filesize' => format_size($file_size),
'%maxsize' => format_size($max_size),
]));
return;
}
}
$options = [
'key' => $element['#entity_type'] . '_' . $element['#bundle'] . '_' . $element['#field_name'] . '_' . $element['#delta'],
'filepath' => $filepath,
];
static::setTransferOptions($options);
$transfer_success = FALSE;
if (isset($file_contents)) {
if ($fp = @fopen($filepath, 'w')) {
fwrite($fp, $file_contents);
fclose($fp);
$transfer_success = TRUE;
}
}
else {
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_HEADER, FALSE);
curl_setopt($ch, CURLOPT_WRITEFUNCTION, [
get_called_class(),
'curlWrite',
]);
@curl_setopt($ch, CURLOPT_FOLLOWLOCATION, TRUE);
$transfer_success = curl_exec($ch);
curl_close($ch);
}
if ($transfer_success && ($file = filefield_sources_save_file($filepath, $element['#upload_validators'], $element['#upload_location']))) {
if (!in_array($file
->id(), $input['fids'])) {
$input['fids'][] = $file
->id();
}
}
@unlink($filepath);
}
}
protected static function setTransferOptions($options = NULL) {
static $current = FALSE;
if (isset($options)) {
$current = $options;
}
return $current;
}
protected static function getTransferOptions() {
return static::setTransferOptions();
}
protected static function curlWrite(&$ch, $data) {
$progress_update = 0;
$options = static::getTransferOptions();
if (curl_getinfo($ch, CURLINFO_SIZE_DOWNLOAD) / 65536 > $progress_update) {
$progress_update++;
$progress = [
'current' => curl_getinfo($ch, CURLINFO_SIZE_DOWNLOAD),
'total' => curl_getinfo($ch, CURLINFO_CONTENT_LENGTH_DOWNLOAD),
];
$cid = 'filefield_transfer:' . session_id() . ':' . $options['key'];
if ($progress['current'] != $progress['total']) {
\Drupal::cache()
->set($cid, $progress, time() + 300);
}
else {
\Drupal::cache()
->delete($cid);
}
}
$data_length = 0;
if ($fp = @fopen($options['filepath'], 'a')) {
fwrite($fp, $data);
fclose($fp);
$data_length = strlen($data);
}
return $data_length;
}
protected static function parseHeader(&$ch, $header) {
if (preg_match('/Content-Disposition:.*?filename="(.+?)"/', $header, $matches)) {
static::filename($matches[1]);
}
elseif (preg_match('/Content-Disposition:.*?filename=([^; ]+)/', $header, $matches)) {
$uri = trim($matches[1]);
static::filename($uri);
}
elseif (preg_match('/Content-Type:[ ]*([a-z0-9_\\-]+\\/[a-z0-9_\\-]+)/i', $header, $matches)) {
$mime_type = $matches[1];
static::mimeExtension($mime_type);
}
return strlen($header);
}
protected static function mimeExtension($curl_mime_type = NULL) {
static $extension = NULL;
$mimetype = mb_strtolower($curl_mime_type);
$result = \Drupal::service('file.mime_type.guesser.extension')
->convertMimeTypeToMostCommonExtension($mimetype);
if ($result) {
$extension = $result;
}
return $extension;
}
protected static function filename($curl_filename = NULL) {
static $filename = NULL;
if (isset($curl_filename)) {
$filename = $curl_filename;
}
return $filename;
}
public static function process(array &$element, FormStateInterface $form_state, array &$complete_form) {
$element['filefield_remote'] = [
'#weight' => 100.5,
'#theme' => 'filefield_sources_element',
'#source_id' => 'remote',
'#filefield_source' => TRUE,
'#filefield_sources_hint_text' => FILEFIELD_SOURCE_REMOTE_HINT_TEXT,
];
$element['filefield_remote']['url'] = [
'#type' => 'textfield',
'#description' => filefield_sources_element_validation_help($element['#upload_validators']),
'#maxlength' => NULL,
];
$class = '\\Drupal\\file\\Element\\ManagedFile';
$ajax_settings = [
'callback' => [
$class,
'uploadAjaxCallback',
],
'options' => [
'query' => [
'element_parents' => implode('/', $element['#array_parents']),
],
],
'wrapper' => $element['upload_button']['#ajax']['wrapper'],
'effect' => 'fade',
'progress' => [
'type' => 'bar',
'path' => 'file/remote/progress/' . $element['#entity_type'] . '/' . $element['#bundle'] . '/' . $element['#field_name'] . '/' . $element['#delta'],
'message' => t('Starting transfer...'),
],
];
$element['filefield_remote']['transfer'] = [
'#name' => implode('_', $element['#parents']) . '_transfer',
'#type' => 'submit',
'#value' => t('Transfer'),
'#validate' => [],
'#submit' => [
'filefield_sources_field_submit',
],
'#limit_validation_errors' => [
$element['#parents'],
],
'#ajax' => $ajax_settings,
];
return $element;
}
public static function element($variables) {
$element = $variables['element'];
$element['url']['#field_suffix'] = \Drupal::service('renderer')
->render($element['transfer']);
return '<div class="filefield-source filefield-source-remote clear-block">' . \Drupal::service('renderer')
->render($element['url']) . '</div>';
}
public static function progress($entity_type, $bundle_name, $field_name, $delta) {
$key = $entity_type . '_' . $bundle_name . '_' . $field_name . '_' . $delta;
$progress = [
'message' => t('Starting transfer...'),
'percentage' => -1,
];
if ($cache = \Drupal::cache()
->get('filefield_transfer:' . session_id() . ':' . $key)) {
$current = $cache->data['current'];
$total = $cache->data['total'];
$progress['message'] = t('Transferring... (@current of @total)', [
'@current' => format_size($current),
'@total' => format_size($total),
]);
$progress['percentage'] = round(100 * $current / $total);
}
return new JsonResponse($progress);
}
public static function routes() {
$routes = [];
$routes['filefield_sources.remote'] = new Route('/file/remote/progress/{entity_type}/{bundle_name}/{field_name}/{delta}', [
'_controller' => get_called_class() . '::progress',
], [
'_access' => 'TRUE',
]);
return $routes;
}
public static function settings(WidgetInterface $plugin) {
$return = [];
if (!filefield_sources_curl_enabled()) {
\Drupal::messenger()
->addError(t('<strong>Filefield sources:</strong> remote plugin will be disabled without php-curl extension.'), 'warning');
}
return $return;
}
}