2020-09-08 01:47:04 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace SilverStripe\Security\Tests;
|
|
|
|
|
|
|
|
use SilverStripe\Control\Controller;
|
|
|
|
use SilverStripe\Control\NullHTTPRequest;
|
|
|
|
use SilverStripe\Core\Config\Config;
|
|
|
|
use SilverStripe\Core\Injector\Injector;
|
2024-09-25 06:11:39 +02:00
|
|
|
use SilverStripe\Dev\Deprecation;
|
2020-09-08 01:47:04 +02:00
|
|
|
use SilverStripe\Dev\SapphireTest;
|
|
|
|
use SilverStripe\ORM\FieldType\DBDatetime;
|
|
|
|
use SilverStripe\ORM\ValidationResult;
|
|
|
|
use SilverStripe\Security\Authenticator;
|
|
|
|
use SilverStripe\Security\DefaultAdminService;
|
|
|
|
use SilverStripe\Security\IdentityStore;
|
|
|
|
use SilverStripe\Security\LoginAttempt;
|
|
|
|
use SilverStripe\Security\Member;
|
|
|
|
use SilverStripe\Security\MemberAuthenticator\CMSMemberAuthenticator;
|
|
|
|
use SilverStripe\Security\MemberAuthenticator\CMSMemberLoginForm;
|
|
|
|
use SilverStripe\Security\MemberAuthenticator\MemberAuthenticator;
|
|
|
|
use SilverStripe\Security\MemberAuthenticator\MemberLoginForm;
|
|
|
|
use SilverStripe\Security\PasswordValidator;
|
|
|
|
use SilverStripe\Security\Security;
|
|
|
|
use SilverStripe\Versioned\Versioned;
|
|
|
|
|
|
|
|
class VersionedMemberAuthenticatorTest extends SapphireTest
|
|
|
|
{
|
|
|
|
|
|
|
|
protected $usesDatabase = true;
|
|
|
|
|
|
|
|
protected static $required_extensions = [
|
|
|
|
Member::class => [
|
|
|
|
Versioned::class
|
|
|
|
]
|
|
|
|
];
|
|
|
|
|
2021-10-27 04:39:47 +02:00
|
|
|
protected function setUp(): void
|
2020-09-08 01:47:04 +02:00
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
|
|
|
|
if (!class_exists(Versioned::class)) {
|
|
|
|
$this->markTestSkipped("Versioned is required");
|
|
|
|
return;
|
|
|
|
}
|
2020-10-06 05:05:04 +02:00
|
|
|
|
2024-10-22 06:56:01 +02:00
|
|
|
// Explicity add the Versioned extension to Member, even though it's already in $required_extensions.
|
|
|
|
// This is done to call `unset(self::class::$extra_methods[strtolower($subclass)]);` in
|
|
|
|
// Extensible::add_extension() so when CustomMethods::getExtraMethodConfig() updates the $extra_methods
|
|
|
|
// it will include methods of Versioned such as publishSingle()
|
|
|
|
// This issue will only occur when running subsequent unit test classes in the same process, rather than this
|
|
|
|
// this unit test class in isolation
|
|
|
|
Member::add_extension(Versioned::class);
|
|
|
|
|
2020-10-06 05:05:04 +02:00
|
|
|
// Enforce dummy validation (this can otherwise be influenced by recipe config)
|
2024-09-25 06:11:39 +02:00
|
|
|
Deprecation::withSuppressedNotice(
|
|
|
|
fn() => PasswordValidator::singleton()
|
2020-10-06 05:05:04 +02:00
|
|
|
->setMinLength(0)
|
2024-09-25 06:11:39 +02:00
|
|
|
->setTestNames([])
|
|
|
|
);
|
2020-09-08 01:47:04 +02:00
|
|
|
}
|
|
|
|
|
2021-10-27 04:39:47 +02:00
|
|
|
protected function tearDown(): void
|
2020-09-08 01:47:04 +02:00
|
|
|
{
|
|
|
|
$this->logOut();
|
|
|
|
parent::tearDown();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testAuthenticate()
|
|
|
|
{
|
|
|
|
$mockDate1 = '2010-01-01 10:00:00';
|
|
|
|
$readingMode = sprintf('Archive.%s.Stage', $mockDate1);
|
|
|
|
|
|
|
|
/** @var Member $member */
|
|
|
|
$member = DBDatetime::withFixedNow($mockDate1, function () {
|
|
|
|
$member = Member::create();
|
|
|
|
$member->update([
|
|
|
|
'FirstName' => 'Jane',
|
|
|
|
'Surname' => 'Doe',
|
|
|
|
'Email' => 'jane.doe@example.com'
|
|
|
|
]);
|
|
|
|
$member->write();
|
|
|
|
$member->changePassword('password', true);
|
|
|
|
|
|
|
|
return $member;
|
|
|
|
});
|
|
|
|
|
|
|
|
$member->changePassword('new-password', true);
|
|
|
|
|
|
|
|
/** @var ValidationResult $results */
|
|
|
|
$results = Versioned::withVersionedMode(function () use ($readingMode) {
|
|
|
|
Versioned::set_reading_mode($readingMode);
|
|
|
|
$authenticator = new MemberAuthenticator();
|
|
|
|
|
|
|
|
// Test correct login
|
|
|
|
/** @var ValidationResult $message */
|
|
|
|
$authenticator->authenticate(
|
|
|
|
[
|
|
|
|
'Email' => 'jane.doe@example.com',
|
|
|
|
'Password' => 'password'
|
|
|
|
],
|
|
|
|
Controller::curr()->getRequest(),
|
|
|
|
$result
|
|
|
|
);
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertFalse(
|
|
|
|
$results->isValid(),
|
|
|
|
'Authenticate using old credentials fails even when using an old reading mode'
|
|
|
|
);
|
|
|
|
|
|
|
|
/** @var ValidationResult $results */
|
|
|
|
$results = Versioned::withVersionedMode(function () use ($readingMode) {
|
|
|
|
Versioned::set_reading_mode($readingMode);
|
|
|
|
$authenticator = new MemberAuthenticator();
|
|
|
|
|
|
|
|
// Test correct login
|
|
|
|
/** @var ValidationResult $message */
|
|
|
|
$authenticator->authenticate(
|
|
|
|
[
|
|
|
|
'Email' => 'jane.doe@example.com',
|
|
|
|
'Password' => 'new-password'
|
|
|
|
],
|
|
|
|
Controller::curr()->getRequest(),
|
|
|
|
$result
|
|
|
|
);
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
$results->isValid(),
|
|
|
|
'Authenticate using current credentials succeeds even when using an old reading mode'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testAuthenticateAgainstLiveStage()
|
|
|
|
{
|
|
|
|
/** @var Member $member */
|
|
|
|
$member = Member::create();
|
|
|
|
$member->update([
|
|
|
|
'FirstName' => 'Jane',
|
|
|
|
'Surname' => 'Doe',
|
|
|
|
'Email' => 'jane.doe@example.com'
|
|
|
|
]);
|
|
|
|
$member->write();
|
|
|
|
$member->changePassword('password', true);
|
|
|
|
$member->publishSingle();
|
|
|
|
|
|
|
|
$member->changePassword('new-password', true);
|
|
|
|
|
|
|
|
/** @var ValidationResult $results */
|
|
|
|
$results = Versioned::withVersionedMode(function () {
|
|
|
|
Versioned::set_stage(Versioned::LIVE);
|
|
|
|
$authenticator = new MemberAuthenticator();
|
|
|
|
|
|
|
|
// Test correct login
|
|
|
|
/** @var ValidationResult $message */
|
|
|
|
$authenticator->authenticate(
|
|
|
|
[
|
|
|
|
'Email' => 'jane.doe@example.com',
|
|
|
|
'Password' => 'password'
|
|
|
|
],
|
|
|
|
Controller::curr()->getRequest(),
|
|
|
|
$result
|
|
|
|
);
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertFalse(
|
|
|
|
$results->isValid(),
|
|
|
|
'Authenticate using "published" credentials fails when draft credentials have changed'
|
|
|
|
);
|
|
|
|
|
|
|
|
/** @var ValidationResult $results */
|
|
|
|
$results = Versioned::withVersionedMode(function () {
|
|
|
|
Versioned::set_stage(Versioned::LIVE);
|
|
|
|
$authenticator = new MemberAuthenticator();
|
|
|
|
|
|
|
|
// Test correct login
|
|
|
|
/** @var ValidationResult $message */
|
|
|
|
$authenticator->authenticate(
|
|
|
|
[
|
|
|
|
'Email' => 'jane.doe@example.com',
|
|
|
|
'Password' => 'new-password'
|
|
|
|
],
|
|
|
|
Controller::curr()->getRequest(),
|
|
|
|
$result
|
|
|
|
);
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
$results->isValid(),
|
|
|
|
'Authenticate using current credentials succeeds even when "published" credentials are different'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|