Project

General

Profile

Paste
Download (43.9 KB) Statistics
| Branch: | Revision:

root / drupal7 / sites / all / modules / captcha / captcha.test @ a8cee257

1
<?php
2

    
3
/**
4
 * @file
5
 * Tests for CAPTCHA module.
6
 */
7

    
8
// TODO: write test for CAPTCHAs on admin pages
9
// TODO: test for default challenge type
10
// TODO: test about placement (comment form, node forms, log in form, etc)
11
// TODO: test if captcha_cron does it work right
12
// TODO: test custom CAPTCHA validation stuff
13
// TODO: test if entry on status report (Already X blocked form submissions) works
14
// TODO: test space ignoring validation of image CAPTCHA
15

    
16
// TODO: refactor the 'comment_body[' . LANGUAGE_NONE . '][0][value]' stuff
17

    
18
// Some constants for better reuse.
19
define('CAPTCHA_WRONG_RESPONSE_ERROR_MESSAGE',
20
  'The answer you entered for the CAPTCHA was not correct.');
21

    
22
define('CAPTCHA_SESSION_REUSE_ATTACK_ERROR_MESSAGE',
23
  'CAPTCHA session reuse attack detected.');
24

    
25
define('CAPTCHA_UNKNOWN_CSID_ERROR_MESSAGE',
26
  'CAPTCHA validation error: unknown CAPTCHA session ID. Contact the site administrator if this problem persists.');
27

    
28

    
29

    
30
/**
31
 * Base class for CAPTCHA tests.
32
 *
33
 * Provides common setup stuff and various helper functions
34
 */
35
abstract class CaptchaBaseWebTestCase extends DrupalWebTestCase {
36

    
37
  /**
38
   * User with various administrative permissions.
39
   * @var Drupal user
40
   */
41
  protected $admin_user;
42

    
43
  /**
44
   * Normal visitor with limited permissions
45
   * @var Drupal user;
46
   */
47
  protected $normal_user;
48

    
49
  /**
50
   * Form ID of comment form on standard (page) node
51
   * @var string
52
   */
53
  const COMMENT_FORM_ID = 'comment_node_page_form';
54

    
55
  /**
56
   * Drupal path of the (general) CAPTCHA admin page
57
   */
58
  const CAPTCHA_ADMIN_PATH = 'admin/config/people/captcha';
59

    
60

    
61
  function setUp() {
62
    // Load two modules: the captcha module itself and the comment module for testing anonymous comments.
63
    $modules = func_get_args();
64
    if (isset($modules[0]) && is_array($modules[0])) {
65
      $modules = $modules[0];
66
    }
67
    parent::setUp(array_merge(array('captcha', 'comment'), $modules));
68
    module_load_include('inc', 'captcha');
69

    
70
    // Create a normal user.
71
    $permissions = array(
72
      'access comments', 'post comments', 'skip comment approval',
73
      'access content', 'create page content', 'edit own page content',
74
    );
75
    $this->normal_user = $this->drupalCreateUser($permissions);
76

    
77
    // Create an admin user.
78
    $permissions[] = 'administer CAPTCHA settings';
79
    $permissions[] = 'skip CAPTCHA';
80
    $permissions[] = 'administer permissions';
81
    $permissions[] = 'administer content types';
82
    $this->admin_user = $this->drupalCreateUser($permissions);
83

    
84
    // Put comments on page nodes on a separate page (default in D7: below post).
85
    variable_set('comment_form_location_page', COMMENT_FORM_SEPARATE_PAGE);
86

    
87
  }
88

    
89
  /**
90
   * Assert that the response is accepted:
91
   * no "unknown CSID" message, no "CSID reuse attack detection" message,
92
   * no "wrong answer" message.
93
   */
94
  protected function assertCaptchaResponseAccepted() {
95
    // There should be no error message about unknown CAPTCHA session ID.
96
    $this->assertNoText(t(CAPTCHA_UNKNOWN_CSID_ERROR_MESSAGE),
97
      'CAPTCHA response should be accepted (known CSID).',
98
      'CAPTCHA');
99
    // There should be no error message about CSID reuse attack.
100
    $this->assertNoText(t(CAPTCHA_SESSION_REUSE_ATTACK_ERROR_MESSAGE),
101
      'CAPTCHA response should be accepted (no CAPTCHA session reuse attack detection).',
102
      'CAPTCHA');
103
    // There should be no error message about wrong response.
104
    $this->assertNoText(t(CAPTCHA_WRONG_RESPONSE_ERROR_MESSAGE),
105
      'CAPTCHA response should be accepted (correct response).',
106
      'CAPTCHA');
107
  }
108

    
109
  /**
110
   * Assert that there is a CAPTCHA on the form or not.
111
   * @param bool $presence whether there should be a CAPTCHA or not.
112
   */
113
  protected function assertCaptchaPresence($presence) {
114
    if ($presence) {
115
      $this->assertText(_captcha_get_description(),
116
        'There should be a CAPTCHA on the form.', 'CAPTCHA');
117
    }
118
    else {
119
      $this->assertNoText(_captcha_get_description(),
120
        'There should be no CAPTCHA on the form.', 'CAPTCHA');
121
    }
122
  }
123

    
124
  /**
125
   * Helper function to create a node with comments enabled.
126
   *
127
   * @return
128
   *   Created node object.
129
   */
130
  protected function createNodeWithCommentsEnabled($type='page') {
131
    $node_settings = array(
132
      'type' => $type,
133
      'comment' => COMMENT_NODE_OPEN,
134
    );
135
    $node = $this->drupalCreateNode($node_settings);
136
    return $node;
137
  }
138

    
139
  /**
140
   * Helper function to generate a form values array for comment forms
141
   */
142
  protected function getCommentFormValues() {
143
    $edit = array(
144
      'subject' => 'comment_subject ' . $this->randomName(32),
145
      'comment_body[' . LANGUAGE_NONE . '][0][value]' => 'comment_body ' . $this->randomName(256),
146
    );
147
    return $edit;
148
  }
149

    
150
  /**
151
   * Helper function to generate a form values array for node forms
152
   */
153
  protected function getNodeFormValues() {
154
    $edit = array(
155
      'title' => 'node_title ' . $this->randomName(32),
156
      'body[' . LANGUAGE_NONE . '][0][value]' => 'node_body ' . $this->randomName(256),
157
    );
158
    return $edit;
159
  }
160

    
161

    
162
  /**
163
   * Get the CAPTCHA session id from the current form in the browser.
164
   */
165
  protected function getCaptchaSidFromForm() {
166
    $elements = $this->xpath('//input[@name="captcha_sid"]');
167
    $captcha_sid = (int) $elements[0]['value'];
168
    return $captcha_sid;
169
  }
170
  /**
171
   * Get the CAPTCHA token from the current form in the browser.
172
   */
173
  protected function getCaptchaTokenFromForm() {
174
    $elements = $this->xpath('//input[@name="captcha_token"]');
175
    $captcha_token = (int) $elements[0]['value'];
176
    return $captcha_token;
177
  }
178

    
179
  /**
180
   * Get the solution of the math CAPTCHA from the current form in the browser.
181
   */
182
  protected function getMathCaptchaSolutionFromForm() {
183
    // Get the math challenge.
184
    $elements = $this->xpath('//div[@class="form-item form-type-textfield form-item-captcha-response"]/span[@class="field-prefix"]');
185
    $challenge = (string) $elements[0];
186
    // Extract terms and operator from challenge.
187
    $matches = array();
188
    $ret = preg_match('/\\s*(\\d+)\\s*(-|\\+)\\s*(\\d+)\\s*=\\s*/', $challenge, $matches);
189
    // Solve the challenge
190
    $a = (int) $matches[1];
191
    $b = (int) $matches[3];
192
    $solution = $matches[2] == '-' ? $a - $b : $a + $b;
193
    return $solution;
194
  }
195

    
196
  /**
197
   * Helper function to allow comment posting for anonymous users.
198
   */
199
  protected function allowCommentPostingForAnonymousVisitors() {
200
    // Log in as admin.
201
    $this->drupalLogin($this->admin_user);
202
    // Post user permissions form
203
    $edit = array(
204
      '1[access comments]' => true,
205
      '1[post comments]' => true,
206
      '1[skip comment approval]' => true,
207
    );
208
    $this->drupalPost('admin/people/permissions', $edit, 'Save permissions');
209
    $this->assertText('The changes have been saved.');
210
    // Log admin out
211
    $this->drupalLogout();
212
  }
213

    
214
}
215

    
216

    
217

    
218
class CaptchaTestCase extends CaptchaBaseWebTestCase {
219

    
220
  public static function getInfo() {
221
    return array(
222
      'name' => t('General CAPTCHA functionality'),
223
      'description' => t('Testing of the basic CAPTCHA functionality.'),
224
      'group' => t('CAPTCHA'),
225
    );
226
  }
227

    
228
  /**
229
   * Testing the protection of the user log in form.
230
   */
231
  function testCaptchaOnLoginForm() {
232
    // Create user and test log in without CAPTCHA.
233
    $user = $this->drupalCreateUser();
234
    $this->drupalLogin($user);
235
    // Log out again.
236
    $this->drupalLogout();
237

    
238
    // Set a CAPTCHA on login form
239
    captcha_set_form_id_setting('user_login', 'captcha/Math');
240

    
241
    // Check if there is a CAPTCHA on the login form (look for the title).
242
    $this->drupalGet('user');
243
    $this->assertCaptchaPresence(TRUE);
244

    
245
    // Try to log in, which should fail.
246
    $edit = array(
247
      'name' => $user->name,
248
      'pass' => $user->pass_raw,
249
      'captcha_response' => '?',
250
    );
251
    $this->drupalPost('user', $edit, t('Log in'));
252
    // Check for error message.
253
    $this->assertText(t(CAPTCHA_WRONG_RESPONSE_ERROR_MESSAGE),
254
      'CAPTCHA should block user login form', 'CAPTCHA');
255

    
256
    // And make sure that user is not logged in: check for name and password fields on ?q=user
257
    $this->drupalGet('user');
258
    $this->assertField('name', t('Username field found.'), 'CAPTCHA');
259
    $this->assertField('pass', t('Password field found.'), 'CAPTCHA');
260

    
261
  }
262

    
263

    
264
  /**
265
   * Assert function for testing if comment posting works as it should.
266
   *
267
   * Creates node with comment writing enabled, tries to post comment
268
   * with given CAPTCHA response (caller should enable the desired
269
   * challenge on page node comment forms) and checks if the result is as expected.
270
   *
271
   * @param $captcha_response the response on the CAPTCHA
272
   * @param $should_pass boolean describing if the posting should pass or should be blocked
273
   * @param $message message to prefix to nested asserts
274
   */
275
  protected function assertCommentPosting($captcha_response, $should_pass, $message) {
276
    // Make sure comments on pages can be saved directely without preview.
277
    variable_set('comment_preview_page', DRUPAL_OPTIONAL);
278

    
279
    // Create a node with comments enabled.
280
    $node = $this->createNodeWithCommentsEnabled();
281

    
282
    // Post comment on node.
283
    $edit = $this->getCommentFormValues();
284
    $comment_subject = $edit['subject'];
285
    $comment_body = $edit['comment_body[' . LANGUAGE_NONE . '][0][value]'];
286
    $edit['captcha_response'] = $captcha_response;
287
    $this->drupalPost('comment/reply/' . $node->nid, $edit, t('Save'));
288

    
289
    if ($should_pass) {
290
      // There should be no error message.
291
      $this->assertCaptchaResponseAccepted();
292
      // Get node page and check that comment shows up.
293
      $this->drupalGet('node/' . $node->nid);
294
      $this->assertText($comment_subject, $message .' Comment should show up on node page.', 'CAPTCHA');
295
      $this->assertText($comment_body, $message . ' Comment should show up on node page.', 'CAPTCHA');
296
    }
297
    else {
298
      // Check for error message.
299
      $this->assertText(t(CAPTCHA_WRONG_RESPONSE_ERROR_MESSAGE), $message .' Comment submission should be blocked.', 'CAPTCHA');
300
      // Get node page and check that comment is not present.
301
      $this->drupalGet('node/' . $node->nid);
302
      $this->assertNoText($comment_subject, $message .' Comment should not show up on node page.', 'CAPTCHA');
303
      $this->assertNoText($comment_body, $message . ' Comment should not show up on node page.', 'CAPTCHA');
304
    }
305
  }
306

    
307
  /*
308
   * Testing the case sensistive/insensitive validation.
309
   */
310
  function testCaseInsensitiveValidation() {
311
    // Set Test CAPTCHA on comment form
312
    captcha_set_form_id_setting(self::COMMENT_FORM_ID, 'captcha/Test');
313

    
314
    // Log in as normal user.
315
    $this->drupalLogin($this->normal_user);
316

    
317
    // Test case sensitive posting.
318
    variable_set('captcha_default_validation', CAPTCHA_DEFAULT_VALIDATION_CASE_SENSITIVE);
319
    $this->assertCommentPosting('Test 123', TRUE, 'Case sensitive validation of right casing.');
320
    $this->assertCommentPosting('test 123', FALSE, 'Case sensitive validation of wrong casing.');
321
    $this->assertCommentPosting('TEST 123', FALSE, 'Case sensitive validation of wrong casing.');
322

    
323
    // Test case insensitive posting (the default)
324
    variable_set('captcha_default_validation', CAPTCHA_DEFAULT_VALIDATION_CASE_INSENSITIVE);
325
    $this->assertCommentPosting('Test 123', TRUE, 'Case insensitive validation of right casing.');
326
    $this->assertCommentPosting('test 123', TRUE, 'Case insensitive validation of wrong casing.');
327
    $this->assertCommentPosting('TEST 123', TRUE, 'Case insensitive validation of wrong casing.');
328

    
329
  }
330

    
331
  /**
332
   * Test if the CAPTCHA description is only shown if there are challenge widgets to show.
333
   * For example, when a comment is previewed with correct CAPTCHA answer,
334
   * a challenge is generated and added to the form but removed in the pre_render phase.
335
   * The CAPTCHA description should not show up either.
336
   *
337
   * \see testCaptchaSessionReuseOnNodeForms()
338
   */
339
  function testCaptchaDescriptionAfterCommentPreview() {
340
    // Set Test CAPTCHA on comment form.
341
    captcha_set_form_id_setting(self::COMMENT_FORM_ID, 'captcha/Test');
342

    
343
    // Log in as normal user.
344
    $this->drupalLogin($this->normal_user);
345

    
346
    // Create a node with comments enabled.
347
    $node = $this->createNodeWithCommentsEnabled();
348

    
349
    // Preview comment with correct CAPTCHA answer.
350
    $edit = $this->getCommentFormValues();
351
    $edit['captcha_response'] = 'Test 123';
352
    $this->drupalPost('comment/reply/' . $node->nid, $edit, t('Preview'));
353

    
354
    // Check that there is no CAPTCHA after preview.
355
    $this->assertCaptchaPresence(FALSE);
356
  }
357

    
358
  /**
359
   * Test if the CAPTCHA session ID is reused when previewing nodes:
360
   * node preview after correct response should not show CAPTCHA anymore.
361
   * The preview functionality of comments and nodes works slightly different under the hood.
362
   * CAPTCHA module should be able to handle both.
363
   *
364
   * \see testCaptchaDescriptionAfterCommentPreview()
365
   */
366
  function testCaptchaSessionReuseOnNodeForms() {
367
    // Set Test CAPTCHA on page form.
368
    captcha_set_form_id_setting('page_node_form', 'captcha/Test');
369

    
370
    // Log in as normal user.
371
    $this->drupalLogin($this->normal_user);
372

    
373
    // Page settings to post, with correct CAPTCHA answer.
374
    $edit = $this->getNodeFormValues();
375
    $edit['captcha_response'] = 'Test 123';
376
    // Preview the node
377
    $this->drupalPost('node/add/page', $edit, t('Preview'));
378

    
379
    // Check that there is no CAPTCHA after preview.
380
    $this->assertCaptchaPresence(FALSE);
381
  }
382

    
383

    
384
  /**
385
   * CAPTCHA should also be put on admin pages even if visitor
386
   * has no access
387
   */
388
  function testCaptchaOnLoginBlockOnAdminPagesIssue893810() {
389
    // Set a CAPTCHA on login block form
390
    captcha_set_form_id_setting('user_login_block', 'captcha/Math');
391

    
392
    // Check if there is a CAPTCHA on home page.
393
    $this->drupalGet('node');
394
    $this->assertCaptchaPresence(TRUE);
395

    
396
    // Check there is a CAPTCHA on "forbidden" admin pages
397
    $this->drupalGet('admin');
398
    $this->assertCaptchaPresence(TRUE);
399
  }
400

    
401
}
402

    
403

    
404
class CaptchaAdminTestCase extends CaptchaBaseWebTestCase {
405

    
406
  public static function getInfo() {
407
    return array(
408
      'name' => t('CAPTCHA administration functionality'),
409
      'description' => t('Testing of the CAPTCHA administration interface and functionality.'),
410
      'group' => t('CAPTCHA'),
411
    );
412
  }
413

    
414
  /**
415
   * Test access to the admin pages.
416
   */
417
  function testAdminAccess() {
418
    $this->drupalLogin($this->normal_user);
419
    $this->drupalGet(self::CAPTCHA_ADMIN_PATH);
420
    $this->assertText(t('Access denied'), 'Normal users should not be able to access the CAPTCHA admin pages', 'CAPTCHA');
421

    
422
    $this->drupalLogin($this->admin_user);
423
    $this->drupalGet(self::CAPTCHA_ADMIN_PATH);
424
    $this->assertNoText(t('Access denied'), 'Admin users should be able to access the CAPTCHA admin pages', 'CAPTCHA');
425
  }
426

    
427
  /**
428
   * Test the CAPTCHA point setting getter/setter.
429
   */
430
  function testCaptchaPointSettingGetterAndSetter() {
431
    $comment_form_id = self::COMMENT_FORM_ID;
432
    // Set to 'none'.
433
    captcha_set_form_id_setting($comment_form_id, 'none');
434
    $result = captcha_get_form_id_setting($comment_form_id);
435
    $this->assertNotNull($result, 'Setting and getting CAPTCHA point: none', 'CAPTCHA');
436
    $this->assertNull($result->module, 'Setting and getting CAPTCHA point: none', 'CAPTCHA');
437
    $this->assertNull($result->captcha_type, 'Setting and getting CAPTCHA point: none', 'CAPTCHA');
438
    $result = captcha_get_form_id_setting($comment_form_id, TRUE);
439
    $this->assertEqual($result, 'none', 'Setting and symbolic getting CAPTCHA point: "none"', 'CAPTCHA');
440
    // Set to 'default'
441
    captcha_set_form_id_setting($comment_form_id, 'default');
442
    variable_set('captcha_default_challenge', 'foo/bar');
443
    $result = captcha_get_form_id_setting($comment_form_id);
444
    $this->assertNotNull($result, 'Setting and getting CAPTCHA point: default', 'CAPTCHA');
445
    $this->assertEqual($result->module, 'foo', 'Setting and getting CAPTCHA point: default', 'CAPTCHA');
446
    $this->assertEqual($result->captcha_type, 'bar', 'Setting and getting CAPTCHA point: default', 'CAPTCHA');
447
    $result = captcha_get_form_id_setting($comment_form_id, TRUE);
448
    $this->assertEqual($result, 'default', 'Setting and symbolic getting CAPTCHA point: "default"', 'CAPTCHA');
449
    // Set to 'baz/boo'.
450
    captcha_set_form_id_setting($comment_form_id, 'baz/boo');
451
    $result = captcha_get_form_id_setting($comment_form_id);
452
    $this->assertNotNull($result, 'Setting and getting CAPTCHA point: baz/boo', 'CAPTCHA');
453
    $this->assertEqual($result->module, 'baz', 'Setting and getting CAPTCHA point: baz/boo', 'CAPTCHA');
454
    $this->assertEqual($result->captcha_type, 'boo', 'Setting and getting CAPTCHA point: baz/boo', 'CAPTCHA');
455
    $result = captcha_get_form_id_setting($comment_form_id, TRUE);
456
    $this->assertEqual($result, 'baz/boo', 'Setting and symbolic getting CAPTCHA point: "baz/boo"', 'CAPTCHA');
457
    // Set to NULL (which should delete the CAPTCHA point setting entry).
458
    captcha_set_form_id_setting($comment_form_id, NULL);
459
    $result = captcha_get_form_id_setting($comment_form_id);
460
    $this->assertNull($result, 'Setting and getting CAPTCHA point: NULL', 'CAPTCHA');
461
    $result = captcha_get_form_id_setting($comment_form_id, TRUE);
462
    $this->assertNull($result, 'Setting and symbolic getting CAPTCHA point: NULL', 'CAPTCHA');
463
    // Set with object.
464
    $captcha_type = new stdClass;
465
    $captcha_type->module = 'baba';
466
    $captcha_type->captcha_type = 'fofo';
467
    captcha_set_form_id_setting($comment_form_id, $captcha_type);
468
    $result = captcha_get_form_id_setting($comment_form_id);
469
    $this->assertNotNull($result, 'Setting and getting CAPTCHA point: baba/fofo', 'CAPTCHA');
470
    $this->assertEqual($result->module, 'baba', 'Setting and getting CAPTCHA point: baba/fofo', 'CAPTCHA');
471
    $this->assertEqual($result->captcha_type, 'fofo', 'Setting and getting CAPTCHA point: baba/fofo', 'CAPTCHA');
472
    $result = captcha_get_form_id_setting($comment_form_id, TRUE);
473
    $this->assertEqual($result, 'baba/fofo', 'Setting and symbolic getting CAPTCHA point: "baba/fofo"', 'CAPTCHA');
474

    
475
  }
476

    
477

    
478
  /**
479
   * Helper function for checking CAPTCHA setting of a form.
480
   *
481
   * @param $form_id the form_id of the form to investigate.
482
   * @param $challenge_type what the challenge type should be:
483
   *   NULL, 'none', 'default' or something like 'captcha/Math'
484
   */
485
  protected function assertCaptchaSetting($form_id, $challenge_type) {
486
    $result = captcha_get_form_id_setting(self::COMMENT_FORM_ID, TRUE);
487
    $this->assertEqual($result, $challenge_type,
488
      t('Check CAPTCHA setting for form: expected: @expected, received: @received.',
489
      array('@expected' => var_export($challenge_type, TRUE), '@received' => var_export($result, TRUE))),
490
      'CAPTCHA');
491
  }
492

    
493
  /**
494
   * Testing of the CAPTCHA administration links.
495
   */
496
  function testCaptchAdminLinks() {
497
    // Log in as admin
498
    $this->drupalLogin($this->admin_user);
499

    
500
    // Enable CAPTCHA administration links.
501
    $edit = array(
502
      'captcha_administration_mode' => TRUE,
503
    );
504
    $this->drupalPost(self::CAPTCHA_ADMIN_PATH, $edit, 'Save configuration');
505

    
506
    // Create a node with comments enabled.
507
    $node = $this->createNodeWithCommentsEnabled();
508

    
509
    // Go to node page
510
    $this->drupalGet('node/' . $node->nid);
511

    
512
    // Click the add new comment link
513
    $this->clickLink(t('Add new comment'));
514
    $add_comment_url = $this->getUrl();
515
    // Remove fragment part from comment URL to avoid problems with later asserts
516
    $add_comment_url = strtok($add_comment_url, "#");
517

    
518
    ////////////////////////////////////////////////////////////
519
    // Click the CAPTCHA admin link to enable a challenge.
520
    $this->clickLink(t('Place a CAPTCHA here for untrusted users.'));
521
    // Enable Math CAPTCHA.
522
    $edit = array('captcha_type' => 'captcha/Math');
523
    $this->drupalPost($this->getUrl(), $edit, t('Save'));
524

    
525
    // Check if returned to original comment form.
526
    $this->assertUrl($add_comment_url, array(),
527
      'After setting CAPTCHA with CAPTCHA admin links: should return to original form.', 'CAPTCHA');
528
    // Check if CAPTCHA was successfully enabled (on CAPTCHA admin links fieldset).
529
    $this->assertText(t('CAPTCHA: challenge "@type" enabled', array('@type' => 'Math')),
530
      'Enable a challenge through the CAPTCHA admin links', 'CAPTCHA');
531
    // Check if CAPTCHA was successfully enabled (through API).
532
    $this->assertCaptchaSetting(self::COMMENT_FORM_ID, 'captcha/Math');
533

    
534
    //////////////////////////////////////////////////////
535
    // Edit challenge type through CAPTCHA admin links.
536
    $this->clickLink(t('change'));
537
    // Enable Math CAPTCHA.
538
    $edit = array('captcha_type' => 'default');
539
    $this->drupalPost($this->getUrl(), $edit, t('Save'));
540

    
541
    // Check if returned to original comment form.
542
    $this->assertEqual($add_comment_url, $this->getUrl(),
543
      'After editing challenge type CAPTCHA admin links: should return to original form.', 'CAPTCHA');
544
    // Check if CAPTCHA was successfully changed (on CAPTCHA admin links fieldset).
545
    // This is actually the same as the previous setting because the captcha/Math is the
546
    // default for the default challenge. TODO Make sure the edit is a real change.
547
    $this->assertText(t('CAPTCHA: challenge "@type" enabled', array('@type' => 'Math')),
548
      'Enable a challenge through the CAPTCHA admin links', 'CAPTCHA');
549
    // Check if CAPTCHA was successfully edited (through API).
550
    $this->assertCaptchaSetting(self::COMMENT_FORM_ID, 'default');
551

    
552

    
553

    
554
    //////////////////////////////////////////////////////
555
    // Disable challenge through CAPTCHA admin links.
556
    $this->clickLink(t('disable'));
557
    // And confirm.
558
    $this->drupalPost($this->getUrl(), array(), 'Disable');
559

    
560
    // Check if returned to original comment form.
561
    $this->assertEqual($add_comment_url, $this->getUrl(),
562
      'After disablin challenge with CAPTCHA admin links: should return to original form.', 'CAPTCHA');
563
    // Check if CAPTCHA was successfully disabled (on CAPTCHA admin links fieldset).
564
    $this->assertText(t('CAPTCHA: no challenge enabled'),
565
      'Disable challenge through the CAPTCHA admin links', 'CAPTCHA');
566
    // Check if CAPTCHA was successfully disabled (through API).
567
    $this->assertCaptchaSetting(self::COMMENT_FORM_ID, 'none');
568

    
569
  }
570

    
571

    
572
  function testUntrustedUserPosting() {
573
    // Set CAPTCHA on comment form.
574
    captcha_set_form_id_setting(self::COMMENT_FORM_ID, 'captcha/Math');
575

    
576
    // Create a node with comments enabled.
577
    $node = $this->createNodeWithCommentsEnabled();
578

    
579
    // Log in as normal (untrusted) user.
580
    $this->drupalLogin($this->normal_user);
581

    
582
    // Go to node page and click the "add comment" link.
583
    $this->drupalGet('node/' . $node->nid);
584
    $this->clickLink(t('Add new comment'));
585
    $add_comment_url = $this->getUrl();
586

    
587
    // Check if CAPTCHA is visible on form.
588
    $this->assertCaptchaPresence(TRUE);
589
    // Try to post a comment with wrong answer.
590
    $edit = $this->getCommentFormValues();
591
    $edit['captcha_response'] = 'xx';
592
    $this->drupalPost($add_comment_url, $edit, t('Preview'));
593
    $this->assertText(t(CAPTCHA_WRONG_RESPONSE_ERROR_MESSAGE),
594
      'wrong CAPTCHA should block form submission.', 'CAPTCHA');
595

    
596
    //TODO: more testing for untrusted posts.
597
  }
598

    
599

    
600

    
601
  /**
602
   * Test XSS vulnerability on CAPTCHA description.
603
   */
604
  function testXssOnCaptchaDescription() {
605
    // Set CAPTCHA on user register form.
606
    captcha_set_form_id_setting('user_register', 'captcha/Math');
607

    
608
    // Put JavaScript snippet in CAPTCHA description.
609
    $this->drupalLogin($this->admin_user);
610
    $xss = '<script type="text/javascript">alert("xss")</script>';
611
    $edit = array('captcha_description' => $xss);
612
    $this->drupalPost(self::CAPTCHA_ADMIN_PATH, $edit, 'Save configuration');
613

    
614
    // Visit user register form and check if JavaScript snippet is there.
615
    $this->drupalLogout();
616
    $this->drupalGet('user/register');
617
    $this->assertNoRaw($xss, 'JavaScript should not be allowed in CAPTCHA description.', 'CAPTCHA');
618

    
619
  }
620

    
621
  /**
622
   * Test the CAPTCHA placement clearing.
623
   */
624
  function testCaptchaPlacementCacheClearing() {
625
    // Set CAPTCHA on user register form.
626
    captcha_set_form_id_setting('user_register_form', 'captcha/Math');
627
    // Visit user register form to fill the CAPTCHA placement cache.
628
    $this->drupalGet('user/register');
629
    // Check if there is CAPTCHA placement cache.
630
    $placement_map = variable_get('captcha_placement_map_cache', NULL);
631
    $this->assertNotNull($placement_map, 'CAPTCHA placement cache should be set.');
632
    // Clear the cache
633
    $this->drupalLogin($this->admin_user);
634
    $this->drupalPost(self::CAPTCHA_ADMIN_PATH, array(), t('Clear the CAPTCHA placement cache'));
635
    // Check that the placement cache is unset
636
    $placement_map = variable_get('captcha_placement_map_cache', NULL);
637
    $this->assertNull($placement_map, 'CAPTCHA placement cache should be unset after cache clear.');
638
  }
639

    
640
  /**
641
   * Helper function to get the CAPTCHA point setting straight from the database.
642
   * @param string $form_id
643
   * @return stdClass object
644
   */
645
  private function getCaptchaPointSettingFromDatabase($form_id) {
646
    $result = db_query(
647
      "SELECT * FROM {captcha_points} WHERE form_id = :form_id",
648
      array(':form_id' => $form_id)
649
    )->fetchObject();
650
    return $result;
651
  }
652

    
653
  /**
654
   * Method for testing the CAPTCHA point administration
655
   */
656
  function testCaptchaPointAdministration() {
657
    // Generate CAPTCHA point data:
658
    // Drupal form ID should consist of lowercase alphanumerics and underscore)
659
    $captcha_point_form_id = 'form_' . strtolower($this->randomName(32));
660
    // the Math CAPTCHA by the CAPTCHA module is always available, so let's use it
661
    $captcha_point_module = 'captcha';
662
    $captcha_point_type = 'Math';
663

    
664
    // Log in as admin
665
    $this->drupalLogin($this->admin_user);
666

    
667
    // Set CAPTCHA point through admin/user/captcha/captcha/captcha_point
668
    $form_values = array(
669
      'captcha_point_form_id' => $captcha_point_form_id,
670
      'captcha_type' => $captcha_point_module .'/'. $captcha_point_type,
671
    );
672
    $this->drupalPost(self::CAPTCHA_ADMIN_PATH . '/captcha/captcha_point', $form_values, t('Save'));
673
    $this->assertText(t('Saved CAPTCHA point settings.'),
674
      'Saving of CAPTCHA point settings');
675

    
676
    // Check in database
677
    $result = $this->getCaptchaPointSettingFromDatabase($captcha_point_form_id);
678
    $this->assertEqual($result->module, $captcha_point_module,
679
      'Enabled CAPTCHA point should have module set');
680
    $this->assertEqual($result->captcha_type, $captcha_point_type,
681
      'Enabled CAPTCHA point should have type set');
682

    
683
    // Disable CAPTCHA point again
684
    $this->drupalPost(self::CAPTCHA_ADMIN_PATH . '/captcha/captcha_point/'. $captcha_point_form_id .'/disable', array(), t('Disable'));
685
    $this->assertRaw(t('Disabled CAPTCHA for form %form_id.', array('%form_id' => $captcha_point_form_id)), 'Disabling of CAPTCHA point');
686

    
687
    // Check in database
688
    $result = $this->getCaptchaPointSettingFromDatabase($captcha_point_form_id);
689
    $this->assertNull($result->module,
690
      'Disabled CAPTCHA point should have NULL as module');
691
    $this->assertNull($result->captcha_type,
692
      'Disabled CAPTCHA point should have NULL as type');
693

    
694
    // Set CAPTCHA point through admin/user/captcha/captcha/captcha_point/$form_id
695
    $form_values = array(
696
      'captcha_type' => $captcha_point_module .'/'. $captcha_point_type,
697
    );
698
    $this->drupalPost(self::CAPTCHA_ADMIN_PATH . '/captcha/captcha_point/'. $captcha_point_form_id, $form_values, t('Save'));
699
    $this->assertText(t('Saved CAPTCHA point settings.'),
700
      'Saving of CAPTCHA point settings');
701

    
702
    // Check in database
703
    $result = $this->getCaptchaPointSettingFromDatabase($captcha_point_form_id);
704
    $this->assertEqual($result->module, $captcha_point_module,
705
      'Enabled CAPTCHA point should have module set');
706
    $this->assertEqual($result->captcha_type, $captcha_point_type,
707
      'Enabled CAPTCHA point should have type set');
708

    
709
    // Delete CAPTCHA point
710
    $this->drupalPost(self::CAPTCHA_ADMIN_PATH . '/captcha/captcha_point/'. $captcha_point_form_id .'/delete', array(), t('Delete'));
711
    $this->assertRaw(t('Deleted CAPTCHA for form %form_id.', array('%form_id' => $captcha_point_form_id)),
712
      'Deleting of CAPTCHA point');
713

    
714
    // Check in database
715
    $result = $this->getCaptchaPointSettingFromDatabase($captcha_point_form_id);
716
    $this->assertFalse($result, 'Deleted CAPTCHA point should be in database');
717
  }
718

    
719
  /**
720
   * Method for testing the CAPTCHA point administration
721
   */
722
  function testCaptchaPointAdministrationByNonAdmin() {
723
    // First add a CAPTCHA point (as admin)
724
    $this->drupalLogin($this->admin_user);
725
    $captcha_point_form_id = 'form_' . strtolower($this->randomName(32));
726
    $captcha_point_module = 'captcha';
727
    $captcha_point_type = 'Math';
728
    $form_values = array(
729
      'captcha_point_form_id' => $captcha_point_form_id,
730
      'captcha_type' => $captcha_point_module .'/'. $captcha_point_type,
731
    );
732
    $this->drupalPost(self::CAPTCHA_ADMIN_PATH . '/captcha/captcha_point/', $form_values, t('Save'));
733
    $this->assertText(t('Saved CAPTCHA point settings.'),
734
      'Saving of CAPTCHA point settings');
735

    
736
    // Switch from admin to nonadmin
737
    $this->drupalGet(url('logout', array('absolute' => TRUE)));
738
    $this->drupalLogin($this->normal_user);
739

    
740

    
741
    // Try to set CAPTCHA point through admin/user/captcha/captcha/captcha_point
742
    $this->drupalGet(self::CAPTCHA_ADMIN_PATH . '/captcha/captcha_point');
743
    $this->assertText(t('You are not authorized to access this page.'),
744
      'Non admin should not be able to set a CAPTCHA point');
745

    
746
    // Try to set CAPTCHA point through admin/user/captcha/captcha/captcha_point/$form_id
747
    $this->drupalGet(self::CAPTCHA_ADMIN_PATH . '/captcha/captcha_point/' . 'form_' . strtolower($this->randomName(32)));
748
    $this->assertText(t('You are not authorized to access this page.'),
749
      'Non admin should not be able to set a CAPTCHA point');
750

    
751
    // Try to disable the CAPTCHA point
752
    $this->drupalGet(self::CAPTCHA_ADMIN_PATH . '/captcha/captcha_point/'. $captcha_point_form_id .'/disable');
753
    $this->assertText(t('You are not authorized to access this page.'),
754
      'Non admin should not be able to disable a CAPTCHA point');
755

    
756
    // Try to delete the CAPTCHA point
757
    $this->drupalGet(self::CAPTCHA_ADMIN_PATH . '/captcha/captcha_point/'. $captcha_point_form_id .'/delete');
758
    $this->assertText(t('You are not authorized to access this page.'),
759
      'Non admin should not be able to delete a CAPTCHA point');
760

    
761
    // Switch from nonadmin to admin again
762
    $this->drupalGet(url('logout', array('absolute' => TRUE)));
763
    $this->drupalLogin($this->admin_user);
764

    
765
    // Check if original CAPTCHA point still exists in database
766
    $result = $this->getCaptchaPointSettingFromDatabase($captcha_point_form_id);
767
    $this->assertEqual($result->module, $captcha_point_module,
768
      'Enabled CAPTCHA point should still have module set');
769
    $this->assertEqual($result->captcha_type, $captcha_point_type,
770
      'Enabled CAPTCHA point should still have type set');
771

    
772
    // Delete CAPTCHA point
773
    $this->drupalPost(self::CAPTCHA_ADMIN_PATH . '/captcha/captcha_point/'. $captcha_point_form_id .'/delete', array(), t('Delete'));
774
    $this->assertRaw(t('Deleted CAPTCHA for form %form_id.', array('%form_id' => $captcha_point_form_id)),
775
      'Deleting of CAPTCHA point');
776
  }
777

    
778

    
779

    
780
}
781

    
782

    
783

    
784
class CaptchaPersistenceTestCase extends CaptchaBaseWebTestCase {
785

    
786
  public static function getInfo() {
787
    return array(
788
      'name' => t('CAPTCHA persistence functionality'),
789
      'description' => t('Testing of the CAPTCHA persistence functionality.'),
790
      'group' => t('CAPTCHA'),
791
    );
792
  }
793

    
794
  /**
795
   * Set up the persistence and CAPTCHA settings.
796
   * @param int $persistence the persistence value.
797
   */
798
  private function setUpPersistence($persistence) {
799
    // Log in as admin
800
    $this->drupalLogin($this->admin_user);
801
    // Set persistence.
802
    $edit = array('captcha_persistence' => $persistence);
803
    $this->drupalPost(self::CAPTCHA_ADMIN_PATH, $edit, 'Save configuration');
804
    // Log admin out.
805
    $this->drupalLogout();
806

    
807
    // Set the Test123 CAPTCHA on user register and comment form.
808
    // We have to do this with the function captcha_set_form_id_setting()
809
    // (because the CATCHA admin form does not show the Test123 option).
810
    // We also have to do this after all usage of the CAPTCHA admin form
811
    // (because posting the CAPTCHA admin form would set the CAPTCHA to 'none').
812
    captcha_set_form_id_setting('user_login', 'captcha/Test');
813
    $this->drupalGet('user');
814
    $this->assertCaptchaPresence(TRUE);
815
    captcha_set_form_id_setting('user_register_form', 'captcha/Test');
816
    $this->drupalGet('user/register');
817
    $this->assertCaptchaPresence(TRUE);
818
  }
819

    
820
  protected function assertPreservedCsid($captcha_sid_initial) {
821
    $captcha_sid = $this->getCaptchaSidFromForm();
822
    $this->assertEqual($captcha_sid_initial, $captcha_sid,
823
      "CAPTCHA session ID should be preserved (expected: $captcha_sid_initial, found: $captcha_sid).");
824
  }
825

    
826
  protected function assertDifferentCsid($captcha_sid_initial) {
827
    $captcha_sid = $this->getCaptchaSidFromForm();
828
    $this->assertNotEqual($captcha_sid_initial, $captcha_sid,
829
      "CAPTCHA session ID should be different.");
830
  }
831

    
832
  function testPersistenceAlways(){
833
    // Set up of persistence and CAPTCHAs.
834
    $this->setUpPersistence(CAPTCHA_PERSISTENCE_SHOW_ALWAYS);
835

    
836
    // Go to login form and check if there is a CAPTCHA on the login form (look for the title).
837
    $this->drupalGet('user');
838
    $this->assertCaptchaPresence(TRUE);
839
    $captcha_sid_initial = $this->getCaptchaSidFromForm();
840

    
841
    // Try to with wrong user name and password, but correct CAPTCHA.
842
    $edit = array(
843
      'name' => 'foobar',
844
      'pass' => 'bazlaz',
845
      'captcha_response' => 'Test 123',
846
    );
847
    $this->drupalPost(NULL, $edit, t('Log in'));
848
    // Check that there was no error message for the CAPTCHA.
849
    $this->assertCaptchaResponseAccepted();
850

    
851
    // Name and password were wrong, we should get an updated form with a fresh CAPTCHA.
852
    $this->assertCaptchaPresence(TRUE);
853
    $this->assertPreservedCsid($captcha_sid_initial);
854

    
855
    // Post from again.
856
    $this->drupalPost(NULL, $edit, t('Log in'));
857
    // Check that there was no error message for the CAPTCHA.
858
    $this->assertCaptchaResponseAccepted();
859
    $this->assertPreservedCsid($captcha_sid_initial);
860

    
861
  }
862

    
863
  function testPersistencePerFormInstance(){
864
    // Set up of persistence and CAPTCHAs.
865
    $this->setUpPersistence(CAPTCHA_PERSISTENCE_SKIP_ONCE_SUCCESSFUL_PER_FORM_INSTANCE);
866

    
867
    // Go to login form and check if there is a CAPTCHA on the login form.
868
    $this->drupalGet('user');
869
    $this->assertCaptchaPresence(TRUE);
870
    $captcha_sid_initial = $this->getCaptchaSidFromForm();
871

    
872
    // Try to with wrong user name and password, but correct CAPTCHA.
873
    $edit = array(
874
      'name' => 'foobar',
875
      'pass' => 'bazlaz',
876
      'captcha_response' => 'Test 123',
877
    );
878
    $this->drupalPost(NULL, $edit, t('Log in'));
879
    // Check that there was no error message for the CAPTCHA.
880
    $this->assertCaptchaResponseAccepted();
881
    // There shouldn't be a CAPTCHA on the new form.
882
    $this->assertCaptchaPresence(FALSE);
883
    $this->assertPreservedCsid($captcha_sid_initial);
884

    
885
    // Start a new form instance/session
886
    $this->drupalGet('node');
887
    $this->drupalGet('user');
888
    $this->assertCaptchaPresence(TRUE);
889
    $this->assertDifferentCsid($captcha_sid_initial);
890

    
891
    // Check another form
892
    $this->drupalGet('user/register');
893
    $this->assertCaptchaPresence(TRUE);
894
    $this->assertDifferentCsid($captcha_sid_initial);
895

    
896
  }
897

    
898
  function testPersistencePerFormType(){
899
    // Set up of persistence and CAPTCHAs.
900
    $this->setUpPersistence(CAPTCHA_PERSISTENCE_SKIP_ONCE_SUCCESSFUL_PER_FORM_TYPE);
901

    
902
    // Go to login form and check if there is a CAPTCHA on the login form.
903
    $this->drupalGet('user');
904
    $this->assertCaptchaPresence(TRUE);
905
    $captcha_sid_initial = $this->getCaptchaSidFromForm();
906

    
907
    // Try to with wrong user name and password, but correct CAPTCHA.
908
    $edit = array(
909
      'name' => 'foobar',
910
      'pass' => 'bazlaz',
911
      'captcha_response' => 'Test 123',
912
    );
913
    $this->drupalPost(NULL, $edit, t('Log in'));
914
    // Check that there was no error message for the CAPTCHA.
915
    $this->assertCaptchaResponseAccepted();
916
    // There shouldn't be a CAPTCHA on the new form.
917
    $this->assertCaptchaPresence(FALSE);
918
    $this->assertPreservedCsid($captcha_sid_initial);
919

    
920
    // Start a new form instance/session
921
    $this->drupalGet('node');
922
    $this->drupalGet('user');
923
    $this->assertCaptchaPresence(FALSE);
924
    $this->assertDifferentCsid($captcha_sid_initial);
925

    
926
    // Check another form
927
    $this->drupalGet('user/register');
928
    $this->assertCaptchaPresence(TRUE);
929
    $this->assertDifferentCsid($captcha_sid_initial);
930
  }
931

    
932
  function testPersistenceOnlyOnce(){
933
    // Set up of persistence and CAPTCHAs.
934
    $this->setUpPersistence(CAPTCHA_PERSISTENCE_SKIP_ONCE_SUCCESSFUL);
935

    
936
    // Go to login form and check if there is a CAPTCHA on the login form.
937
    $this->drupalGet('user');
938
    $this->assertCaptchaPresence(TRUE);
939
    $captcha_sid_initial = $this->getCaptchaSidFromForm();
940

    
941
    // Try to with wrong user name and password, but correct CAPTCHA.
942
    $edit = array(
943
      'name' => 'foobar',
944
      'pass' => 'bazlaz',
945
      'captcha_response' => 'Test 123',
946
    );
947
    $this->drupalPost(NULL, $edit, t('Log in'));
948
    // Check that there was no error message for the CAPTCHA.
949
    $this->assertCaptchaResponseAccepted();
950
    // There shouldn't be a CAPTCHA on the new form.
951
    $this->assertCaptchaPresence(FALSE);
952
    $this->assertPreservedCsid($captcha_sid_initial);
953

    
954
    // Start a new form instance/session
955
    $this->drupalGet('node');
956
    $this->drupalGet('user');
957
    $this->assertCaptchaPresence(FALSE);
958
    $this->assertDifferentCsid($captcha_sid_initial);
959

    
960
    // Check another form
961
    $this->drupalGet('user/register');
962
    $this->assertCaptchaPresence(FALSE);
963
    $this->assertDifferentCsid($captcha_sid_initial);
964
  }
965

    
966
}
967

    
968

    
969
class CaptchaSessionReuseAttackTestCase extends CaptchaBaseWebTestCase {
970

    
971
  public static function getInfo() {
972
    return array(
973
      'name' => t('CAPTCHA session reuse attack tests'),
974
      'description' => t('Testing of the protection against CAPTCHA session reuse attacks.'),
975
      'group' => t('CAPTCHA'),
976
    );
977
  }
978

    
979
  /**
980
   * Assert that the CAPTCHA session ID reuse attack was detected.
981
   */
982
  protected function assertCaptchaSessionIdReuseAttackDetection() {
983
    $this->assertText(t(CAPTCHA_SESSION_REUSE_ATTACK_ERROR_MESSAGE),
984
      'CAPTCHA session ID reuse attack should be detected.',
985
      'CAPTCHA');
986
    // There should be an error message about wrong response.
987
    $this->assertText(t(CAPTCHA_WRONG_RESPONSE_ERROR_MESSAGE),
988
      'CAPTCHA response should flagged as wrong.',
989
      'CAPTCHA');
990
  }
991

    
992
  function testCaptchaSessionReuseAttackDetectionOnCommentPreview() {
993
    // Create commentable node
994
    $node = $this->createNodeWithCommentsEnabled();
995
    // Set Test CAPTCHA on comment form.
996
    captcha_set_form_id_setting(self::COMMENT_FORM_ID, 'captcha/Math');
997
    variable_set('captcha_persistence', CAPTCHA_PERSISTENCE_SKIP_ONCE_SUCCESSFUL_PER_FORM_INSTANCE);
998

    
999
    // Log in as normal user.
1000
    $this->drupalLogin($this->normal_user);
1001

    
1002
    // Go to comment form of commentable node.
1003
    $this->drupalGet('comment/reply/' . $node->nid);
1004
    $this->assertCaptchaPresence(TRUE);
1005

    
1006
    // Get CAPTCHA session ID and solution of the challenge.
1007
    $captcha_sid = $this->getCaptchaSidFromForm();
1008
    $captcha_token = $this->getCaptchaTokenFromForm();
1009
    $solution = $this->getMathCaptchaSolutionFromForm();
1010

    
1011
    // Post the form with the solution.
1012
    $edit = $this->getCommentFormValues();
1013
    $edit['captcha_response'] = $solution;
1014
    $this->drupalPost(NULL, $edit, t('Preview'));
1015
    // Answer should be accepted and further CAPTCHA ommitted.
1016
    $this->assertCaptchaResponseAccepted();
1017
    $this->assertCaptchaPresence(FALSE);
1018

    
1019
    // Post a new comment, reusing the previous CAPTCHA session.
1020
    $edit = $this->getCommentFormValues();
1021
    $edit['captcha_sid'] = $captcha_sid;
1022
    $edit['captcha_token'] = $captcha_token;
1023
    $edit['captcha_response'] = $solution;
1024
    $this->drupalPost('comment/reply/' . $node->nid, $edit, t('Preview'));
1025
    // CAPTCHA session reuse attack should be detected.
1026
    $this->assertCaptchaSessionIdReuseAttackDetection();
1027
    // There should be a CAPTCHA.
1028
    $this->assertCaptchaPresence(TRUE);
1029

    
1030
  }
1031

    
1032
  function testCaptchaSessionReuseAttackDetectionOnNodeForm() {
1033
    // Set CAPTCHA on page form.
1034
    captcha_set_form_id_setting('page_node_form', 'captcha/Math');
1035
    variable_set('captcha_persistence', CAPTCHA_PERSISTENCE_SKIP_ONCE_SUCCESSFUL_PER_FORM_INSTANCE);
1036

    
1037
    // Log in as normal user.
1038
    $this->drupalLogin($this->normal_user);
1039

    
1040
    // Go to node add form.
1041
    $this->drupalGet('node/add/page');
1042
    $this->assertCaptchaPresence(TRUE);
1043

    
1044
    // Get CAPTCHA session ID and solution of the challenge.
1045
    $captcha_sid = $this->getCaptchaSidFromForm();
1046
    $captcha_token = $this->getCaptchaTokenFromForm();
1047
    $solution = $this->getMathCaptchaSolutionFromForm();
1048

    
1049
    // Page settings to post, with correct CAPTCHA answer.
1050
    $edit = $this->getNodeFormValues();
1051
    $edit['captcha_response'] = $solution;
1052
    // Preview the node
1053
    $this->drupalPost(NULL, $edit, t('Preview'));
1054
    // Answer should be accepted.
1055
    $this->assertCaptchaResponseAccepted();
1056
    // Check that there is no CAPTCHA after preview.
1057
    $this->assertCaptchaPresence(FALSE);
1058

    
1059
    // Post a new comment, reusing the previous CAPTCHA session.
1060
    $edit = $this->getNodeFormValues();
1061
    $edit['captcha_sid'] = $captcha_sid;
1062
    $edit['captcha_token'] = $captcha_token;
1063
    $edit['captcha_response'] = $solution;
1064
    $this->drupalPost('node/add/page', $edit, t('Preview'));
1065
    // CAPTCHA session reuse attack should be detected.
1066
    $this->assertCaptchaSessionIdReuseAttackDetection();
1067
    // There should be a CAPTCHA.
1068
    $this->assertCaptchaPresence(TRUE);
1069

    
1070
  }
1071

    
1072
  function testCaptchaSessionReuseAttackDetectionOnLoginForm() {
1073
    // Set CAPTCHA on login form.
1074
    captcha_set_form_id_setting('user_login', 'captcha/Math');
1075
    variable_set('captcha_persistence', CAPTCHA_PERSISTENCE_SKIP_ONCE_SUCCESSFUL_PER_FORM_INSTANCE);
1076

    
1077
    // Go to log in form.
1078
    $this->drupalGet('user');
1079
    $this->assertCaptchaPresence(TRUE);
1080

    
1081
    // Get CAPTCHA session ID and solution of the challenge.
1082
    $captcha_sid = $this->getCaptchaSidFromForm();
1083
    $captcha_token = $this->getCaptchaTokenFromForm();
1084
    $solution = $this->getMathCaptchaSolutionFromForm();
1085

    
1086
    // Log in through form.
1087
    $edit = array(
1088
      'name' => $this->normal_user->name,
1089
      'pass' => $this->normal_user->pass_raw,
1090
      'captcha_response' => $solution,
1091
    );
1092
    $this->drupalPost(NULL, $edit, t('Log in'));
1093
    $this->assertCaptchaResponseAccepted();
1094
    $this->assertCaptchaPresence(FALSE);
1095
    // If a "log out" link appears on the page, it is almost certainly because
1096
    // the login was successful.
1097
    $pass = $this->assertLink(t('Log out'), 0, t('User %name successfully logged in.', array('%name' => $this->normal_user->name)), t('User login'));
1098

    
1099
    // Log out again.
1100
    $this->drupalLogout();
1101

    
1102
    // Try to log in again, reusing the previous CAPTCHA session.
1103
    $edit += array(
1104
      'captcha_sid' => $captcha_sid,
1105
      'captcha_token' => $captcha_token,
1106
    );
1107
    $this->drupalPost('user', $edit, t('Log in'));
1108
    // CAPTCHA session reuse attack should be detected.
1109
    $this->assertCaptchaSessionIdReuseAttackDetection();
1110
    // There should be a CAPTCHA.
1111
    $this->assertCaptchaPresence(TRUE);
1112
  }
1113

    
1114

    
1115
  public function testMultipleCaptchaProtectedFormsOnOnePage()
1116
  {
1117
    // Set Test CAPTCHA on comment form and login block
1118
    captcha_set_form_id_setting(self::COMMENT_FORM_ID, 'captcha/Test');
1119
    captcha_set_form_id_setting('user_login_block', 'captcha/Math');
1120
    $this->allowCommentPostingForAnonymousVisitors();
1121

    
1122
    // Create a node with comments enabled.
1123
    $node = $this->createNodeWithCommentsEnabled();
1124

    
1125
    // Preview comment with correct CAPTCHA answer.
1126
    $edit = $this->getCommentFormValues();
1127
    $comment_subject = $edit['subject'];
1128
    $edit['captcha_response'] = 'Test 123';
1129
    $this->drupalPost('comment/reply/' . $node->nid, $edit, t('Preview'));
1130
    // Post should be accepted: no warnings,
1131
    // no CAPTCHA reuse detection (which could be used by user log in block).
1132
    $this->assertCaptchaResponseAccepted();
1133
    $this->assertText($comment_subject);
1134

    
1135
  }
1136

    
1137
}
1138

    
1139

    
1140
// Some tricks to debug:
1141
// drupal_debug($data) // from devel module
1142
// file_put_contents('tmp.simpletest.html', $this->drupalGetContent());