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) { $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 or a directory of files 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); } } }