鍍金池/ 教程/ PHP/ 如何創(chuàng)建自定義表單密碼驗(yàn)證器
如何以非繼承方式自定義方法
如何創(chuàng)建事件監(jiān)聽(tīng)器
如何以非繼承方式擴(kuò)展一個(gè)類
如何記錄消息到不同的文件
如何掌握并創(chuàng)建新的環(huán)境
如何使用 Doctrine DBAL
"XXX is deprecated" E-USER-DEPRECATED 的警告是什么意思?
在登錄表單中使用 CSRF 保護(hù)
如何注冊(cè)自定義 DQL 函數(shù)
如何為表單類配置空數(shù)據(jù)
如何嵌入集合表單
如何創(chuàng)建自定義認(rèn)證提供者
如何使用 Apache Router
如何組織配置文件
部署在 Microsoft Azure 云
如何在路由參數(shù)中允許"/"字符
如何在安全,路由,服務(wù)和驗(yàn)證中使用表達(dá)式
如何對(duì)顯示控制臺(tái)信息配置 Monolog
如何為一個(gè) Bundle 創(chuàng)建友好的配置
如何改變默認(rèn)的目標(biāo)路徑行為
如何在運(yùn)行測(cè)試之前自定義引導(dǎo)過(guò)程
如何從路由向控制器傳輸額外的信息
如何從數(shù)據(jù)庫(kù)(實(shí)體提供者)讀取安全用戶
如何從 Controller 調(diào)用一個(gè)命令
如何創(chuàng)建自定義表單密碼驗(yàn)證器
如何使用內(nèi)建的 PHP Web 服務(wù)器
如何在功能測(cè)試中用 Token 模擬認(rèn)證
配置 Session 文件的保存目錄
理解前端控制器、內(nèi)核及環(huán)境如何協(xié)同工作
如何實(shí)現(xiàn)一個(gè)簡(jiǎn)單的注冊(cè)表單
如何使用 Doctrine 擴(kuò)展:Timestampable, Sluggable, Translatable 等等
如何使用多個(gè)實(shí)體管理器和連接
如何自定義表單渲染
如何安裝第三方 Bundles
使用預(yù)認(rèn)證安全防火墻
如何簡(jiǎn)化多個(gè) Bundle 的配置
會(huì)話代理實(shí)例
安裝 Composer
如何冒充一個(gè)用戶
如何注冊(cè)一個(gè)新的請(qǐng)求格式和 MIME 類型
如何在功能測(cè)試中使用分析器
如何在服務(wù)容器內(nèi)設(shè)置外部參數(shù)
如何重寫(xiě) Symfony 默認(rèn)的目錄結(jié)構(gòu)
如何在一個(gè) Symfony 控制器中創(chuàng)建一個(gè) SOAP 的 Web 服務(wù)
如何使用序列化
部署在 Platform.sh
升級(jí)一個(gè)副版本
如何寫(xiě)一個(gè)自定義的 Twig 擴(kuò)展
如何在 SubVersion 中創(chuàng)建并保存一個(gè) Symfony 項(xiàng)目
使用 PHP 庫(kù)聯(lián)合,編譯和最小化 Web 資產(chǎn)
如何創(chuàng)建一個(gè)自定義的數(shù)據(jù)收集器
如何使用和注冊(cè)命名空間路徑
如何使用 Monolog 記錄日志
如何建立一個(gè)傳統(tǒng)的登錄表單
如何強(qiáng)制路由總是使用 HTTPS 或者 HTTP
如何在模板中使用 PHP 而不是 Twig
如何動(dòng)態(tài)選擇密碼加密算法
避免匿名用戶開(kāi)始 Session 會(huì)話
如何測(cè)試 Doctrine 倉(cāng)庫(kù)
如何在功能測(cè)試中測(cè)試一封電子郵件被發(fā)送
Symfony2 與 Symfony1 的區(qū)別
使用結(jié)尾反斜線重定向 URL
(configuration)如何在數(shù)據(jù)庫(kù)中使用 PdoSessionHandler 存儲(chǔ) Sessions
如何使用匹配器有條件地啟用分析器
部署在 Heroku 云
如何不用自定義控制器配置重定向
如何在 Bundle 內(nèi)部加載服務(wù)配置
如何處理不同的錯(cuò)誤級(jí)別
如何在應(yīng)用中保護(hù)服務(wù)和方法
如何對(duì)表單單元測(cè)試
如何把命令定義為服務(wù)
如何配置 Monolog 從日志中排除 404 錯(cuò)誤
如何使用控制臺(tái)
如何測(cè)試與數(shù)據(jù)庫(kù)交互的代碼
如何在路由中使用除了 GET 和 POST 的 HTTP 方法
如何使用云服務(wù)發(fā)送電子郵件
如何創(chuàng)建一個(gè)控制臺(tái)命令
在遺留的應(yīng)用上使用 Symfony Session
如何使用高級(jí)的訪問(wèn)控制列表
如何不用一個(gè)自定義的控制器渲染一個(gè)模板
如何重寫(xiě)部分 Bundle
升級(jí)一個(gè)主版本
安全訪問(wèn)控制是如何工作的
如何使用 Bundle 的繼承來(lái)重寫(xiě)部分 Bundle
如何使用 Voter 檢查用戶權(quán)限
如何為多個(gè) Doctrine 的實(shí)現(xiàn)提供模型類
如何使用作用域
如何部署一個(gè) Symfony 應(yīng)用
如何用 "inherit-data" 減少代碼冗余
如何注冊(cè)事件監(jiān)聽(tīng)器和訂閱
使用 Bower 安裝 Symfony
如何創(chuàng)建一個(gè)自定義路由加載器
如何創(chuàng)建一個(gè)自定義的驗(yàn)證限制
在獨(dú)立注入類中使用參數(shù)
如何使用 Assetic 和 Twig Functions 進(jìn)行圖像優(yōu)化
如何利用表單事件動(dòng)態(tài)修改表單
如何在過(guò)濾器的前后設(shè)置事件分發(fā)器
如何在 Bundle 中使用 Compiler Passes
緩存包含 CSRF 保護(hù)表單的頁(yè)面
如何注入變量到所有的模板(如全局變量)
如何創(chuàng)建一個(gè)自定義表單域類型
如何限定防火墻使其只允許通過(guò)指定請(qǐng)求
如何把 Controller 定義為服務(wù)
如何使用 Gmail 發(fā)送郵件
升級(jí)一個(gè)補(bǔ)丁版本
如何創(chuàng)建一個(gè)表單類型擴(kuò)展
如何對(duì)不同的 URL 強(qiáng)制使用 HTTPS 或者 HTTP
如何使用 Varnish 加速我的 Web 站點(diǎn)
如何定義虛擬類和接口之間的關(guān)系
如何自定義登錄表單
如何測(cè)試多個(gè)客戶端的交互
PSR-7 Bridge
如何使用 YUI Compressor 裁剪 Javascripts 和 Stylesheets
在用戶的 Session 中使用局部 "Sticky"
如何定制錯(cuò)誤頁(yè)
如何從已存在的數(shù)據(jù)庫(kù)中生成實(shí)體
如何用 Doctrine 上傳文件
可復(fù)用 Bundles 的最佳實(shí)踐
如何發(fā)送一封電子郵件
如何將 Assetic Filter 應(yīng)用到具體的文件擴(kuò)展名上
切換分析器存儲(chǔ)
如何上傳文件
限制 Session 元數(shù)據(jù)的寫(xiě)入
如何使用多用戶提供者
如何使用數(shù)據(jù)轉(zhuǎn)換
配置一個(gè) Web 服務(wù)器
如何編程訪問(wèn)分析器數(shù)據(jù)
如何在路由中使用服務(wù)容器參數(shù)
如何在控制臺(tái)生成 URL 和發(fā)送郵件
如何將你的開(kāi)發(fā)環(huán)境優(yōu)化為調(diào)試環(huán)境
如何在控制臺(tái)命令中啟用日志
如何在功能測(cè)試中模擬 HTTP 認(rèn)證
如何使用 API 驗(yàn)證用戶
如何移除 AcmeDemoBundle
控制臺(tái)命令
如何配置 Symfony 使其工作在負(fù)載均衡和反轉(zhuǎn)代理
如何添加“記住我”登錄功能
如何使用 Assetic 進(jìn)行資產(chǎn)管理
如何限定防火墻使其接受指定主機(jī)
如何使用訪問(wèn)控制列表(ACLs)
如何裁剪 CSS/JS 文件(使用 UglifyJS 和 UglifyCSS)
如何緩存電子郵件
如何使用 submit() 函數(shù)處理表單提交
如何在 Git 中創(chuàng)建并保存一個(gè) Symfony 項(xiàng)目
如何創(chuàng)建自定義用戶提供者
如何對(duì)電子郵件錯(cuò)誤配置 Monolog
如何在開(kāi)發(fā)時(shí)使用電子郵件

如何創(chuàng)建自定義表單密碼驗(yàn)證器

想象一下,如果您想要使您的網(wǎng)站只能在下午兩點(diǎn)到四點(diǎn)之間才能被訪問(wèn)。在 Symfony 2.4 之前必須創(chuàng)建一個(gè)自定義的令牌、 工廠、 監(jiān)聽(tīng)器和提供者才能實(shí)現(xiàn)。在本節(jié)中,您將學(xué)習(xí)如何在一個(gè)登錄表單(即您的用戶提交他們的用戶名和密碼的頁(yè)面)中實(shí)現(xiàn)上述功能 。在 Symfony 2.6 之前,您必須使用密碼編碼器來(lái)驗(yàn)證用戶的密碼。

密碼身份驗(yàn)證器

2.6 在 Symfony 2.6 介紹了 UserPasswordEncoderInterface 接口。

首先,創(chuàng)建新的類來(lái)實(shí)現(xiàn) SimpleFormAuthenticatorInterface 接口。最終,這將允許您創(chuàng)建自定義邏輯來(lái)對(duì)用戶進(jìn)行身份驗(yàn)證:

// src/Acme/HelloBundle/Security/TimeAuthenticator.php
namespace Acme\HelloBundle\Security;

use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\Security\Core\Authentication\SimpleFormAuthenticatorInterface;
use Symfony\Component\Security\Core\Authentication\Token\TokenInterface;
use Symfony\Component\Security\Core\Authentication\Token\UsernamePasswordToken;
use Symfony\Component\Security\Core\Encoder\UserPasswordEncoderInterface;
use Symfony\Component\Security\Core\Exception\AuthenticationException;
use Symfony\Component\Security\Core\Exception\UsernameNotFoundException;
use Symfony\Component\Security\Core\User\UserProviderInterface;

class TimeAuthenticator implements SimpleFormAuthenticatorInterface
{
    private $encoder;

    public function __construct(UserPasswordEncoderInterface $encoder)
    {
        $this->encoder = $encoder;
    }

    public function authenticateToken(TokenInterface $token, UserProviderInterface $userProvider, $providerKey)
    {
        try {
            $user = $userProvider->loadUserByUsername($token->getUsername());
        } catch (UsernameNotFoundException $e) {
            throw new AuthenticationException('Invalid username or password');
        }

        $passwordValid = $this->encoder->isPasswordValid($user, $token->getCredentials());

        if ($passwordValid) {
            $currentHour = date('G');
            if ($currentHour < 14 || $currentHour > 16) {
                throw new AuthenticationException(
                    'You can only log in between 2 and 4!',
                    100
                );
            }

            return new UsernamePasswordToken(
                $user,
                $user->getPassword(),
                $providerKey,
                $user->getRoles()
            );
        }

        throw new AuthenticationException('Invalid username or password');
    }

    public function supportsToken(TokenInterface $token, $providerKey)
    {
        return $token instanceof UsernamePasswordToken
            && $token->getProviderKey() === $providerKey;
    }

    public function createToken(Request $request, $username, $password, $providerKey)
    {
        return new UsernamePasswordToken($username, $password, $providerKey);
    }
}

它是怎樣工作的

很好!現(xiàn)在你只需要設(shè)置一些配置。但首先,你可以了解更多有關(guān)此類中的每個(gè)方法的作用。

1)createToken 方法

當(dāng) Symfony 開(kāi)始處理請(qǐng)求,createToken() 方法會(huì)被調(diào)用,在這個(gè)方法中會(huì)創(chuàng)建一個(gè) TokenInterface 對(duì)象,并且該對(duì)象包含的所有您需要的信息都會(huì)在 authenticateToken() 中進(jìn)行用戶身份驗(yàn)證 (例如用戶名和密碼) 。

您在這里創(chuàng)建的所有令牌對(duì)象隨后都將通過(guò) authenticateToken() 方法傳遞給您。

2)supportsToken 方法

當(dāng) Symfony 調(diào)用 createToken() 方法后,它將會(huì)調(diào)用您創(chuàng)建的類中的 supportsToken() 方法(和任何其它身份驗(yàn)證監(jiān)聽(tīng)器) 來(lái)弄清誰(shuí)應(yīng)該處理令牌。這僅僅是一種允許同一個(gè)防火墻使用幾種身份驗(yàn)證機(jī)制的方法。 (通過(guò)這種方式,例如您可以首先通過(guò)證書(shū)或者 API 密鑰來(lái)對(duì)用戶進(jìn)行身份驗(yàn)證然后再回退到登錄表單)。

大多數(shù)情況下,你只需要確保在這個(gè)方法中,給 createToken() 方法中建立的令牌返回一個(gè)真值。您的程序邏輯應(yīng)該如同這個(gè)例子一樣。

3)authenticateToken 方法

如果 supportsToken 方法返回 true,Symfony 將會(huì)調(diào)用 authenticateToken() 方法。您現(xiàn)在應(yīng)該做的是檢查令牌是否允許首先通過(guò)的用戶提供程序來(lái)獲得用戶對(duì)象,然后通過(guò)檢查密碼和當(dāng)前時(shí)間來(lái)登錄。

關(guān)于如何獲取用戶對(duì)象以及確定令牌是否有效 (例如檢查密碼)的"流",可能會(huì)隨著您的需求而改變。

最終,您的任務(wù)是返回一個(gè)新的并且已經(jīng)"身份驗(yàn)證"過(guò)的令牌對(duì)象(即:至少給它設(shè)定了一個(gè)角色),并且這個(gè)令牌對(duì)象中含有用戶對(duì)象。

在這個(gè)方法中,需要使用密碼編碼器被來(lái)檢查密碼的有效性:

$passwordValid = $this->encoder->isPasswordValid($user, $token->getCredentials());

這已經(jīng)是 Symfony 中可用的服務(wù),并且它使用的是編碼密鑰下的安全配置 (例如 security.yml) 中的密碼算法。下面,您將會(huì)看到如何把它注入 到 TimeAuthenticator 中。

配置

現(xiàn)在,把您的 TimeAuthenticator 作為一種服務(wù)來(lái)配置:

YAML:

# app/config/config.yml
services:
    # ...

    time_authenticator:
        class:     Acme\HelloBundle\Security\TimeAuthenticator
        arguments: ["@security.password_encoder"]

XML:

<!-- app/config/config.xml -->
<?xml version="1.0" ?>
<container xmlns="http://symfony.com/schema/dic/services"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://symfony.com/schema/dic/services
        http://symfony.com/schema/dic/services/services-1.0.xsd">
    <services>
        <!-- ... -->

        <service id="time_authenticator"
            class="Acme\HelloBundle\Security\TimeAuthenticator"
        >
            <argument type="service" id="security.password_encoder" />
        </service>
    </services>
</container>

PHP:

// app/config/config.php
use Symfony\Component\DependencyInjection\Definition;
use Symfony\Component\DependencyInjection\Reference;

// ...

$container->setDefinition('time_authenticator', new Definition(
    'Acme\HelloBundle\Security\TimeAuthenticator',
    array(new Reference('security.password_encoder'))
));

接著,使用 simple_form 密鑰在安全配置中的防火墻中激活它:

YAML:

# app/config/security.yml
security:
    # ...

    firewalls:
        secured_area:
            pattern: ^/admin
            # ...
            simple_form:
                authenticator: time_authenticator
                check_path:    login_check
                login_path:    login

XML:

<!-- app/config/security.xml -->
<?xml version="1.0" encoding="UTF-8"?>
<srv:container xmlns="http://symfony.com/schema/dic/security"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:srv="http://symfony.com/schema/dic/services"
    xsi:schemaLocation="http://symfony.com/schema/dic/services
        http://symfony.com/schema/dic/services/services-1.0.xsd">
    <config>
        <!-- ... -->

        <firewall name="secured_area"
            pattern="^/admin"
            >
            <simple-form authenticator="time_authenticator"
                check-path="login_check"
                login-path="login"
            />
        </firewall>
    </config>
</srv:container>

PHP:

// app/config/security.php

// ..

$container->loadFromExtension('security', array(
    'firewalls' => array(
        'secured_area'    => array(
            'pattern'     => '^/admin',
            'simple_form' => array(
                'provider'      => ...,
                'authenticator' => 'time_authenticator',
                'check_path'    => 'login_check',
                'login_path'    => 'login',
            ),
        ),
    ),
));

Simple_form 密鑰具有和正常的 form_login 相同的選項(xiàng),但在 simple_form 密鑰中具有指向新服務(wù)的附加的身份驗(yàn)證器密鑰。有關(guān)詳細(xì)信息,請(qǐng)參閱表單登錄配置。

一般情況下,如果您對(duì)如何創(chuàng)建一個(gè)新的登錄表單不熟悉或者是不明白 check_pathlogin_path 選項(xiàng),請(qǐng)參閱如何自定義您的表單登錄