123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229 |
- <?php
- /**
- * @file
- * Home of the FeedsFileFetcher and related classes.
- */
- /**
- * Definition of the import batch object created on the fetching stage by
- * FeedsFileFetcher.
- */
- class FeedsFileFetcherResult extends FeedsFetcherResult {
- /**
- * Constructor.
- */
- public function __construct($file_path) {
- parent::__construct('');
- $this->file_path = $file_path;
- }
- /**
- * Overrides parent::getRaw();
- */
- public function getRaw() {
- return $this->sanitizeRaw(file_get_contents($this->file_path));
- }
- /**
- * Overrides parent::getFilePath().
- */
- public function getFilePath() {
- if (!file_exists($this->file_path)) {
- throw new Exception(t('File @filepath is not accessible.', array('@filepath' => $this->file_path)));
- }
- return $this->sanitizeFile($this->file_path);
- }
- }
- /**
- * Fetches data via HTTP.
- */
- class FeedsFileFetcher extends FeedsFetcher {
- /**
- * Implements FeedsFetcher::fetch().
- */
- public function fetch(FeedsSource $source) {
- $source_config = $source->getConfigFor($this);
- // Just return a file fetcher result if this is a file.
- if (is_file($source_config['source'])) {
- return new FeedsFileFetcherResult($source_config['source']);
- }
- // Batch if this is a directory.
- $state = $source->state(FEEDS_FETCH);
- $files = array();
- if (!isset($state->files)) {
- $state->files = $this->listFiles($source_config['source']);
- $state->total = count($state->files);
- }
- if (count($state->files)) {
- $file = array_shift($state->files);
- $state->progress($state->total, $state->total - count($state->files));
- return new FeedsFileFetcherResult($file);
- }
- throw new Exception(t('Resource is not a file or it is an empty directory: %source', array('%source' => $source_config['source'])));
- }
- /**
- * Return an array of files in a directory.
- *
- * @param $dir
- * A stream wreapper URI that is a directory.
- *
- * @return
- * An array of stream wrapper URIs pointing to files. The array is empty
- * if no files could be found. Never contains directories.
- */
- protected function listFiles($dir) {
- $dir = file_stream_wrapper_uri_normalize($dir);
- $files = array();
- if ($items = @scandir($dir)) {
- foreach ($items as $item) {
- if (is_file("$dir/$item") && strpos($item, '.') !== 0) {
- $files[] = "$dir/$item";
- }
- }
- }
- return $files;
- }
- /**
- * Source form.
- */
- public function sourceForm($source_config) {
- $form = array();
- $form['fid'] = array(
- '#type' => 'value',
- '#value' => empty($source_config['fid']) ? 0 : $source_config['fid'],
- );
- if (empty($this->config['direct'])) {
- $form['source'] = array(
- '#type' => 'value',
- '#value' => empty($source_config['source']) ? '' : $source_config['source'],
- );
- $form['upload'] = array(
- '#type' => 'file',
- '#title' => empty($this->config['direct']) ? t('File') : NULL,
- '#description' => empty($source_config['source']) ? t('Select a file from your local system.') : t('Select a different file from your local system.'),
- '#theme' => 'feeds_upload',
- '#file_info' => empty($source_config['fid']) ? NULL : file_load($source_config['fid']),
- '#size' => 10,
- );
- }
- else {
- $form['source'] = array(
- '#type' => 'textfield',
- '#title' => t('File'),
- '#description' => t('Specify a path to a file or a directory. Path must start with @scheme://', array('@scheme' => file_default_scheme())),
- '#default_value' => empty($source_config['source']) ? '' : $source_config['source'],
- );
- }
- return $form;
- }
- /**
- * Override parent::sourceFormValidate().
- */
- public function sourceFormValidate(&$values) {
- $values['source'] = trim($values['source']);
- $feed_dir = 'public://feeds';
- file_prepare_directory($feed_dir, FILE_CREATE_DIRECTORY | FILE_MODIFY_PERMISSIONS);
- // If there is a file uploaded, save it, otherwise validate input on
- // file.
- // @todo: Track usage of file, remove file when removing source.
- if ($file = file_save_upload('feeds', array('file_validate_extensions' => array(0 => $this->config['allowed_extensions'])), $feed_dir)) {
- $values['source'] = $file->uri;
- $values['file'] = $file;
- }
- elseif (empty($values['source'])) {
- form_set_error('feeds][source', t('Upload a file first.'));
- }
- // If a file has not been uploaded and $values['source'] is not empty, make
- // sure that this file is within Drupal's files directory as otherwise
- // potentially any file that the web server has access to could be exposed.
- elseif (strpos($values['source'], file_default_scheme()) !== 0) {
- form_set_error('feeds][source', t('File needs to reside within the site\'s file directory, its path needs to start with @scheme://.', array('@scheme' => file_default_scheme())));
- }
- }
- /**
- * Override parent::sourceSave().
- */
- public function sourceSave(FeedsSource $source) {
- $source_config = $source->getConfigFor($this);
- // If a new file is present, delete the old one and replace it with the new
- // one.
- if (isset($source_config['file'])) {
- $file = $source_config['file'];
- if (isset($source_config['fid'])) {
- $this->deleteFile($source_config['fid'], $source->feed_nid);
- }
- $file->status = FILE_STATUS_PERMANENT;
- file_save($file);
- file_usage_add($file, 'feeds', get_class($this), $source->feed_nid);
- $source_config['fid'] = $file->fid;
- unset($source_config['file']);
- $source->setConfigFor($this, $source_config);
- }
- }
- /**
- * Override parent::sourceDelete().
- */
- public function sourceDelete(FeedsSource $source) {
- $source_config = $source->getConfigFor($this);
- if (isset($source_config['fid'])) {
- $this->deleteFile($source_config['fid'], $source->feed_nid);
- }
- }
- /**
- * Override parent::configDefaults().
- */
- public function configDefaults() {
- return array(
- 'allowed_extensions' => 'txt csv tsv xml opml',
- 'direct' => FALSE,
- );
- }
- /**
- * Override parent::configForm().
- */
- public function configForm(&$form_state) {
- $form = array();
- $form['allowed_extensions'] = array(
- '#type' => 'textfield',
- '#title' => t('Allowed file extensions'),
- '#description' => t('Allowed file extensions for upload.'),
- '#default_value' => $this->config['allowed_extensions'],
- );
- $form['direct'] = array(
- '#type' => 'checkbox',
- '#title' => t('Supply path to file or directory directly'),
- '#description' => t('For experts. Lets users specify a path to a file <em>or a directory of files</em> directly,
- instead of a file upload through the browser. This is useful when the files that need to be imported
- are already on the server.'),
- '#default_value' => $this->config['direct'],
- );
- return $form;
- }
- /**
- * Helper. Deletes a file.
- */
- protected function deleteFile($fid, $feed_nid) {
- if ($file = file_load($fid)) {
- file_usage_delete($file, 'feeds', get_class($this), $feed_nid);
- file_delete($file);
- }
- }
- }
|