CSRF令牌生成

时间:2009-11-26 21:53:05

标签: csrf

这是关于生成CSRF令牌的问题。

通常我想基于与用户会话相关联的唯一数据生成令牌,并使用密钥进行散列和腌制。

我的问题是在没有唯一用户数据时生成令牌。没有可用的会话,cookie不是一个选项,IP地址和那种性质的东西是不可靠的。

为什么我不能将字符串包含在请求中作为请求的一部分? 示例伪代码生成令牌并嵌入它:

var $stringToHash = random()
var $csrfToken = hash($stringToHash + $mySecretKey)
<a href="http://foo.com?csrfToken={$csrfToken}&key={$stringToHash}">click me</a>

CSRF令牌的服务器端验证示例

var $stringToHash = request.get('key')
var $isValidToken = hash($stringToHash + $mySecrtKey) == request.get('csrfToken')

哈希中使用的字符串在每个请求上都是不同的。只要它包含在每个请求中,CSRF令牌验证就可以继续进行。由于每个请求都是新的,并且只嵌入在页面中,因此无法访问令牌。然后令牌的安全性落到我只知道的$ mySecretKey上。

这是一种天真的做法吗?我错过了为什么这不起作用的原因?

由于

9 个答案:

答案 0 :(得分:28)

我是否有任何理由不能将字符串作为请求的一部分包含在哈希中?

CSRF代币有两部分。表单中嵌入的令牌,以及其他位置的相应令牌,可以是cookie,存储在会话中或其他地方。这种在别处的使用会阻止页面自包含。

如果在请求中包含要散列的字符串,那么请求是自包含的,因此复制表单是攻击者需要做的全部内容,因为它们具有令牌的两个部分,因此没有保护。 / p>

即使将其放在表单URL中也意味着它是自包含的,攻击者只需复制表单和提交URL。

答案 1 :(得分:7)

试试base64_encode(openssl_random_pseudo_bytes(16))https://github.com/codeguy/php-the-right-way/issues/272#issuecomment-18688498我将其用于https://gist.github.com/mikaelz/5668195

中的表单示例

答案 2 :(得分:2)

CSRF令牌旨在防止(无意)数据修改,这些修改通常应用于POST请求。

因此,您必须为每个更改数据的请求(GET或POST请求)包含CSRF令牌。

  

我的问题是关于   没有时生成令牌   要使用的唯一用户数据。没有会议   可用,cookie不是   选项,IP地址和事情   自然不可靠。

然后只需为每个访问者创建一个唯一的用户ID。 在cookie或URL中包含该ID(如果禁用了cookie)。

编辑:

考虑以下事件:

您已登录到您的Facebook帐户,然后进入某个任意网站。

在该网站中,您提交了一个表单,告知您的浏览器向您的Facebook帐户发送POST请求。

该POST请求可能会更改您的密码或添加评论等,因为Facebook应用程序将您识别为已注册&amp;登录用户。 (除非有另一种阻止机制,如CAPTCHA)

答案 3 :(得分:1)

您只需在网址/表单和Cookie中使用相同的“令牌”即可。这意味着您可以让您的页面通过JavaScript将令牌cookie设置为它想要的任何内容(最好是一些随机值),然后在发送到服务器的所有请求中传递相同的值(作为URI?param或form-领域)。无需让您的服务器生成cookie。

只要我们相信浏览器不允许来自域的页面编辑/读取其他域的cookie,这是安全的,并且这被认为是非常安全的。

让您的服务器生成令牌将假定此令牌可以安全地传输到您的浏览器,而不会被任何CSRF尝试接收(为什么要承担风险?)。虽然您可以将更多逻辑放入服务器生成的令牌中,但是为了防止CSRF,没有必要。

(如果我错了,请告诉我)

答案 4 :(得分:1)

CSRF令牌有多种实现方式。关键是这个csrf令牌是在客户端还是服务器端生成的。因为这两种情况的实施变化很大,而且令牌的熵也是如此。

对于服务器端,SecureRandom是首选方式,但在您的情况下,您希望在识别任何用户之前生成CSRF令牌,window.crypto提供此功能,您可以在其中生成不足够的用于CSRF令牌的字符串。

答案 5 :(得分:0)

我认为最好的想法是基于HMAC制作哈希,即通过一些密码加密哈希这个序列:username + user_id + timestamp。每个请求的哈希必须是不同的,如果你不想在攻击中简单地重放哈希,那么时间戳必须是。

答案 6 :(得分:0)

我想说你的方法有效,因为CSRF攻击是攻击者利用受害者的浏览器来建立登录状态,为什么他们这样做呢?因为在大多数服务器端,会话检查是基于cookie中的SessionID,而cookie是一段数据将自动附加到发送给服务器的HTTP请求。

因此,保护​​CSRF有两个关键因素

  1. 生成质询令牌,并要求客户端以非cookie方式将其传递给服务器,无论是URL参数还是POST表单都可以。
  2. 保持令牌安全,就像您对SessionID所做的那样,例如,使用SSL。
  3. 我建议阅读CSRF Prevention Cheat Sheet

答案 7 :(得分:0)

借助CSRF令牌,我们可以确保对传入请求进行身份验证(知道用户不是黑客)

请注意,我需要以下方法,但即使在stackoverflow上,谷歌也无法帮助我,但下面没有提到代码,但是在收集了stackoverflow答案之后,我就开始了。因此对于进一步搜索/特别是对于初学者来说很有用

我在下面介绍了带有Spring Interceptor的Spring MVC

注意-我已使用Google缓存将盐存储在缓存中以进行重新验证

以下依赖项需要添加pom.xml

    <!-- https://mvnrepository.com/artifact/com.google.guava/guava -->
    <dependency>
        <groupId>com.google.guava</groupId>
        <artifactId>guava</artifactId>
        <version>28.0-jre</version>
    </dependency>

在HandlerInterceptorAdapter实现之下


    package com.august.security;

    import java.security.SecureRandom;
    import java.util.Enumeration;
    import java.util.LinkedList;
    import java.util.List;
    import java.util.concurrent.TimeUnit;

    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    import javax.servlet.http.HttpSession;

    import org.apache.commons.lang3.RandomStringUtils;
    import org.springframework.web.servlet.ModelAndView;
    import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;

    import com.google.common.cache.Cache;
    import com.google.common.cache.CacheBuilder;

    public class CsrfSecurity extends HandlerInterceptorAdapter {
        List<String> urlList= new LinkedList<>();
        private static final String CSRF_TAG = "CSRF-CHECK";

        @SuppressWarnings("unchecked")
        @Override
        public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handleer)
                throws Exception {
            System.out.println("Inside Pre Handler");

            String reqUrl = request.getRequestURI().toString();
            System.out.println("Request URL : " + reqUrl);
            String ipAddress = request.getHeader("X-FORWARDED-FOR");
            if (ipAddress == null) {
                ipAddress = request.getRemoteAddr();
            }
            //local host url http://localhost:8080/august/
            if (request.getRequestURI().contains("/august/")) {
                System.out.println("pre handler return true");
                //it will return and next executed postHandelr method
                //because of on above url my webApplication page working
                return true;
            }
            if (ignoreUrl().contains(request.getRequestURI())) {
                System.out.println("inside ignore uri");
                return true;
            } else {
                System.out.println("CSRF Security intercepter preHandle method started.......");
                String salt = request.getParameter("csrfPreventionSalt");
                HttpSession sessionAttribute = request.getSession();
                Cache<String, Boolean> csrfPreventionSalt = (Cache<String, Boolean>) sessionAttribute
                        .getAttribute("csrfPreventionSalt");
                if (csrfPreventionSalt == null) {
                    System.out.println("Salt not matched session expired..");
                    parameterValuesPrint(request, "saltCacheNotFound");
                    response.sendRedirect("error");
                    return false;
                } else if (salt == null) {
                    parameterValuesPrint(request, "noSaltValue");
                    System.out.println("Potential CSRF detected !! inform ASAP");
                    response.sendRedirect("error");
                    return false;
                } else if (csrfPreventionSalt.getIfPresent(salt) == null) {
                    System.out.println("saltValueMisMatch");
                    System.out.println("Potential CSRF detected !! inform ASAP");
                    response.sendRedirect("error");
                } else {
                    request.setAttribute("csrfPreventionSalt", csrfPreventionSalt);
                }
                return true;
            }

        }

        @SuppressWarnings("unchecked")
        @Override
        public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
                ModelAndView modelAndView) {
            System.out.println("Inside post Handler");
            System.out.println("CSRF Security key generator method started");
            try {
                //localhost url http://localhost:8080/august/
                //api is my controller path so no need to genrate token for api
                if (request.getRequestURI().contains("/august/api/")) {
                    System.out.println("No need to genrate salt for api");
                } else {
                    HttpSession sessionAttribute = request.getSession();
                    Cache<String, Boolean> csrfPreventionSaltCache = (Cache<String, Boolean>) sessionAttribute
                            .getAttribute("csrfPreventionSalt");
                    System.out.println("csrfPreventionSaltCache ::: " + csrfPreventionSaltCache);
                    if (csrfPreventionSaltCache == null) {
                        csrfPreventionSaltCache = CacheBuilder.newBuilder().maximumSize(5000)
                                .expireAfterWrite(20, TimeUnit.MINUTES).build();
                        request.getSession().setAttribute("csrfPreventionSaltCache", csrfPreventionSaltCache);
                    }

                    String salt = RandomStringUtils.random(20, 0, 0, true, true, null, new SecureRandom());
                    System.out.println("csrfPreventionSalt genrated ::: " + salt);
                    csrfPreventionSaltCache.put(salt, Boolean.TRUE);
                    if (modelAndView != null) {
                        System.out.println("Model and view not null and salt is added in modelAndView");
                        modelAndView.addObject("csrfPreventionSalt", salt);
                    }
                }
            } catch (Exception ex) {
                System.out.println(ex.getMessage());
                ex.printStackTrace();
            }
        }

        @Override
        public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
                throws Exception {
            System.out.println("afterCompletion : ");
            if (ex != null) {
                System.out.println("exception : " + ex.getMessage());
                ex.printStackTrace();
            }
        }

        private List<String> ignoreUrl() {
            if(urlList == null) {
                urlList.add("/august/error");
                //add here your ignored url.
            }
            return urlList;
        }

        private void parameterValuesPrint(HttpServletRequest request, String err) {
            StringBuilder reqParamAndValue = new StringBuilder();
            Enumeration<?> params = request.getParameterNames();
            while (params.hasMoreElements()) {
                Object objOri = params.nextElement();
                String param = (String) objOri;
                String value = request.getParameter(param);
                reqParamAndValue = reqParamAndValue.append(param + "=" + value + ",");
            }
            System.out.println(CSRF_TAG + " " + err + "RequestedURL : " + request.getRequestURL());
        }
    }

下面是带有Spring上下文的拦截器注册


package com.august.configuration;

    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.ComponentScan;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.web.servlet.ViewResolver;
    import org.springframework.web.servlet.config.annotation.EnableWebMvc;
    import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
    import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;
    import org.springframework.web.servlet.view.InternalResourceViewResolver;

    import com.august.security.CsrfSecurity;

    @Configuration
    @EnableWebMvc
    @ComponentScan(basePackages="com.august")
    public class SpringConfiguration extends WebMvcConfigurerAdapter  {

        @Bean
        public ViewResolver viewResolver() {
            InternalResourceViewResolver viewResolver = new InternalResourceViewResolver();
            //viewResolver.setViewClass(JstlView.class);
            viewResolver.setPrefix("/WEB-INF/views/");
            viewResolver.setSuffix(".jsp");
            return viewResolver;

        }

        @Bean
        public CsrfSecurity csrfSecurity() {
            return new CsrfSecurity();
        }
        @Override
        public void addInterceptors(InterceptorRegistry registry) {
            registry.addInterceptor(new CsrfSecurity());
        }
    }

下面是我的控制器


    package com.august.v1.appcontroller;

    import javax.servlet.http.HttpSession;

    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Controller;
    import org.springframework.ui.Model;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestMethod;


    @Controller
    public class HomeController {

        @Autowired 
        HttpSession httpSession;

        @RequestMapping("/")
        public String index(Model model) {
            httpSession.invalidate();
            System.out.println("Home page loaded");
            return "index";
        }
    }

下面是我的index.jsp jsp页面


    <%@ page language="java" contentType="text/html; charset=ISO-8859-1"
        pageEncoding="ISO-8859-1" isELIgnored="false"%>
         //don't forget to add isELIgnored="false" on old(version) jsp page because of i 
         //have wasted 1 hour for this
    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
    <html>
    <head>
    <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
    <title>ADS Home</title>
    </head>
    <body>
    <h1>${csrfPreventionSalt}</h1>
    <input type="hidden" name="csrfPreventionSalt" value=${csrfPreventionSalt}>
    </body>
    </html>

对于CSRF的理解-CSRF explanation

答案 8 :(得分:-2)

CSRF利用用户的会话,因此,如果您没有会话,则没有CSRF。