Пользовательский фильтр Spring Security - PullRequest
8 голосов
/ 12 мая 2011

Я хотел бы настроить Spring security 3.0.5 и изменить URL-адрес входа на / login вместо / j_spring_security_check.

Что мне нужно сделать, это разрешить вход в каталог «/» и защитить страницу «/admin/report.html».

Прежде всего я создаю свой собственный фильтр с помощью учебника и исходного кода Spring Security:

public class MyFilter extends AbstractAuthenticationProcessingFilter {
    private static final String DEFAULT_FILTER_PROCESSES_URL = "/login";
    private static final String POST = "POST";
    public static final String SPRING_SECURITY_FORM_USERNAME_KEY = "j_username";
    public static final String SPRING_SECURITY_FORM_PASSWORD_KEY = "j_password";
    public static final String SPRING_SECURITY_LAST_USERNAME_KEY = "SPRING_SECURITY_LAST_USERNAME";

    private String usernameParameter = SPRING_SECURITY_FORM_USERNAME_KEY;
    private String passwordParameter = SPRING_SECURITY_FORM_PASSWORD_KEY;

    protected MyFilter() {
        super(DEFAULT_FILTER_PROCESSES_URL);
    }

    @Override
    public Authentication attemptAuthentication(HttpServletRequest request,
                                                HttpServletResponse response) 
                          throws AuthenticationException, IOException, ServletException {
        String username = obtainUsername(request);
        String password = obtainPassword(request);

        if (username == null) {
            username = "";
        }

        if (password == null) {
            password = "";
        }

        username = username.trim();
        UsernamePasswordAuthenticationToken authRequest = new UsernamePasswordAuthenticationToken(username, password);
        HttpSession session = request.getSession(false);
        if (session != null || getAllowSessionCreation()) {
            request.getSession().setAttribute(SPRING_SECURITY_LAST_USERNAME_KEY, TextEscapeUtils.escapeEntities(username));
        }
        setDetails(request, authRequest);

        return this.getAuthenticationManager().authenticate(authRequest);
    }

    protected void setDetails(HttpServletRequest request, UsernamePasswordAuthenticationToken authRequest) {
        authRequest.setDetails(authenticationDetailsSource.buildDetails(request));
    }

    @Override
    public void doFilter(ServletRequest req, ServletResponse res,
                         FilterChain chain) throws IOException, ServletException {
        final HttpServletRequest request = (HttpServletRequest) req;
        final HttpServletResponse response = (HttpServletResponse) res;
        if (request.getMethod().equals(POST)) {
            // If the incoming request is a POST, then we send it up
            // to the AbstractAuthenticationProcessingFilter.
            super.doFilter(request, response, chain);
        } else {
            // If it's a GET, we ignore this request and send it
            // to the next filter in the chain.  In this case, that
            // pretty much means the request will hit the /login
            // controller which will process the request to show the
            // login page.
            chain.doFilter(request, response);
        }
    }

    protected String obtainUsername(HttpServletRequest request) {
        return request.getParameter(usernameParameter);
    }

    protected String obtainPassword(HttpServletRequest request) {
        return request.getParameter(passwordParameter);
    }
}

после этого я делаю следующие изменения в xml

 <security:http auto-config="true">
        <!--<session-management session-fixation-protection="none"/>-->
        <security:custom-filter ref="myFilter" before="FORM_LOGIN_FILTER"/>
        <security:intercept-url pattern="/admin/login.jsp*" filters="none"/>
        <security:intercept-url pattern="/admin/report.html" access="ROLE_ADMIN"/>
        <security:form-login login-page="/admin/login.jsp" login-processing-url="/login" always-use-default-target="true"/>
        <security:logout logout-url="/logout" logout-success-url="/login.jsp" invalidate-session="true"/>
    </security:http>   
<security:authentication-manager alias="authenticationManager">
  <security:authentication-provider>
    <security:password-encoder hash="md5" />
    <security:user-service>
    <!-- peter/opal -->
      <security:user name="peter" password="22b5c9accc6e1ba628cedc63a72d57f8" authorities="ROLE_ADMIN" />
     </security:user-service>
  </security:authentication-provider>
</security:authentication-manager>
<bean id="myFilter" class="com.vanilla.springMVC.controllers.MyFilter">
<property name="authenticationManager" ref="authenticationManager"/>
</bean>

и тогда у меня есть JSP с моим кодом.

<form action="../login" method="post">
    <label for="j_username">Username</label>
    <input type="text" name="j_username" id="j_username" />
    <br/>
    <label for="j_password">Password</label>
    <input type="password" name="j_password" id="j_password"/>
    <br/>
    <input type='checkbox' name='_spring_security_remember_me'/> Remember me on this computer.
    <br/>
    <input type="submit" value="Login"/>
</form>

при попытке перейти к /admin/report.html я перенаправлен на страницу входа. но после отправки учетных данных я получаю:

HTTP Status 404 - /SpringMVC/login/

type Status report

message /SpringMVC/login/

description The requested resource (/SpringMVC/login/) is not available.

Похоже, у меня проблемы с конфигурацией, но я не могу понять, что вызвало это. Вы можете помочь?

Ответы [ 2 ]

10 голосов
/ 22 мая 2012

Я опаздываю примерно на 12 месяцев, но для того, чтобы настроить URL-адрес для входа в форму Spring Security, вам не нужно создавать свой собственный фильтр. Один из атрибутов тега form-login позволяет вам установить пользовательский URL. Фактически вы также можете изменить имена полей j_username и j_password по умолчанию, используя атрибуты тега form-login. Вот пример:

<form-login login-page="/login" login-processing-url="/login.do" default-target-url="/" always-use-default-target="true" authentication-failure-url="/login?error=1" username-parameter="username" password-parameter="password"/>
0 голосов
/ 10 июня 2011

Я думаю, что @Ischin правильно задавать вопрос о URL действия формы. Попробуйте ввести полный путь и посмотреть, работает ли это. Если это произойдет, вы можете работать оттуда, чтобы выяснить, что не соответствует.

Единственное, что я могу проверить, это сопоставление фильтров в вашем файле web.xml. Поскольку вы заходите на страницу входа, у вас есть эта настройка, но я бы хотел убедиться, что вы перехватываете не только URL-адреса с определенными расширениями и т. Д.

Также, как к вашему сведению, если вы хотите, чтобы запрос (как только форма входа в систему аутентифицирует пользователя) для перехода к защищенному ресурсу (в данном случае /admin/report.html), вам следует удалить форму: логин всегда -использовать по умолчанию целевой = «истина». Установка этого флага в true приведет к тому, что запрос будет всегда идти к целевому URL-адресу по умолчанию, что обычно не то, что вам нужно. Из пружинных документов по безопасности :

Сопоставляется со свойством defaultTargetUrl. из UsernamePasswordAuthenticationFilter. Если не установлено, значением по умолчанию является "/" (корень приложения). Пользователь будет перешли на этот URL после входа в систему, при условии, что их не попросили войти при попытке доступа к защищенному ресурс, когда они будут приняты первоначально запрошенный URL.

...