t('RSS import to nodes'), 'description' => t('Tests a feed configuration that is attached to a content type, uses HTTP fetcher, common syndication parser and a node processor. Repeats the same test for an importer configuration that is not attached to a content type and for a configuration that is attached to a content type and uses the file fetcher.'), 'group' => t('Feeds'), ); } /** * Set up test. */ public function setUp() { parent::setUp(); // Set the front page to show 20 nodes so we can easily see what is aggregated. $edit = array('default_nodes_main' => 20); $this->drupalPost('admin/config/system/site-information', $edit, 'Save configuration'); // Set the teaser length display to unlimited otherwise tests looking for // text on nodes will fail. $edit = array('fields[body][type]' => 'text_default'); $this->drupalPost('admin/structure/types/manage/article/display/teaser', $edit, 'Save'); } /** * Test node creation, refreshing/deleting feeds and feed items. */ public function test() { // Create an importer configuration. $this->createImporterConfiguration('Syndication', 'syndication'); $this->addMappings('syndication', array( array( 'source' => 'title', 'target' => 'title', 'unique' => FALSE, ), array( 'source' => 'description', 'target' => 'body', 'unique' => FALSE, ), array( 'source' => 'timestamp', 'target' => 'created', 'unique' => FALSE, ), array( 'source' => 'url', 'target' => 'url', 'unique' => TRUE, ), array( 'source' => 'guid', 'target' => 'guid', 'unique' => TRUE, ), ) ); $nid = $this->createFeedNode(); // Assert 10 items aggregated after creation of the node. $this->assertText('Created 10 nodes'); // Navigate to feed node, there should be Feeds tabs visible. $this->drupalGet('node/'. $nid); $this->assertRaw('node/'. $nid .'/import'); $this->assertRaw('node/'. $nid .'/delete-items'); // Navigate to a non-feed node, there should be no Feeds tabs visible. $article_nid = db_query_range("SELECT nid FROM {node} WHERE type = 'article'", 0, 1)->fetchField(); $this->drupalGet('node/'. $article_nid); $this->assertNoRaw('node/'. $article_nid .'/import'); $this->assertNoRaw('node/'. $article_nid .'/delete-items'); $this->assertEqual("Created by FeedsNodeProcessor", db_query("SELECT nr.log FROM {node} n JOIN {node_revision} nr ON n.vid = nr.vid WHERE n.nid = :nid", array(':nid' => $article_nid))->fetchField()); // Assert accuracy of aggregated information. $this->drupalGet('node'); $this->assertRaw('Anonymous (not verified)'); $this->assertDevseedFeedContent(); // Assert DB status. $count = db_query("SELECT COUNT(*) FROM {node} n INNER JOIN {feeds_item} fi ON fi.entity_type = 'node' AND n.nid = fi.entity_id")->fetchField(); $this->assertEqual($count, 10, 'Accurate number of items in database.'); // Assert default input format on first imported feed node. /* // NEEDS update. $format = db_query_range("SELECT nr.format FROM {feeds_node_item} fi JOIN {node} n ON fi.nid = n.nid JOIN {node_revision} nr ON n.vid = nr.vid", 0, 1)->fetchField(); $this->assertEqual($format, filter_fallback_format(), 'Using default Input format.'); */ // Import again. $this->drupalPost('node/'. $nid .'/import', array(), 'Import'); $this->assertText('There are no new nodes'); // Assert DB status, there still shouldn't be more than 10 items. $count = db_query("SELECT COUNT(*) FROM {node} n INNER JOIN {feeds_item} fi ON fi.entity_type = 'node' AND n.nid = fi.entity_id")->fetchField(); $this->assertEqual($count, 10, 'Accurate number of items in database.'); // All of the above tests should have produced published nodes, set default // to unpublished, import again. $count = db_query("SELECT COUNT(*) FROM {node} n INNER JOIN {feeds_item} fi ON fi.entity_type = 'node' AND n.nid = fi.entity_id WHERE n.status = 1")->fetchField(); $this->assertEqual($count, 10, 'All items are published.'); $edit = array( 'node_options[status]' => FALSE, ); $this->drupalPost('admin/structure/types/manage/article', $edit, t('Save content type')); $this->drupalPost('node/'. $nid .'/delete-items', array(), 'Delete'); $this->drupalPost('node/'. $nid .'/import', array(), 'Import'); $count = db_query("SELECT COUNT(*) FROM {node} n INNER JOIN {feeds_item} fi ON fi.entity_type = 'node' AND n.nid = fi.entity_id WHERE n.status = 0")->fetchField(); $this->assertEqual($count, 10, 'No items are published.'); $edit = array( 'node_options[status]' => TRUE, ); $this->drupalPost('admin/structure/types/manage/article', $edit, t('Save content type')); $this->drupalPost('node/'. $nid .'/delete-items', array(), 'Delete'); // Enable replace existing and import updated feed file. $this->drupalPost('node/'. $nid .'/import', array(), 'Import'); $this->setSettings('syndication', 'FeedsNodeProcessor', array('update_existing' => 1)); $feed_url = $GLOBALS['base_url'] .'/'. drupal_get_path('module', 'feeds') .'/tests/feeds/developmentseed_changes.rss2'; $this->editFeedNode($nid, $feed_url); $this->drupalPost('node/' . $nid . '/import', array(), 'Import'); $this->assertText('Updated 2 nodes'); // Assert accuracy of aggregated content (check 2 updates, one original). $this->drupalGet('node'); $this->assertText('Managing News Translation Workflow: Two Way Translation Updates'); $this->assertText('Presenting on Features in Drupal and Managing News'); $this->assertText('Scaling the Open Atrium UI'); // Import again. $this->drupalPost('node/'. $nid .'/import', array(), 'Import'); $this->assertText('There are no new nodes'); $this->assertFeedItemCount(10); // Now delete all items. $this->drupalPost('node/'. $nid .'/delete-items', array(), 'Delete'); $this->assertText('Deleted 10 nodes'); $this->assertFeedItemCount(0); // Change author. $this->auth_user = $this->drupalCreateUser(array('access content')); $this->setSettings('syndication', 'FeedsNodeProcessor', array('author' => $this->auth_user->name)); // Change input format. $this->setSettings('syndication', 'FeedsNodeProcessor', array('input_format' => 'plain_text')); // Import again. $this->drupalPost('node/'. $nid .'/import', array(), 'Import'); $this->assertText('Created 10 nodes'); // Assert author. $this->drupalGet('node'); $this->assertPattern('/' . check_plain($this->auth_user->name) . '<\/span>/'); $count = db_query("SELECT COUNT(*) FROM {feeds_item} fi JOIN {node} n ON fi.entity_type = 'node' AND fi.entity_id = n.nid WHERE n.uid = :uid", array(':uid' => $this->auth_user->uid))->fetchField(); $this->assertEqual($count, 10, 'Accurate number of items in database.'); // Assert input format. /* // NEEDS update. $format = db_query_range("SELECT nr.format FROM {feeds_node_item} fi JOIN {node} n ON fi.nid = n.nid JOIN {node_revision} nr ON n.vid = nr.vid", 0, 1)->fetchField(); $this->assertEqual($format, filter_fallback_format() + 1, 'Set non-default Input format.'); */ // Set to update existing, remove authorship of above nodes and import again. $this->setSettings('syndication', 'FeedsNodeProcessor', array('update_existing' => 2)); db_query("UPDATE {node} n JOIN {feeds_item} fi ON fi.entity_type = 'node' AND n.nid = fi.entity_id SET n.uid = 0, fi.hash=''"); $this->drupalPost('node/'. $nid .'/import', array(), 'Import'); $this->drupalGet('node'); $this->assertNoPattern('/' . check_plain($this->auth_user->name) . '<\/span>/'); $count = db_query("SELECT COUNT(*) FROM {feeds_item} fi JOIN {node} n ON fi.entity_type = 'node' AND fi.entity_id = n.nid WHERE n.uid = :uid", array(':uid' => $this->auth_user->uid))->fetchField(); $this->assertEqual($count, 0, 'Accurate number of items in database.'); // Map feed node's author to feed item author, update - feed node's items // should now be assigned to feed node author. $this->addMappings('syndication', array( array( 'source' => 'parent:uid', 'target' => 'uid', ), ) ); $this->drupalPost('node/'. $nid .'/import', array(), 'Import'); $this->drupalGet('node'); $this->assertNoPattern('/' . check_plain($this->auth_user->name) . '<\/span>/'); $uid = db_query("SELECT uid FROM {node} WHERE nid = :nid", array(':nid' => $nid))->fetchField(); $count = db_query("SELECT COUNT(*) FROM {node} WHERE uid = :uid", array(':uid' =>$uid))->fetchField(); $this->assertEqual($count, 11, 'All feed item nodes are assigned to feed node author.'); // Login with new user with only access content permissions. $this->drupalLogin($this->auth_user); // Navigate to feed node, there should be no Feeds tabs visible. $this->drupalGet('node/'. $nid); $this->assertNoRaw('node/'. $nid .'/import'); $this->assertNoRaw('node/'. $nid .'/delete-items'); // Now create a second feed configuration that is not attached to a content // type and run tests on importing/purging. // Login with sufficient permissions. $this->drupalLogin($this->admin_user); // Remove all items again so that next test can check for them. $this->drupalPost('node/'. $nid .'/delete-items', array(), 'Delete'); // Create an importer, not attached to content type. $this->createImporterConfiguration('Syndication standalone', 'syndication_standalone'); $edit = array( 'content_type' => '', ); $this->drupalPost('admin/structure/feeds/edit/syndication_standalone/settings', $edit, 'Save'); $this->addMappings('syndication_standalone', array( array( 'source' => 'title', 'target' => 'title', 'unique' => FALSE, ), array( 'source' => 'description', 'target' => 'body', 'unique' => FALSE, ), array( 'source' => 'timestamp', 'target' => 'created', 'unique' => FALSE, ), array( 'source' => 'url', 'target' => 'url', 'unique' => TRUE, ), array( 'source' => 'guid', 'target' => 'guid', 'unique' => TRUE, ), ) ); // Import, assert 10 items aggregated after creation of the node. $this->importURL('syndication_standalone'); $this->assertText('Created 10 nodes'); // Assert accuracy of aggregated information. $this->drupalGet('node'); $this->assertDevseedFeedContent(); $this->assertFeedItemCount(10); // Import again. $this->drupalPost('import/syndication_standalone', array(), 'Import'); $this->assertText('There are no new nodes'); $this->assertFeedItemCount(10); // Enable replace existing and import updated feed file. $this->setSettings('syndication_standalone', 'FeedsNodeProcessor', array('update_existing' => 1)); $feed_url = $GLOBALS['base_url'] .'/'. drupal_get_path('module', 'feeds') . '/tests/feeds/developmentseed_changes.rss2'; $this->importURL('syndication_standalone', $feed_url); $this->assertText('Updated 2 nodes'); // Assert accuracy of aggregated information (check 2 updates, one orig). $this->drupalGet('node'); $this->assertText('Managing News Translation Workflow: Two Way Translation Updates'); $this->assertText('Presenting on Features in Drupal and Managing News'); $this->assertText('Scaling the Open Atrium UI'); // Import again. $this->drupalPost('import/syndication_standalone', array(), 'Import'); $this->assertText('There are no new nodes'); $this->assertFeedItemCount(10); // Now delete all items. $this->drupalPost('import/syndication_standalone/delete-items', array(), 'Delete'); $this->assertText('Deleted 10 nodes'); $this->assertFeedItemCount(0); // Import again, we should find new content. $this->drupalPost('import/syndication_standalone', array(), 'Import'); $this->assertText('Created 10 nodes'); $this->assertFeedItemCount(10); // Login with new user with only access content permissions. $this->drupalLogin($this->auth_user); // Navigate to feed import form, access should be denied. $this->drupalGet('import/syndication_standalone'); $this->assertResponse(403); // Use File Fetcher. $this->drupalLogin($this->admin_user); $edit = array('plugin_key' => 'FeedsFileFetcher'); $this->drupalPost('admin/structure/feeds/edit/syndication_standalone/fetcher', $edit, 'Save'); $edit = array( 'allowed_extensions' => 'rss2', ); $this->drupalPost('admin/structure/feeds/edit/syndication_standalone/settings/FeedsFileFetcher', $edit, 'Save'); // Create a feed node. $edit = array( 'files[feeds]' => $this->absolutePath() .'/tests/feeds/drupalplanet.rss2', ); $this->drupalPost('import/syndication_standalone', $edit, 'Import'); $this->assertText('Created 25 nodes'); } /** * Check that the total number of entries in the feeds_item table is correct. */ function assertFeedItemCount($num) { // Assert DB status, there should be 10 again. $count = db_query("SELECT COUNT(*) FROM {feeds_item} WHERE entity_type = 'node'")->fetchField(); $this->assertEqual($count, $num, 'Accurate number of items in database.'); } /** * Check thet contents of the current page for the DS feed. */ function assertDevseedFeedContent() { $this->assertText('Open Atrium Translation Workflow: Two Way Translation Updates'); $this->assertText('Tue, 10/06/2009'); $this->assertText('A new translation process for Open Atrium & integration with Localize Drupal'); $this->assertText('Week in DC Tech: October 5th Edition'); $this->assertText('Mon, 10/05/2009'); $this->assertText('There are some great technology events happening this week'); $this->assertText('Mapping Innovation at the World Bank with Open Atrium'); $this->assertText('Fri, 10/02/2009'); $this->assertText('is being used as a base platform for collaboration at the World Bank'); $this->assertText('September GeoDC Meetup Tonight'); $this->assertText('Wed, 09/30/2009'); $this->assertText('Today is the last Wednesday of the month'); $this->assertText('Week in DC Tech: September 28th Edition'); $this->assertText('Mon, 09/28/2009'); $this->assertText('Looking to geek out this week? There are a bunch of'); $this->assertText('Open Data for Microfinance: The New MIXMarket.org'); $this->assertText('Thu, 09/24/2009'); $this->assertText('There are profiles for every country that the MIX Market is hosting.'); $this->assertText('Integrating the Siteminder Access System in an Open Atrium-based Intranet'); $this->assertText('Tue, 09/22/2009'); $this->assertText('In addition to authentication, the Siteminder system'); $this->assertText('Week in DC Tech: September 21 Edition'); $this->assertText('Mon, 09/21/2009'); $this->assertText('an interesting variety of technology events happening in Washington, DC '); $this->assertText('s Software Freedom Day: Impressions & Photos'); $this->assertText('Mon, 09/21/2009'); $this->assertText('Presenting on Features in Drupal and Open Atrium'); $this->assertText('Scaling the Open Atrium UI'); $this->assertText('Fri, 09/18/2009'); $this->assertText('The first major change is switching'); } }