banFile)) { unlink($this->banFile); } $this->passwordHash = sha1($this->password . $this->login . $this->salt); $this->configManager = new FakeConfigManager([ 'credentials.login' => $this->login, 'credentials.hash' => $this->passwordHash, 'credentials.salt' => $this->salt, 'resource.ban_file' => $this->banFile, 'resource.log' => $this->logFile, 'security.ban_after' => 2, 'security.ban_duration' => 3600, 'security.trusted_proxies' => [$this->trustedProxy], 'ldap.host' => '', ]); $this->cookie = []; $this->session = []; $this->cookieManager = $this->createMock(CookieManager::class); $this->cookieManager->method('getCookieParameter')->willReturnCallback(function (string $key) { return $this->cookie[$key] ?? null; }); $this->sessionManager = new SessionManager($this->session, $this->configManager, 'session_path'); $this->banManager = $this->createMock(BanManager::class); $this->loginManager = new LoginManager( $this->configManager, $this->sessionManager, $this->cookieManager, $this->banManager, $this->createMock(LoggerInterface::class) ); $this->server['REMOTE_ADDR'] = $this->ipAddr; } /** * Record a failed login attempt */ public function testHandleFailedLogin(): void { $this->banManager->expects(static::exactly(2))->method('handleFailedAttempt'); $this->banManager->method('isBanned')->willReturn(true); $this->loginManager->handleFailedLogin($this->server); $this->loginManager->handleFailedLogin($this->server); static::assertFalse($this->loginManager->canLogin($this->server)); } /** * Record a failed login attempt - IP behind a trusted proxy */ public function testHandleFailedLoginBehindTrustedProxy() { $server = [ 'REMOTE_ADDR' => $this->trustedProxy, 'HTTP_X_FORWARDED_FOR' => $this->ipAddr, ]; $this->banManager->expects(static::exactly(2))->method('handleFailedAttempt'); $this->banManager->method('isBanned')->willReturn(true); $this->loginManager->handleFailedLogin($server); $this->loginManager->handleFailedLogin($server); $this->assertFalse($this->loginManager->canLogin($server)); } /** * Record a failed login attempt - IP behind a trusted proxy but not forwarded */ public function testHandleFailedLoginBehindTrustedProxyNoIp() { $server = [ 'REMOTE_ADDR' => $this->trustedProxy, ]; $this->loginManager->handleFailedLogin($server); $this->loginManager->handleFailedLogin($server); $this->assertTrue($this->loginManager->canLogin($server)); } /** * Nothing to do */ public function testHandleSuccessfulLogin() { $this->assertTrue($this->loginManager->canLogin($this->server)); $this->loginManager->handleSuccessfulLogin($this->server); $this->assertTrue($this->loginManager->canLogin($this->server)); } /** * Erase failure records after successfully logging in from this IP */ public function testHandleSuccessfulLoginAfterFailure() { $this->loginManager->handleFailedLogin($this->server); $this->assertTrue($this->loginManager->canLogin($this->server)); $this->loginManager->handleSuccessfulLogin($this->server); $this->loginManager->handleFailedLogin($this->server); $this->assertTrue($this->loginManager->canLogin($this->server)); } /** * The IP is not banned */ public function testCanLoginIpNotBanned() { $this->assertTrue($this->loginManager->canLogin($this->server)); } /** * Generate a token depending on the user credentials and client IP */ public function testGenerateStaySignedInToken() { $this->loginManager->generateStaySignedInToken($this->clientIpAddress); $this->assertEquals( sha1($this->passwordHash . $this->clientIpAddress . $this->salt), $this->loginManager->getStaySignedInToken() ); } /** * Generate a token depending on the user credentials with session protected disabled */ public function testGenerateStaySignedInTokenSessionProtectionDisabled() { $this->configManager->set('security.session_protection_disabled', true); $this->loginManager->generateStaySignedInToken($this->clientIpAddress); $this->assertEquals( sha1($this->passwordHash . $this->salt), $this->loginManager->getStaySignedInToken() ); } /** * Check user login - Shaarli has not yet been configured */ public function testCheckLoginStateNotConfigured() { $configManager = new FakeConfigManager([ 'resource.ban_file' => $this->banFile, ]); $loginManager = new LoginManager( $configManager, $this->sessionManager, $this->cookieManager, $this->banManager, $this->createMock(LoggerInterface::class) ); $loginManager->checkLoginState(''); $this->assertFalse($loginManager->isLoggedIn()); } /** * Check user login - the client cookie does not match the server token */ public function testCheckLoginStateStaySignedInWithInvalidToken() { // simulate a previous login $this->session = [ 'ip' => $this->clientIpAddress, 'expires_on' => time() + 100, ]; $this->loginManager->generateStaySignedInToken($this->clientIpAddress); $this->cookie[CookieManager::STAY_SIGNED_IN] = 'nope'; $this->loginManager->checkLoginState($this->clientIpAddress); $this->assertTrue($this->loginManager->isLoggedIn()); $this->assertTrue(empty($this->session['username'])); } /** * Check user login - the client cookie matches the server token */ public function testCheckLoginStateStaySignedInWithValidToken() { $this->loginManager->generateStaySignedInToken($this->clientIpAddress); $this->cookie[CookieManager::STAY_SIGNED_IN] = $this->loginManager->getStaySignedInToken(); $this->loginManager->checkLoginState($this->clientIpAddress); $this->assertTrue($this->loginManager->isLoggedIn()); $this->assertEquals($this->login, $this->session['username']); $this->assertEquals($this->clientIpAddress, $this->session['ip']); } /** * Check user login - the session has expired */ public function testCheckLoginStateSessionExpired() { $this->loginManager->generateStaySignedInToken($this->clientIpAddress); $this->session['expires_on'] = time() - 100; $this->loginManager->checkLoginState($this->clientIpAddress); $this->assertFalse($this->loginManager->isLoggedIn()); } /** * Check user login - the remote client IP has changed */ public function testCheckLoginStateClientIpChanged() { $this->loginManager->generateStaySignedInToken($this->clientIpAddress); $this->loginManager->checkLoginState('10.7.157.98'); $this->assertFalse($this->loginManager->isLoggedIn()); } /** * Check user credentials - wrong login supplied */ public function testCheckCredentialsWrongLogin() { $this->assertFalse( $this->loginManager->checkCredentials('', 'b4dl0g1n', $this->password) ); } /** * Check user credentials - wrong password supplied */ public function testCheckCredentialsWrongPassword() { $this->assertFalse( $this->loginManager->checkCredentials('', $this->login, 'b4dp455wd') ); } /** * Check user credentials - wrong login and password supplied */ public function testCheckCredentialsWrongLoginAndPassword() { $this->assertFalse( $this->loginManager->checkCredentials('', 'b4dl0g1n', 'b4dp455wd') ); } /** * Check user credentials - correct login and password supplied */ public function testCheckCredentialsGoodLoginAndPassword() { $this->assertTrue( $this->loginManager->checkCredentials('', $this->login, $this->password) ); } /** * Check user credentials through LDAP - server unreachable */ public function testCheckCredentialsFromUnreachableLdap() { $this->configManager->set('ldap.host', 'dummy'); $this->assertFalse( $this->loginManager->checkCredentials('', $this->login, $this->password) ); } /** * Check user credentials through LDAP - wrong login and password supplied */ public function testCheckCredentialsFromLdapWrongLoginAndPassword() { $this->configManager->set('ldap.host', 'dummy'); $this->assertFalse( $this->loginManager->checkCredentialsFromLdap($this->login, $this->password, function () { return null; }, function () { return false; }) ); } /** * Check user credentials through LDAP - correct login and password supplied */ public function testCheckCredentialsFromLdapGoodLoginAndPassword() { $this->configManager->set('ldap.host', 'dummy'); $this->assertTrue( $this->loginManager->checkCredentialsFromLdap($this->login, $this->password, function () { return null; }, function () { return true; }) ); } }