FOSUserBundle登录时的区域设置开关



我正在尝试在应用程序的登录屏幕中切换语言环境。为了做到这一点,我在登录页面上设置了指向site.com/(默认语言环境)和site.com/en(我支持的第二个语言环境)的链接。我一登录,切换就像施了魔法一样。但是,如果我还没有通过身份验证,登录总是返回到默认语言环境。我的理解是,如果我使用FOSUserBundle中的命名路由,那么它应该能够自动处理区域设置,但我不能让它工作。

app/config/security.yml

security:
    encoders:
        FOSUserBundleModelUserInterface: sha512
    role_hierarchy:
        ROLE_ADMIN:       ROLE_USER
        ROLE_SUPER_ADMIN: ROLE_ADMIN
    providers:
        fos_userbundle:
            id: fos_user.user_provider.username
    firewalls:
        main:
            pattern: ^/
            form_login:
                provider: fos_userbundle
                csrf_provider: form.csrf_provider
                login_path: fos_user_security_login
                check_path: fos_user_security_check
            logout:       true
            anonymous:    true
    access_control:
        - { path: ^/_wdt, roles: IS_AUTHENTICATED_ANONYMOUSLY }         # allow wdt for debugging
        - { path: ^/_profiler/, role: IS_AUTHENTICATED_ANONYMOUSLY }    # allow profiler for debugging
        - { path: ^/bundles/, role: IS_AUTHENTICATED_ANONYMOUSLY }      # allow assets to be loaded anonymously
        - { path: ^/login$, role: IS_AUTHENTICATED_ANONYMOUSLY }
        - { path: ^/register, role: IS_AUTHENTICATED_ANONYMOUSLY }
        - { path: ^/resetting, role: IS_AUTHENTICATED_ANONYMOUSLY }
        - { path: ^/admin, role: ROLE_ADMIN }
        - { path: ^/, role: ROLE_USER } 

app/config/routing.yml

# FOS User bundle
fos_user_security:
    resource: "@FOSUserBundle/Resources/config/routing/security.xml"
fos_user_profile:
    resource: "@FOSUserBundle/Resources/config/routing/profile.xml"
    prefix: /profile
#fos_user_register:
#    resource: "@FOSUserBundle/Resources/config/routing/registration.xml"
#    prefix: /register
fos_user_resetting:
    resource: "@FOSUserBundle/Resources/config/routing/resetting.xml"
    prefix: /resetting
fos_user_change_password:
    resource: "@FOSUserBundle/Resources/config/routing/change_password.xml"
    prefix: /profile

任何提示非常感谢,因为我已经被这个困了几天了

最终的解决方案是在FOSUserBundle的路由导入前加上区域设置前缀:

# FOS User bundle
fos_user_security:
    resource: "@FOSUserBundle/Resources/config/routing/security.xml"
    prefix: /{_locale}
fos_user_profile:
    resource: "@FOSUserBundle/Resources/config/routing/profile.xml"
    prefix: /{_locale}/profile
fos_user_register:
    resource: "@FOSUserBundle/Resources/config/routing/registration.xml"
    prefix: /{_locale}/register
fos_user_resetting:
    resource: "@FOSUserBundle/Resources/config/routing/resetting.xml"
    prefix: /{_locale}/resetting
fos_user_change_password:
    resource: "@FOSUserBundle/Resources/config/routing/change_password.xml"
    prefix: /{_locale}/profile

还要更改防火墙以允许匿名路由中的区域设置,并配置logout_path:

security:
    encoders:
        FOSUserBundleModelUserInterface: sha512
    role_hierarchy:
        ROLE_ADMIN:       ROLE_USER
        ROLE_SUPER_ADMIN: ROLE_ADMIN
    providers:
        fos_userbundle:
            id: fos_user.user_provider.username
    firewalls:
        main:
            pattern: ^/
            form_login:
                provider: fos_userbundle
                csrf_provider: form.csrf_provider
                login_path: fos_user_security_login
                check_path: fos_user_security_check
            logout:
                path:   fos_user_security_logout
            anonymous:    true
    access_control:
        - { path: ^/_wdt, roles: IS_AUTHENTICATED_ANONYMOUSLY }         # allow wdt for debugging
        - { path: ^/_profiler/, role: IS_AUTHENTICATED_ANONYMOUSLY }    # allow profiler for debugging
        - { path: ^/bundles/, role: IS_AUTHENTICATED_ANONYMOUSLY }      # allow assets to be loaded anonymously
        - { path: ^/[a-z]+/login$, role: IS_AUTHENTICATED_ANONYMOUSLY }
        - { path: ^/[a-z]+/register, role: IS_AUTHENTICATED_ANONYMOUSLY }
        - { path: ^/[a-z]+/resetting, role: IS_AUTHENTICATED_ANONYMOUSLY }
        - { path: ^/admin, role: ROLE_ADMIN }
        - { path: ^/, role: ROLE_USER } # this must be the last match, as url patterns are checked in the order they appear

从长远来看,使用JMSI18nRoutingBundle可能会更好,但是当我尝试它时,它并没有作为一个解决方案,而且这个项目的预算不允许我开始弄清楚为什么不这样做,所以这将留给未来的更新。

我不知道你是如何处理语言环境检测/切换的,但是使用JMSI18nRoutingBundle,你可以这样做:

添加所需的bundle到composer.json:

"require": {
    ...
    "jms/i18n-routing-bundle": "1.1.*",
    "jms/translation-bundle": "1.1.*",
    "friendsofsymfony/user-bundle": "1.3.*"
},

配置bundle:

jms_i18n_routing:
    default_locale: en
    locales: [en, fr, it, sp]
    strategy: prefix

引导bundle:

$bundles = array(
    ...
    new JMSI18nRoutingBundleJMSI18nRoutingBundle(),
    new FOSUserBundleFOSUserBundle(),
);

修改现有路由,在它们前面加上所需的区域设置:

access_control:
    # Routes are prefixed by ther user locale.
    - { path: ^/[^/]+/login$, role: IS_AUTHENTICATED_ANONYMOUSLY }
    - { path: ^/[^/]+/register, role: IS_AUTHENTICATED_ANONYMOUSLY }
    - { path: ^/[^/]+/resetting, role: IS_AUTHENTICATED_ANONYMOUSLY }
    - { path: ^/[^/]+/admin, role: ROLE_ADMIN }
    - { path: ^/[^/]+/, role: ROLE_USER }

现在它应该工作了!

我的解决方案:您必须为be_simple_i18n类型格式重新定义fos_user路由。(我选择了yaml版本):

在app/config/routing.yml

[...]
#register the path to the file with the be_simple_i18n type fos_user routes
fos_user:       
    resource: "config/routing/fos_user_i18n.yml"
    type: be_simple_i18n
my_yaml_i18n_routes:
    resource: "config/routing/i18n.yml"
    type: be_simple_i18n

在app/资源/config/路由/fos_user_i18n.yml:

#you have to make entries for all the fos_user routes here!
#you can find them all in vendor/friendsofsymfony/user-bundle/Resources/routing
fos_user_security_login:
    locales:  { en: "/login", de: "/anmelden" }
    defaults: { _controller: FOSUserBundle:Security:login }
fos_user_security_check:
    locales:  { en: "/login_check", de: "/login_pruefung" }
    defaults: { _controller: FOSUserBundle:Security:check }
#...
在app/config/security.yml

firewalls:       
    [...]
    main:
        pattern: ^/
        form_login:
            provider: fos_userbundle
            login_path: fos_user_security_login
            check_path: fos_user_security_check
            csrf_provider: security.csrf.token_manager # Use form.csrf_provider instead for Symfony <2.4
        logout:
            path: fos_user_security_logout
            target: #where_no_one_has_gone_before
        [...]
access_control:
    - { path: ^/login$, role: IS_AUTHENTICATED_ANONYMOUSLY }
    - { path: ^/anmelden$, role: IS_AUTHENTICATED_ANONYMOUSLY }
    - { path: ^/register, role: IS_AUTHENTICATED_ANONYMOUSLY }
    - { path: ^/registrieren, role: IS_AUTHENTICATED_ANONYMOUSLY }
    [...]

我一直在努力解决这个问题。我希望将纯登录(会话中没有URL)重定向到适当的本地化页面。我可以在Symfony文档中找到答案:

# app/config/security.yml
security:
# ...
 firewalls:
    main:
        form_login:
            # ...
            default_target_path: index [ or your named route ]

最新更新