Java SSL Certificate Revocation Checking

前端 未结 4 1401
旧巷少年郎
旧巷少年郎 2020-12-15 12:53

I\'m currently writing a network TCP server using SSL. In production, we\'ll finally require clients to authenticate with a certificate.

In order to revoke certifica

相关标签:
4条回答
  • 2020-12-15 13:18

    First of all, the revocation checking that you can configure in jcontrol (from 1.8) applies only for applet and WebStart downloads and signer certificate checks ! For a programmed https client you can use the PKIXRevocationChecker mentioned above, but by my experience the Oracle implementation doesnt support LDAP CDP downloads at all. When I had this problem I needed to implement the full certificate chain check with CRL and immediate CA cert downloads from LDAP, behind my custom TrustManager's checkXXXX functions...

    0 讨论(0)
  • 2020-12-15 13:24

    I figured how to enable CRL checking within a SSLContext without implementing a custom validator, as suggested in the comments.

    It is mainly about properly initializing the SSLContext's TrustManagers with a revocation checker, only a few lines, no custom check logic and the CRL is now checked automatically as well as the verification path.

    Here's a snippet...

    KeyStore ts = KeyStore.getInstance("JKS");
    FileInputStream tfis = new FileInputStream(trustStorePath);
    ts.load(tfis, trustStorePass.toCharArray());
    
    KeyManagerFactory kmf =  KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm());
    TrustManagerFactory tmf = TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm());
    
    // initialize certification path checking for the offered certificates and revocation checks against CLRs
    CertPathBuilder cpb = CertPathBuilder.getInstance("PKIX");
    PKIXRevocationChecker rc = (PKIXRevocationChecker)cpb.getRevocationChecker();
    rc.setOptions(EnumSet.of(
        PKIXRevocationChecker.Option.PREFER_CRLS, // prefer CLR over OCSP
        PKIXRevocationChecker.Option.ONLY_END_ENTITY, 
    PKIXRevocationChecker.Option.NO_FALLBACK)); // don't fall back to OCSP checking
    
    PKIXBuilderParameters pkixParams = new PKIXBuilderParameters(ts, new X509CertSelector());
    pkixParams.addCertPathChecker(rc);
    
    tmf.init( new CertPathTrustManagerParameters(pkixParams) );
    // init KeyManagerFactory
    kmf.init(...)
    
    SSLContext ctx = SSLContext.getInstance("TLS");
    ctx.init(kmf.getKeyManagers), tmf.getTrustManagers(), null);
    

    That essentially did what I needed in my application, checking whether a certificate issued to a client is revoked in our CRL. Only checking the end entity and allowing the CRL check to fail is accepted because its all our infrastructure.

    0 讨论(0)
  • 2020-12-15 13:27

    Is OCSP OK for you?

    bellow code enables OCSP for me:

    // for debugging:
    System.setProperty("javax.net.debug", "all");
    System.setProperty("java.security.debug", "all");
    
    System.setProperty("com.sun.net.ssl.checkRevocation", "true");
    Security.setProperty("ocsp.enable", "true");
    

    failed on CRL for errors like: How to check revocation status of X509Certificate chain using JAVA?

    0 讨论(0)
  • 2020-12-15 13:41

    Notice that disabling revocation checking is a bad security practice. You can do it, but make sure you know the risk!

    The currently accepted answer by @DoNuT works by setting PKIXRevocationChecker.Option.SOFT_FAIL, which causes the validator not to throw an exception even if revocation checking fails. The following answer disables revocation checking altogether, thus it is faster in case you don't want validation at all. This is because performing revocation checks needs contacting CRL distribution points or OCSP servers, and if you don't want that, you need not pay the price.

    You can simply use setRevocationEnabled(false) on an object of type PKIXBuilderParameters.

    // Initialize "anchors" to trusted certificates
    // Initialize "selector" to the certificate you want to validate
    PKIXBuilderParameters pbParams = new PKIXBuilderParameters(anchors, selector);
    
    pbParams.setRevocationEnabled(false); // disable revocation check
    
    CertPathBuilder cpb = CertPathBuilder.getInstance("PKIX");
    CertPathBuilderResult cpbResult = cpb.build(pbParams);
    
    CertPathValidator cpv = CertPathValidator.getInstance("PKIX");
    CertPathValidatorResult result = cpv.validate(cpbResult.getCertPath(), pbParams);
    
    System.out.println(result);
    
    0 讨论(0)
提交回复
热议问题