[ Versioned::class ] ]; protected function setUp(): void { parent::setUp(); if (!class_exists(Versioned::class)) { $this->markTestSkipped("Versioned is required"); return; } // 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); // Enforce dummy validation (this can otherwise be influenced by recipe config) PasswordValidator::singleton() ->setMinLength(0) ->setTestNames([]); } protected function tearDown(): void { $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' ); } }