Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

In the Jetty server how can I obtain the client certificate used when client authentication is required?

It is very easy to set up an embedded Jetty server that requests client authentication: One just needs to add the statement SslContextFactory.setNeedClientAuth(true); to the ssl context when configuring the server. Any client that has its certificate in the server's truststore will be able to establish a TLS connection to the server.

However I need to know which client of all the possible trusted clients is currently making a request; in other words I need to know the client certificate used in this connection, in particular in the handler. Does anyone know how to access this certificate or if it is even possible?

like image 669
Brian Reinhold Avatar asked Nov 18 '13 19:11

Brian Reinhold


People also ask

How do I get client authentication certificate?

Create a client certificate request. After receiving the certificate, export it to a password-protected PKCS12 file and send the password and the file to the user. Make sure the file is securely sent.

How does a server authenticate client certificate?

SSL-enabled client software always requires server authentication, or cryptographic validation by a client of the server's identity. The server sends the client a certificate to authenticate itself. The client uses the certificate to authenticate the identity the certificate claims to represent.

Where is client certificate stored?

The client certificates that you generated are, by default, located in 'Certificates - Current User\Personal\Certificates'.


2 Answers

Updated Aug 2019: for Jetty 9.4.20.v20190813 release.

The certificates are added to the Request objects (such as HttpServletRequest), by a HttpConfiguration Customizer.

Specifically, the SecureRequestCustomizer.

Your code to use this would be as follows (scroll down)...

Server server = new Server();

// === HTTP Configuration ===
HttpConfiguration http_config = new HttpConfiguration();
http_config.setSecureScheme("https");
http_config.setSecurePort(8443);
http_config.setOutputBufferSize(32768);
http_config.setRequestHeaderSize(8192);
http_config.setResponseHeaderSize(8192);
http_config.setSendServerVersion(true);
http_config.setSendDateHeader(false);

// === Add HTTP Connector ===
ServerConnector http = new ServerConnector(server,
    new HttpConnectionFactory(http_config));
http.setPort(8080);
http.setIdleTimeout(30000);
server.addConnector(http);

// === Configure SSL KeyStore, TrustStore, and Ciphers ===
SslContextFactory sslContextFactory = new SslContextFactory.Server();
sslContextFactory.setKeyStorePath("/path/to/keystore");
sslContextFactory.setKeyStorePassword("changeme");
sslContextFactory.setKeyManagerPassword("changeme");
sslContextFactory.setTrustStorePath("/path/to/truststore");
sslContextFactory.setTrustStorePassword("changeme");
// OPTIONAL - for client certificate auth (both are not needed)
// sslContextFactory.getWantClientAuth(true)
// sslContextFactory.setNeedClientAuth(true)

// === SSL HTTP Configuration ===
HttpConfiguration https_config = new HttpConfiguration(http_config);
https_config.addCustomizer(new SecureRequestCustomizer()); // <-- HERE

// == Add SSL Connector ===
ServerConnector sslConnector = new ServerConnector(server,
    new SslConnectionFactory(sslContextFactory,"http/1.1"),
    new HttpConnectionFactory(https_config));
sslConnector.setPort(8443);
server.addConnector(sslConnector);

With this SecureRequestCustomizer in place you can access various pieces about the SSL connection from the HttpServletRequest.getAttribute(String) calls using the following attribute names.

javax.servlet.request.X509Certificate

an array of java.security.cert.X509Certificate[]

javax.servlet.request.cipher_suite

the String name of the cipher suite. (same as what is returned from javax.net.ssl.SSLSession.getCipherSuite())

javax.servlet.request.key_size

Integer of the key length in use

javax.servlet.request.ssl_session_id

String representation (hexified) of the active SSL Session ID

like image 151
Joakim Erdfelt Avatar answered Sep 19 '22 20:09

Joakim Erdfelt


There's a standard servlet request property: javax.servlet.request.X509Certificate

It returns an array of X509Certificates.

We use this to get the name and look up the DN from the cert:

x509Cert[0].getSubjectX500Principal().getName()
like image 22
Will Hartung Avatar answered Sep 22 '22 20:09

Will Hartung