1<?php 2 3/** 4 * @file 5 * Tests for rdf.module. 6 */ 7 8class RdfMappingHookTestCase extends DrupalWebTestCase { 9 public static function getInfo() { 10 return array( 11 'name' => 'RDF mapping hook', 12 'description' => 'Test hook_rdf_mapping().', 13 'group' => 'RDF', 14 ); 15 } 16 17 function setUp() { 18 parent::setUp('rdf', 'rdf_test', 'field_test'); 19 } 20 21 /** 22 * Test that hook_rdf_mapping() correctly returns and processes mapping. 23 */ 24 function testMapping() { 25 // Test that the mapping is returned correctly by the hook. 26 $mapping = rdf_mapping_load('test_entity', 'test_bundle'); 27 $this->assertIdentical($mapping['rdftype'], array('sioc:Post'), 'Mapping for rdftype is sioc:Post.'); 28 $this->assertIdentical($mapping['title'], array('predicates' => array('dc:title')), 'Mapping for title is dc:title.'); 29 $this->assertIdentical($mapping['created'], array( 30 'predicates' => array('dc:created'), 31 'datatype' => 'xsd:dateTime', 32 'callback' => 'date_iso8601', 33 ), t('Mapping for created is dc:created with datatype xsd:dateTime and callback date_iso8601.')); 34 $this->assertIdentical($mapping['uid'], array('predicates' => array('sioc:has_creator', 'dc:creator'), 'type' => 'rel'), 'Mapping for uid is sioc:has_creator and dc:creator, and type is rel.'); 35 36 $mapping = rdf_mapping_load('test_entity', 'test_bundle_no_mapping'); 37 $this->assertEqual($mapping, array(), 'Empty array returned when an entity type, bundle pair has no mapping.'); 38 } 39} 40 41/** 42 * Test RDFa markup generation. 43 */ 44class RdfRdfaMarkupTestCase extends DrupalWebTestCase { 45 public static function getInfo() { 46 return array( 47 'name' => 'RDFa markup', 48 'description' => 'Test RDFa markup generation.', 49 'group' => 'RDF', 50 ); 51 } 52 53 function setUp() { 54 parent::setUp('rdf', 'field_test', 'rdf_test'); 55 } 56 57 /** 58 * Test rdf_rdfa_attributes(). 59 */ 60 function testDrupalRdfaAttributes() { 61 // Same value as the one in the HTML tag (no callback function). 62 $expected_attributes = array( 63 'property' => array('dc:title'), 64 ); 65 $mapping = rdf_mapping_load('test_entity', 'test_bundle'); 66 $attributes = rdf_rdfa_attributes($mapping['title']); 67 ksort($expected_attributes); 68 ksort($attributes); 69 $this->assertEqual($expected_attributes, $attributes); 70 71 // Value different from the one in the HTML tag (callback function). 72 $date = 1252750327; 73 $isoDate = date('c', $date); 74 $expected_attributes = array( 75 'datatype' => 'xsd:dateTime', 76 'property' => array('dc:created'), 77 'content' => $isoDate, 78 ); 79 $mapping = rdf_mapping_load('test_entity', 'test_bundle'); 80 $attributes = rdf_rdfa_attributes($mapping['created'], $date); 81 ksort($expected_attributes); 82 ksort($attributes); 83 $this->assertEqual($expected_attributes, $attributes); 84 85 // Same value as the one in the HTML tag with datatype. 86 $expected_attributes = array( 87 'datatype' => 'foo:bar1type', 88 'property' => array('foo:bar1'), 89 ); 90 $mapping = rdf_mapping_load('test_entity', 'test_bundle'); 91 $attributes = rdf_rdfa_attributes($mapping['foobar1']); 92 ksort($expected_attributes); 93 ksort($attributes); 94 $this->assertEqual($expected_attributes, $attributes); 95 96 // ObjectProperty mapping (rel). 97 $expected_attributes = array( 98 'rel' => array('sioc:has_creator', 'dc:creator'), 99 ); 100 $mapping = rdf_mapping_load('test_entity', 'test_bundle'); 101 $attributes = rdf_rdfa_attributes($mapping['foobar_objproperty1']); 102 ksort($expected_attributes); 103 ksort($attributes); 104 $this->assertEqual($expected_attributes, $attributes); 105 106 // Inverse ObjectProperty mapping (rev). 107 $expected_attributes = array( 108 'rev' => array('sioc:reply_of'), 109 ); 110 $mapping = rdf_mapping_load('test_entity', 'test_bundle'); 111 $attributes = rdf_rdfa_attributes($mapping['foobar_objproperty2']); 112 ksort($expected_attributes); 113 ksort($attributes); 114 $this->assertEqual($expected_attributes, $attributes); 115 } 116 117 /** 118 * Ensure that file fields have the correct resource as the object in RDFa 119 * when displayed as a teaser. 120 */ 121 function testAttributesInMarkupFile() { 122 // Create a user to post the image. 123 $admin_user = $this->drupalCreateUser(array('edit own article content', 'revert revisions', 'administer content types')); 124 $this->drupalLogin($admin_user); 125 126 $langcode = LANGUAGE_NONE; 127 $bundle_name = "article"; 128 129 $field_name = 'file_test'; 130 $field = array( 131 'field_name' => $field_name, 132 'type' => 'file', 133 ); 134 field_create_field($field); 135 $instance = array( 136 'field_name' => $field_name, 137 'entity_type' => 'node', 138 'bundle' => $bundle_name, 139 'display' => array( 140 'teaser' => array( 141 'type' => 'file_default', 142 ), 143 ), 144 ); 145 field_create_instance($instance); 146 147 // Set the RDF mapping for the new field. 148 $rdf_mapping = rdf_mapping_load('node', $bundle_name); 149 $rdf_mapping += array($field_name => array('predicates' => array('rdfs:seeAlso'), 'type' => 'rel')); 150 $rdf_mapping_save = array('mapping' => $rdf_mapping, 'type' => 'node', 'bundle' => $bundle_name); 151 rdf_mapping_save($rdf_mapping_save); 152 153 // Get the test file that simpletest provides. 154 $file = current($this->drupalGetTestFiles('text')); 155 156 // Prepare image variables. 157 $image_field = "field_image"; 158 // Get the test image that simpletest provides. 159 $image = current($this->drupalGetTestFiles('image')); 160 161 // Create an array for drupalPost with the field names as the keys and 162 // the URIs for the test files as the values. 163 $edit = array("files[" . $field_name . "_" . $langcode . "_0]" => drupal_realpath($file->uri), 164 "files[" . $image_field . "_" . $langcode . "_0]" => drupal_realpath($image->uri)); 165 166 // Create node and save, then edit node to upload files. 167 $node = $this->drupalCreateNode(array('type' => 'article', 'promote' => 1)); 168 $this->drupalPost('node/' . $node->nid . '/edit', $edit, t('Save')); 169 170 // Get filenames and nid for comparison with HTML output. 171 $file_filename = $file->filename; 172 $image_filename = $image->filename; 173 $nid = $node->nid; 174 // Navigate to front page, where node is displayed in teaser form. 175 $this->drupalGet('node'); 176 177 // We only check to make sure that the resource attribute contains '.txt' 178 // instead of the full file name because the filename is altered on upload. 179 $file_rel = $this->xpath('//div[contains(@about, :node-uri)]//div[contains(@rel, "rdfs:seeAlso") and contains(@resource, ".txt")]', array( 180 ':node-uri' => 'node/' . $nid, 181 )); 182 $this->assertTrue(!empty($file_rel), "Attribute 'rel' set on file field. Attribute 'resource' is also set."); 183 $image_rel = $this->xpath('//div[contains(@about, :node-uri)]//div[contains(@rel, "rdfs:seeAlso") and contains(@resource, :image)]//img[contains(@typeof, "foaf:Image")]', array( 184 ':node-uri' => 'node/' . $nid, 185 ':image' => $image_filename, 186 )); 187 188 $this->assertTrue(!empty($image_rel), "Attribute 'rel' set on image field. Attribute 'resource' is also set."); 189 190 // Edits the node to add tags. 191 $tag1 = $this->randomName(8); 192 $tag2 = $this->randomName(8); 193 $edit = array(); 194 $edit['field_tags[' . LANGUAGE_NONE . ']'] = "$tag1, $tag2"; 195 $this->drupalPost('node/' . $node->nid . '/edit', $edit, t('Save')); 196 // Ensures the RDFa markup for the relationship between the node and its 197 // tags is correct. 198 $term_rdfa_meta = $this->xpath('//div[@about=:node-url and contains(@typeof, "sioc:Item") and contains(@typeof, "foaf:Document")]//ul[@class="links"]/li[@rel="dc:subject"]/a[@typeof="skos:Concept" and @datatype="" and text()=:term-name]', array( 199 ':node-url' => url('node/' . $node->nid), 200 ':term-name' => $tag1, 201 )); 202 $this->assertTrue(!empty($term_rdfa_meta), 'Property dc:subject is present for the tag1 field item.'); 203 $term_rdfa_meta = $this->xpath('//div[@about=:node-url and contains(@typeof, "sioc:Item") and contains(@typeof, "foaf:Document")]//ul[@class="links"]/li[@rel="dc:subject"]/a[@typeof="skos:Concept" and @datatype="" and text()=:term-name]', array( 204 ':node-url' => url('node/' . $node->nid), 205 ':term-name' => $tag2, 206 )); 207 $this->assertTrue(!empty($term_rdfa_meta), 'Property dc:subject is present for the tag2 field item.'); 208 } 209} 210 211class RdfCrudTestCase extends DrupalWebTestCase { 212 public static function getInfo() { 213 return array( 214 'name' => 'RDF mapping CRUD functions', 215 'description' => 'Test the RDF mapping CRUD functions.', 216 'group' => 'RDF', 217 ); 218 } 219 220 function setUp() { 221 parent::setUp('rdf', 'rdf_test'); 222 } 223 224 /** 225 * Test inserting, loading, updating, and deleting RDF mappings. 226 */ 227 function testCRUD() { 228 // Verify loading of a default mapping. 229 $mapping = _rdf_mapping_load('test_entity', 'test_bundle'); 230 $this->assertTrue(count($mapping), 'Default mapping was found.'); 231 232 // Verify saving a mapping. 233 $mapping = array( 234 'type' => 'crud_test_entity', 235 'bundle' => 'crud_test_bundle', 236 'mapping' => array( 237 'rdftype' => array('sioc:Post'), 238 'title' => array( 239 'predicates' => array('dc:title'), 240 ), 241 'uid' => array( 242 'predicates' => array('sioc:has_creator', 'dc:creator'), 243 'type' => 'rel', 244 ), 245 ), 246 ); 247 $this->assertTrue(rdf_mapping_save($mapping) === SAVED_NEW, 'Mapping was saved.'); 248 249 // Read the raw record from the {rdf_mapping} table. 250 $result = db_query('SELECT * FROM {rdf_mapping} WHERE type = :type AND bundle = :bundle', array(':type' => $mapping['type'], ':bundle' => $mapping['bundle'])); 251 $stored_mapping = $result->fetchAssoc(); 252 $stored_mapping['mapping'] = unserialize($stored_mapping['mapping']); 253 $this->assertEqual($mapping, $stored_mapping, 'Mapping was stored properly in the {rdf_mapping} table.'); 254 255 // Verify loading of saved mapping. 256 $this->assertEqual($mapping['mapping'], _rdf_mapping_load($mapping['type'], $mapping['bundle']), 'Saved mapping loaded successfully.'); 257 258 // Verify updating of mapping. 259 $mapping['mapping']['title'] = array( 260 'predicates' => array('dc2:bar2'), 261 ); 262 $this->assertTrue(rdf_mapping_save($mapping) === SAVED_UPDATED, 'Mapping was updated.'); 263 264 // Read the raw record from the {rdf_mapping} table. 265 $result = db_query('SELECT * FROM {rdf_mapping} WHERE type = :type AND bundle = :bundle', array(':type' => $mapping['type'], ':bundle' => $mapping['bundle'])); 266 $stored_mapping = $result->fetchAssoc(); 267 $stored_mapping['mapping'] = unserialize($stored_mapping['mapping']); 268 $this->assertEqual($mapping, $stored_mapping, 'Updated mapping was stored properly in the {rdf_mapping} table.'); 269 270 // Verify loading of saved mapping. 271 $this->assertEqual($mapping['mapping'], _rdf_mapping_load($mapping['type'], $mapping['bundle']), 'Saved mapping loaded successfully.'); 272 273 // Verify deleting of mapping. 274 $this->assertTrue(rdf_mapping_delete($mapping['type'], $mapping['bundle']), 'Mapping was deleted.'); 275 $this->assertFalse(_rdf_mapping_load($mapping['type'], $mapping['bundle']), 'Deleted mapping is no longer found in the database.'); 276 } 277} 278 279class RdfMappingDefinitionTestCase extends TaxonomyWebTestCase { 280 public static function getInfo() { 281 return array( 282 'name' => 'RDF mapping definition functionality', 283 'description' => 'Test the different types of RDF mappings and ensure the proper RDFa markup in included in nodes and user profile pages.', 284 'group' => 'RDF', 285 ); 286 } 287 288 function setUp() { 289 parent::setUp('rdf', 'rdf_test', 'blog'); 290 } 291 292 /** 293 * Create a node of type blog and test whether the RDF mapping defined for 294 * this node type in rdf_test.module is used in the node page. 295 */ 296 function testAttributesInMarkup1() { 297 $node = $this->drupalCreateNode(array('type' => 'blog')); 298 $isoDate = date('c', $node->changed); 299 $url = url('node/' . $node->nid); 300 $this->drupalGet('node/' . $node->nid); 301 302 // Ensure the default bundle mapping for node is used. These attributes come 303 // from the node default bundle definition. 304 $blog_title = $this->xpath("//div[@about='$url']/span[@property='dc:title' and @content='$node->title']"); 305 $blog_meta = $this->xpath("//div[(@about='$url') and (@typeof='sioct:Weblog')]//span[contains(@property, 'dc:date') and contains(@property, 'dc:created') and @datatype='xsd:dateTime' and @content='$isoDate']"); 306 $this->assertTrue(!empty($blog_title), 'Property dc:title is present in meta tag.'); 307 $this->assertTrue(!empty($blog_meta), 'RDF type is present on post. Properties dc:date and dc:created are present on post date.'); 308 } 309 310 /** 311 * Create a content type and a node of type test_bundle_hook_install and test 312 * whether the RDF mapping defined in rdf_test.install is used. 313 */ 314 function testAttributesInMarkup2() { 315 $type = $this->drupalCreateContentType(array('type' => 'test_bundle_hook_install')); 316 // Create node with single quotation mark title to ensure it does not get 317 // escaped more than once. 318 $node = $this->drupalCreateNode(array( 319 'type' => 'test_bundle_hook_install', 320 'title' => $this->randomName(8) . "'", 321 )); 322 $isoDate = date('c', $node->changed); 323 $url = url('node/' . $node->nid); 324 $this->drupalGet('node/' . $node->nid); 325 326 // Ensure the mapping defined in rdf_module.test is used. 327 $test_bundle_title = $this->xpath("//div[@about='$url']/span[@property='dc:title' and @content=\"$node->title\"]"); 328 $test_bundle_meta = $this->xpath("//div[(@about='$url') and contains(@typeof, 'foo:mapping_install1') and contains(@typeof, 'bar:mapping_install2')]//span[contains(@property, 'dc:date') and contains(@property, 'dc:created') and @datatype='xsd:dateTime' and @content='$isoDate']"); 329 $this->assertTrue(!empty($test_bundle_title), 'Property dc:title is present in meta tag.'); 330 $this->assertTrue(!empty($test_bundle_meta), 'RDF type is present on post. Properties dc:date and dc:created are present on post date.'); 331 } 332 333 /** 334 * Create a random content type and node and ensure the default mapping for 335 * node is used. 336 */ 337 function testAttributesInMarkup3() { 338 $type = $this->drupalCreateContentType(); 339 $node = $this->drupalCreateNode(array('type' => $type->type)); 340 $isoDate = date('c', $node->changed); 341 $url = url('node/' . $node->nid); 342 $this->drupalGet('node/' . $node->nid); 343 344 // Ensure the default bundle mapping for node is used. These attributes come 345 // from the node default bundle definition. 346 $random_bundle_title = $this->xpath("//div[@about='$url']/span[@property='dc:title' and @content='$node->title']"); 347 $random_bundle_meta = $this->xpath("//div[(@about='$url') and contains(@typeof, 'sioc:Item') and contains(@typeof, 'foaf:Document')]//span[contains(@property, 'dc:date') and contains(@property, 'dc:created') and @datatype='xsd:dateTime' and @content='$isoDate']"); 348 $this->assertTrue(!empty($random_bundle_title), 'Property dc:title is present in meta tag.'); 349 $this->assertTrue(!empty($random_bundle_meta), 'RDF type is present on post. Properties dc:date and dc:created are present on post date.'); 350 } 351 352 /** 353 * Create a random user and ensure the default mapping for user is used. 354 */ 355 function testUserAttributesInMarkup() { 356 // Create two users, one with access to user profiles. 357 $user1 = $this->drupalCreateUser(array('access user profiles')); 358 $user2 = $this->drupalCreateUser(); 359 $username = $user2->name; 360 $this->drupalLogin($user1); 361 // Browse to the user profile page. 362 $this->drupalGet('user/' . $user2->uid); 363 // Ensure the default bundle mapping for user is used on the user profile 364 // page. These attributes come from the user default bundle definition. 365 $account_uri = url('user/' . $user2->uid); 366 $person_uri = url('user/' . $user2->uid, array('fragment' => 'me')); 367 368 $user2_profile_about = $this->xpath('//div[@class="profile" and @typeof="sioc:UserAccount" and @about=:account-uri]', array( 369 ':account-uri' => $account_uri, 370 )); 371 $this->assertTrue(!empty($user2_profile_about), 'RDFa markup found on user profile page'); 372 373 $user_account_holder = $this->xpath('//meta[contains(@typeof, "foaf:Person") and @about=:person-uri and @resource=:account-uri and contains(@rel, "foaf:account")]', array( 374 ':person-uri' => $person_uri, 375 ':account-uri' => $account_uri, 376 )); 377 $this->assertTrue(!empty($user_account_holder), 'URI created for account holder and username set on sioc:UserAccount.'); 378 379 $user_username = $this->xpath('//meta[@about=:account-uri and contains(@property, "foaf:name") and @content=:username]', array( 380 ':account-uri' => $account_uri, 381 ':username' => $username, 382 )); 383 $this->assertTrue(!empty($user_username), 'foaf:name set on username.'); 384 385 // User 2 creates node. 386 $this->drupalLogin($user2); 387 $node = $this->drupalCreateNode(array('type' => 'article', 'promote' => 1)); 388 $this->drupalLogin($user1); 389 $this->drupalGet('node/' . $node->nid); 390 // Ensures the default bundle mapping for user is used on the Authored By 391 // information on the node. 392 $author_about = $this->xpath('//a[@typeof="sioc:UserAccount" and @about=:account-uri and @property="foaf:name" and @datatype="" and contains(@xml:lang, "")]', array( 393 ':account-uri' => $account_uri, 394 )); 395 $this->assertTrue(!empty($author_about), 'RDFa markup found on author information on post. xml:lang on username is set to empty string.'); 396 } 397 398 /** 399 * Creates a random term and ensures the right RDFa markup is used. 400 */ 401 function testTaxonomyTermRdfaAttributes() { 402 $vocabulary = $this->createVocabulary(); 403 $term = $this->createTerm($vocabulary); 404 405 // Views the term and checks that the RDFa markup is correct. 406 $this->drupalGet('taxonomy/term/' . $term->tid); 407 $term_url = url('taxonomy/term/' . $term->tid); 408 $term_name = $term->name; 409 $term_rdfa_meta = $this->xpath('//meta[@typeof="skos:Concept" and @about=:term-url and contains(@property, "rdfs:label") and contains(@property, "skos:prefLabel") and @content=:term-name]', array( 410 ':term-url' => $term_url, 411 ':term-name' => $term_name, 412 )); 413 $this->assertTrue(!empty($term_rdfa_meta), 'RDFa markup found on term page.'); 414 } 415} 416 417class RdfCommentAttributesTestCase extends CommentHelperCase { 418 419 public static function getInfo() { 420 return array( 421 'name' => 'RDF comment mapping', 422 'description' => 'Tests the RDFa markup of comments.', 423 'group' => 'RDF', 424 ); 425 } 426 427 public function setUp() { 428 parent::setUp('comment', 'rdf', 'rdf_test'); 429 430 $this->admin_user = $this->drupalCreateUser(array('administer content types', 'administer comments', 'administer permissions', 'administer blocks')); 431 $this->web_user = $this->drupalCreateUser(array('access comments', 'post comments', 'create article content', 'access user profiles')); 432 433 // Enables anonymous user comments. 434 user_role_change_permissions(DRUPAL_ANONYMOUS_RID, array( 435 'access comments' => TRUE, 436 'post comments' => TRUE, 437 'skip comment approval' => TRUE, 438 )); 439 // Allows anonymous to leave their contact information. 440 $this->setCommentAnonymous(COMMENT_ANONYMOUS_MAY_CONTACT); 441 $this->setCommentPreview(DRUPAL_OPTIONAL); 442 $this->setCommentForm(TRUE); 443 $this->setCommentSubject(TRUE); 444 $this->setCommentSettings('comment_default_mode', COMMENT_MODE_THREADED, 'Comment paging changed.'); 445 446 // Creates the nodes on which the test comments will be posted. 447 $this->drupalLogin($this->web_user); 448 $this->node1 = $this->drupalCreateNode(array('type' => 'article', 'promote' => 1)); 449 $this->node2 = $this->drupalCreateNode(array('type' => 'article', 'promote' => 1)); 450 $this->drupalLogout(); 451 } 452 453 /** 454 * Tests the presence of the RDFa markup for the number of comments. 455 */ 456 public function testNumberOfCommentsRdfaMarkup() { 457 // Posts 2 comments as a registered user. 458 $this->drupalLogin($this->web_user); 459 $this->postComment($this->node1, $this->randomName(), $this->randomName()); 460 $this->postComment($this->node1, $this->randomName(), $this->randomName()); 461 462 // Tests number of comments in teaser view. 463 $this->drupalGet('node'); 464 $node_url = url('node/' . $this->node1->nid); 465 $comment_count_teaser = $this->xpath('//div[@about=:node-url]/span[@property="sioc:num_replies" and @content="2" and @datatype="xsd:integer"]', array(':node-url' => $node_url)); 466 $this->assertTrue(!empty($comment_count_teaser), 'RDFa markup for the number of comments found on teaser view.'); 467 468 // Tests number of comments in full node view. 469 $this->drupalGet('node/' . $this->node1->nid); 470 $comment_count_teaser = $this->xpath('//div[@about=:node-url]/span[@property="sioc:num_replies" and @content="2" and @datatype="xsd:integer"]', array(':node-url' => $node_url)); 471 $this->assertTrue(!empty($comment_count_teaser), 'RDFa markup for the number of comments found on full node view.'); 472 } 473 474 /** 475 * Tests the presence of the RDFa markup for the title, date and author and 476 * homepage on registered users and anonymous comments. 477 */ 478 public function testCommentRdfaMarkup() { 479 480 // Posts comment #1 as a registered user. 481 $this->drupalLogin($this->web_user); 482 $comment1_subject = $this->randomName(); 483 $comment1_body = $this->randomName(); 484 $comment1 = $this->postComment($this->node1, $comment1_body, $comment1_subject); 485 486 // Tests comment #1 with access to the user profile. 487 $this->drupalGet('node/' . $this->node1->nid); 488 $this->_testBasicCommentRdfaMarkup($comment1); 489 490 // Tests comment #1 with no access to the user profile (as anonymous user). 491 $this->drupalLogout(); 492 $this->drupalGet('node/' . $this->node1->nid); 493 $this->_testBasicCommentRdfaMarkup($comment1); 494 495 // Posts comment #2 as anonymous user. 496 $comment2_subject = $this->randomName(); 497 $comment2_body = $this->randomName(); 498 $anonymous_user = array(); 499 $anonymous_user['name'] = $this->randomName(); 500 $anonymous_user['mail'] = 'tester@simpletest.org'; 501 $anonymous_user['homepage'] = 'http://example.org/'; 502 $comment2 = $this->postComment($this->node2, $comment2_body, $comment2_subject, $anonymous_user); 503 $this->drupalGet('node/' . $this->node2->nid); 504 505 // Tests comment #2 as anonymous user. 506 $this->_testBasicCommentRdfaMarkup($comment2, $anonymous_user); 507 // Tests the RDFa markup for the homepage (specific to anonymous comments). 508 $comment_homepage = $this->xpath('//div[contains(@class, "comment") and contains(@typeof, "sioct:Comment")]//span[@rel="sioc:has_creator"]/a[contains(@class, "username") and @typeof="sioc:UserAccount" and @property="foaf:name" and @datatype="" and @href="http://example.org/" and contains(@rel, "foaf:page")]'); 509 $this->assertTrue(!empty($comment_homepage), 'RDFa markup for the homepage of anonymous user found.'); 510 // There should be no about attribute on anonymous comments. 511 $comment_homepage = $this->xpath('//div[contains(@class, "comment") and contains(@typeof, "sioct:Comment")]//span[@rel="sioc:has_creator"]/a[@about]'); 512 $this->assertTrue(empty($comment_homepage), 'No about attribute is present on anonymous user comment.'); 513 514 // Tests comment #2 as logged in user. 515 $this->drupalLogin($this->web_user); 516 $this->drupalGet('node/' . $this->node2->nid); 517 $this->_testBasicCommentRdfaMarkup($comment2, $anonymous_user); 518 // Tests the RDFa markup for the homepage (specific to anonymous comments). 519 $comment_homepage = $this->xpath('//div[contains(@class, "comment") and contains(@typeof, "sioct:Comment")]//span[@rel="sioc:has_creator"]/a[contains(@class, "username") and @typeof="sioc:UserAccount" and @property="foaf:name" and @datatype="" and @href="http://example.org/" and contains(@rel, "foaf:page")]'); 520 $this->assertTrue(!empty($comment_homepage), "RDFa markup for the homepage of anonymous user found."); 521 // There should be no about attribute on anonymous comments. 522 $comment_homepage = $this->xpath('//div[contains(@class, "comment") and contains(@typeof, "sioct:Comment")]//span[@rel="sioc:has_creator"]/a[@about]'); 523 $this->assertTrue(empty($comment_homepage), "No about attribute is present on anonymous user comment."); 524 } 525 526 /** 527 * Test RDF comment replies. 528 */ 529 public function testCommentReplyOfRdfaMarkup() { 530 // Posts comment #1 as a registered user. 531 $this->drupalLogin($this->web_user); 532 $comments[] = $this->postComment($this->node1, $this->randomName(), $this->randomName()); 533 534 // Tests the reply_of relationship of a first level comment. 535 $result = $this->xpath("(id('comments')//div[contains(@class,'comment ')])[position()=1]//span[@rel='sioc:reply_of' and @resource=:node]", array(':node' => url("node/{$this->node1->nid}"))); 536 $this->assertEqual(1, count($result), 'RDFa markup referring to the node is present.'); 537 $result = $this->xpath("(id('comments')//div[contains(@class,'comment ')])[position()=1]//span[@rel='sioc:reply_of' and @resource=:comment]", array(':comment' => url('comment/1#comment-1'))); 538 $this->assertFalse($result, 'No RDFa markup referring to the comment itself is present.'); 539 540 // Posts a reply to the first comment. 541 $this->drupalGet('comment/reply/' . $this->node1->nid . '/' . $comments[0]->id); 542 $comments[] = $this->postComment(NULL, $this->randomName(), $this->randomName(), TRUE); 543 544 // Tests the reply_of relationship of a second level comment. 545 $result = $this->xpath("(id('comments')//div[contains(@class,'comment ')])[position()=2]//span[@rel='sioc:reply_of' and @resource=:node]", array(':node' => url("node/{$this->node1->nid}"))); 546 $this->assertEqual(1, count($result), 'RDFa markup referring to the node is present.'); 547 $result = $this->xpath("(id('comments')//div[contains(@class,'comment ')])[position()=2]//span[@rel='sioc:reply_of' and @resource=:comment]", array(':comment' => url('comment/1', array('fragment' => 'comment-1')))); 548 $this->assertEqual(1, count($result), 'RDFa markup referring to the parent comment is present.'); 549 $comments = $this->xpath("(id('comments')//div[contains(@class,'comment ')])[position()=2]"); 550 } 551 552 /** 553 * Helper function for testCommentRdfaMarkup(). 554 * 555 * Tests the current page for basic comment RDFa markup. 556 * 557 * @param $comment 558 * Comment object. 559 * @param $account 560 * An array containing information about an anonymous user. 561 */ 562 function _testBasicCommentRdfaMarkup($comment, $account = array()) { 563 $comment_container = $this->xpath('//div[contains(@class, "comment") and contains(@typeof, "sioct:Comment")]'); 564 $this->assertTrue(!empty($comment_container), "Comment RDF type for comment found."); 565 $comment_title = $this->xpath('//div[contains(@class, "comment") and contains(@typeof, "sioct:Comment")]//h3[@property="dc:title"]'); 566 $this->assertEqual((string)$comment_title[0]->a, $comment->subject, "RDFa markup for the comment title found."); 567 $comment_date = $this->xpath('//div[contains(@class, "comment") and contains(@typeof, "sioct:Comment")]//*[contains(@property, "dc:date") and contains(@property, "dc:created")]'); 568 $this->assertTrue(!empty($comment_date), "RDFa markup for the date of the comment found."); 569 // The author tag can be either a or span 570 $comment_author = $this->xpath('//div[contains(@class, "comment") and contains(@typeof, "sioct:Comment")]//span[@rel="sioc:has_creator"]/*[contains(@class, "username") and @typeof="sioc:UserAccount" and @property="foaf:name" and @datatype=""]'); 571 $name = empty($account["name"]) ? $this->web_user->name : $account["name"] . " (not verified)"; 572 $this->assertEqual((string)$comment_author[0], $name, "RDFa markup for the comment author found."); 573 $comment_body = $this->xpath('//div[contains(@class, "comment") and contains(@typeof, "sioct:Comment")]//div[@class="content"]//div[contains(@class, "comment-body")]//div[@property="content:encoded"]'); 574 $this->assertEqual((string)$comment_body[0]->p, $comment->comment, "RDFa markup for the comment body found."); 575 } 576} 577 578class RdfTrackerAttributesTestCase extends DrupalWebTestCase { 579 public static function getInfo() { 580 return array( 581 'name' => 'RDF tracker page mapping', 582 'description' => 'Test the mapping for the tracker page and ensure the proper RDFa markup in included.', 583 'group' => 'RDF', 584 ); 585 } 586 587 function setUp() { 588 parent::setUp('rdf', 'rdf_test', 'tracker'); 589 // Enable anonymous posting of content. 590 user_role_change_permissions(DRUPAL_ANONYMOUS_RID, array( 591 'create article content' => TRUE, 592 'access comments' => TRUE, 593 'post comments' => TRUE, 594 'skip comment approval' => TRUE, 595 )); 596 } 597 598 /** 599 * Create nodes as both admin and anonymous user and test for correct RDFa 600 * markup on the tracker page for those nodes and their comments. 601 */ 602 function testAttributesInTracker() { 603 // Create node as anonymous user. 604 $node_anon = $this->drupalCreateNode(array('type' => 'article', 'uid' => 0)); 605 // Create node as admin user. 606 $node_admin = $this->drupalCreateNode(array('type' => 'article', 'uid' => 1)); 607 608 // Pass both the anonymously posted node and the administrator posted node 609 // through to test for the RDF attributes. 610 $this->_testBasicTrackerRdfaMarkup($node_anon); 611 $this->_testBasicTrackerRdfaMarkup($node_admin); 612 613 } 614 615 /** 616 * Helper function for testAttributesInTracker(). 617 * 618 * Tests the tracker page for RDFa markup. 619 * 620 * @param $node 621 * The node just created. 622 */ 623 function _testBasicTrackerRdfaMarkup($node) { 624 $url = url('node/' . $node->nid); 625 626 $user = ($node->uid == 0) ? 'Anonymous user' : 'Registered user'; 627 628 // Navigate to tracker page. 629 $this->drupalGet('tracker'); 630 631 // Tests whether the about property is applied. This is implicit in the 632 // success of the following tests, but making it explicit will make 633 // debugging easier in case of failure. 634 $tracker_about = $this->xpath('//tr[@about=:url]', array(':url' => $url)); 635 $this->assertTrue(!empty($tracker_about), format_string('About attribute found on table row for @user content.', array('@user'=> $user))); 636 637 // Tests whether the title has the correct property attribute. 638 $tracker_title = $this->xpath('//tr[@about=:url]/td[@property="dc:title" and @datatype=""]', array(':url' => $url)); 639 $this->assertTrue(!empty($tracker_title), format_string('Title property attribute found on @user content.', array('@user'=> $user))); 640 641 // Tests whether the relationship between the content and user has been set. 642 $tracker_user = $this->xpath('//tr[@about=:url]//td[contains(@rel, "sioc:has_creator")]//*[contains(@typeof, "sioc:UserAccount") and contains(@property, "foaf:name")]', array(':url' => $url)); 643 $this->assertTrue(!empty($tracker_user), format_string('Typeof and name property attributes found on @user.', array('@user'=> $user))); 644 // There should be an about attribute on logged in users and no about 645 // attribute for anonymous users. 646 $tracker_user = $this->xpath('//tr[@about=:url]//td[@rel="sioc:has_creator"]/*[@about]', array(':url' => $url)); 647 if ($node->uid == 0) { 648 $this->assertTrue(empty($tracker_user), format_string('No about attribute is present on @user.', array('@user'=> $user))); 649 } 650 elseif ($node->uid > 0) { 651 $this->assertTrue(!empty($tracker_user), format_string('About attribute is present on @user.', array('@user'=> $user))); 652 } 653 654 // Tests whether the property has been set for number of comments. 655 $tracker_replies = $this->xpath('//tr[@about=:url]//td[contains(@property, "sioc:num_replies") and contains(@content, "0") and @datatype="xsd:integer"]', array(':url' => $url)); 656 $this->assertTrue($tracker_replies, format_string('Num replies property and content attributes found on @user content.', array('@user'=> $user))); 657 658 // Tests that the appropriate RDFa markup to annotate the latest activity 659 // date has been added to the tracker output before comments have been 660 // posted, meaning the latest activity reflects changes to the node itself. 661 $isoDate = date('c', $node->changed); 662 $tracker_activity = $this->xpath('//tr[@about=:url]//td[contains(@property, "dc:modified") and contains(@property, "sioc:last_activity_date") and contains(@datatype, "xsd:dateTime") and @content=:date]', array(':url' => $url, ':date' => $isoDate)); 663 $this->assertTrue(!empty($tracker_activity), format_string('Latest activity date and changed properties found when there are no comments on @user content. Latest activity date content is correct.', array('@user'=> $user))); 664 665 // Tests that the appropriate RDFa markup to annotate the latest activity 666 // date has been added to the tracker output after a comment is posted. 667 $comment = array( 668 'subject' => $this->randomName(), 669 'comment_body[' . LANGUAGE_NONE . '][0][value]' => $this->randomName(), 670 ); 671 $this->drupalPost('comment/reply/' . $node->nid, $comment, t('Save')); 672 $this->drupalGet('tracker'); 673 674 // Tests whether the property has been set for number of comments. 675 $tracker_replies = $this->xpath('//tr[@about=:url]//td[contains(@property, "sioc:num_replies") and contains(@content, "1") and @datatype="xsd:integer"]', array(':url' => $url)); 676 $this->assertTrue($tracker_replies, format_string('Num replies property and content attributes found on @user content.', array('@user'=> $user))); 677 678 // Need to query database directly to obtain last_activity_date because 679 // it cannot be accessed via node_load(). 680 $result = db_query('SELECT t.changed FROM {tracker_node} t WHERE t.nid = (:nid)', array(':nid' => $node->nid)); 681 foreach ($result as $node) { 682 $expected_last_activity_date = $node->changed; 683 } 684 $isoDate = date('c', $expected_last_activity_date); 685 $tracker_activity = $this->xpath('//tr[@about=:url]//td[@property="sioc:last_activity_date" and @datatype="xsd:dateTime" and @content=:date]', array(':url' => $url, ':date' => $isoDate)); 686 $this->assertTrue(!empty($tracker_activity), format_string('Latest activity date found when there are comments on @user content. Latest activity date content is correct.', array('@user'=> $user))); 687 } 688} 689 690/** 691 * Tests for RDF namespaces declaration with hook_rdf_namespaces(). 692 */ 693class RdfGetRdfNamespacesTestCase extends DrupalWebTestCase { 694 public static function getInfo() { 695 return array( 696 'name' => 'RDF namespaces', 697 'description' => 'Test hook_rdf_namespaces() and ensure only "safe" namespaces are returned.', 698 'group' => 'RDF', 699 ); 700 } 701 702 function setUp() { 703 parent::setUp('rdf', 'rdf_test'); 704 } 705 706 /** 707 * Test getting RDF namesapces. 708 */ 709 function testGetRdfNamespaces() { 710 // Get all RDF namespaces. 711 $ns = rdf_get_namespaces(); 712 713 $this->assertEqual($ns['rdfs'], 'http://www.w3.org/2000/01/rdf-schema#', 'A prefix declared once is included.'); 714 $this->assertEqual($ns['foaf'], 'http://xmlns.com/foaf/0.1/', 'The same prefix declared in several implementations of hook_rdf_namespaces() is valid as long as all the namespaces are the same.'); 715 $this->assertEqual($ns['foaf1'], 'http://xmlns.com/foaf/0.1/', 'Two prefixes can be assigned the same namespace.'); 716 $this->assertTrue(!isset($ns['dc']), 'A prefix with conflicting namespaces is discarded.'); 717 } 718} 719