Jakarta NoSQL Driver for the AppDev Pack, Part 2
Sep 26, 2022, 9:55 AM
- Jakarta NoSQL Driver for the AppDev Pack, Part 1
- Jakarta NoSQL Driver for the AppDev Pack, Part 2
In my last post, I talked about how I implemented a partial Jakarta NoSQL driver using the AppDev Pack as a back end instead of the Notes.jar classes used by the primary implementation. Though the limitations in the ADP mean that it lacks a number of useful features compared to the primary one, it was still an interesting experiment and has the nice side effect of working with essentially any Java app server and Java version 8 or above.
Beyond the Proton API calls, the driver brought up the interesting topic of handling authentication. Proton has three ways of working in this regard:
- Anonymous, which is what you might expect based on how that works elsewhere in Domino. This is easy but not particularly useful except in specific circumstances.
- Client certificate authentication, where you create a TLS keychain for a given user and associate it with a Directory user (e.g.
CN=My Proton App/O=MyOrg
), and then your app performs all operations as that user. This is basically like if you ran a remote app with NRPC using a client Notes ID. - Act-as-User, which builds on the above authentication by configuring an OAuth broker service that can hand out OIDC tokens on behalf of named users. This is sort of like server-to-server communication with the "Trusted Servers" config field in the server doc, but different in key ways.
Client Certificate Authentication
When doing app development, the middle route makes sense as your starting point, since most of your actual work will likely be the same regardless of whether you later then add on act-as-user support. For that, you'll follow the guide to set up your TLS keychain and then feed those files to the com.hcl.domino.db.model.Server
object:
1 2 3 4 5 6 7 | Path base = Paths.get(BASE_PATH); File ca = base.resolve("rootcrt.pem").toFile(); File cert = base.resolve("clientcrt.pem").toFile(); File key = base.resolve("clientkey.pem").toFile(); Server server = new Server("ceres.frostillic.us", 3003, ca, cert, key, null, null, Executors.newSingleThreadExecutor()); |
The use of java.io.File
classes here is a bit of a shame, but not the end of the world. In practice, you'd likely store your keychain somewhere on the filesystem anyway and then feed the BASE_PATH
property to your app via an environment variable. Otherwise, if they were pulled from some other source, you could use Files.createTempFile
to store them on the filesystem while your app is running. Those null
s are for the passphrases for the certificates, so they might be populated for you. For the last parameter, making a new executor is fine, but you might want to hand it a ManagedExecutorService
.
You can initialize this connection basically anywhere - I put it in a ServletRequestListener
to init and term the object per-request, but I think it would actually be fine to do it in a ServletContextListener
and keep it app-wide. I did it out of habit from Notes.jar and its heavy requirements on threads and the way Session
objects are different per-user, but that's not how Proton connections work.
This form of authenticate is a prerequisite for Act-as-User below, but it might suffice for your needs anyway. For example, if you have a "utility" app, like a bot that looks up data and posts messages to Slack or something, you can call it good here.
Act-as-User
But though the above may suffice sometimes, the integration of user identity with data access is one of the hallmarks of Domino, so Domino apps that wouldn't need Act-as-User support are few and far between.
Act-as-User is a bit daunting to set up, though. In traditional server-to-server communication in Domino, it suffices to just add a server's name or group to the "Trusted Servers" list in the server doc of the server being accessed. Then, it will trust any old name that the app-housing server sends along. Generally, this will be a user that was already authenticated with Domino, like using an XPages-supplied Session
object to access a remote server, but it doesn't have to be.
Act-as-User, though, uses OpenID Connect with an authentication server to do the actual authentication, and then the Proton task is told to accept those tokens as legal for acting on behalf of a given user. While you could in theory write your own OIDC server that dispenses tokens for any user name willy-nilly, in practice you'll almost definitely use an existing implementation. In the default case, that implementation will in turn almost definitely be IAM, which is an OAuth broker service with the AppDev Pack that stores its configuration data in an NSF but and reads users from Domino (or elsewhere) via LDAP.
IAM, though, isn't special in this regard. It's packaged with the ADP, sure, but the way it deals with tokens is entirely standards-based. That means that any compatible implementation can fill this role, and, since I'd heard great things about Keycloak, I figured I'd give that a shot. With some gracious assistance from Heiko Voigt, I was able to get this working - I don't want to steal his future thunder by going into too much detail, but honestly the main hurdles for me were just around learning how Keycloak works. Once you have the concepts down, you basically plug in the Keycloak client details in for the IAM ones in the same configuration.
With that set up, you can feed your user token from the web app into your Proton API calls, and then your actions will be running as your user in the same way as if you were authenticated in an on-server XPages or other app. The way this manifests in the Java API is a little weird, but it works well enough: almost all Proton calls have a varargs portion at the end of their method signature that takes OptionalArg
instances. One such type is OptionalAccessToken
, which takes your auth token as a String. I have a method that will stitch in an access token when present. That gets passed in when making calls, such as to read documents:
1 2 3 4 5 6 7 8 9 10 11 12 | OptionalItemNames itemNamesArg = new OptionalItemNames(itemNames); OptionalStart startArg = new OptionalStart((int)skip); OptionalCount countArg = new OptionalCount(limit < 1 ? Integer.MAX_VALUE : (int)limit); List<Document> docs = database.readDocuments( dql, composeArgs( itemNamesArg, startArg, countArg ) ).get(); |
App Authentication
Okay, so that's what you do when you have a setup and a token, but that leaves the process of the user actually acquiring the token. From the user's perspective, this will generally take the form of doing an "OAuth dance" where, when the user tries to access a protected resource, they're sent over to Keycloak to authenticate, which then sends them back to the app with token in hand.
There are a lot of ways one might accomplish this, varying language-to-language, framework-to-framework, and server-to-server. You will be shocked to learn that I'm using Open Liberty for my app here, and that comes with built-in support for OIDC.
Before I go further, I should put forth a big caveat: I'm really muddling through with this one for the time being. The setup I have only kind of works, and is clearly not the ideal one, but it was enough to make the connection happen. I'm not sure if the right path long-term is to keep using this built-in feature or to switch to either a different built-in option or another library entirely. So... absolutely do not take anything here as advice in the correct way to do this.
Anyway, with that out of the way, you can configure your Liberty server to talk to your Keycloak server (or IAM, probably, but I didn't do that):
1 2 3 4 5 6 7 8 9 10 11 12 13 | <openidConnectClient id="client01" clientId="liberty-tester" clientSecret="some-client-secret" discoveryEndpointUrl="https://some.keycloak.server/auth/realms/master/.well-known/openid-configuration" signatureAlgorithm="RS256" sslRef="httpSsl" accessTokenInLtpaCookie="true" userIdentifier="preferred_username" groupIdentifier="groups"> </openidConnectClient> <ssl id="httpSsl" trustDefaultCerts="true" keyStoreRef="myKeyStore" trustStoreRef="OIDCTrustStore" /> <keyStore id="myKeyStore" password="super-secure-password1" type="PKCS12" location="${server.config.dir}/BasicKeyStore.p12" /> <keyStore id="OIDCTrustStore" password="super-secure-password2" type="PKCS12" location="${server.config.dir}/OIDCTrustStore.p12" /> |
The keyStore
s there contain appropriate certificate chains for the TLS connection to your Keycloak server, while the clientId
and clientSecret
match what you configure/generate on Keyclaok for this new client app.
What got me able to actually use this token for downstream access was the accessTokenInLtpaCookie
property. If you set that, then your HttpServletRequest
objects after the initial one will have an oidc_access_token
property on them containing your token in the format that Proton needs. So that's where the ContextDatabaseSupplier
in the previous post got it:
1 2 3 4 5 6 7 | @Produces public AccessTokenSupplier getAccessToken() { return () -> { HttpServletRequest request = CDI.current().select(HttpServletRequest.class).get(); return (String)request.getAttribute("oidc_access_token"); }; } |
This is also one of the parts that makes me think I'm not quite doing this ideally. It's weird that the token shows up in only requests after the first, though that wouldn't be an impediment in a lot of app types. It's also very unfortunate to have the app use a server-specific property like that.
Fortunately, Jakarta Security 3.0 sprouted official OIDC support, though the build of Open Liberty I was using didn't quite have all the pieces in place for that - reasonable, considering Jakarta EE 10 only officially came out yesterday and this was weeks ago. It looks like that may provide the token in a contextual object, so I'll have to give that a shot once support settles in.
With my setup in place, janky as it may be, I'm able to access a resource (e.g. a JAX-RS endpoint) marked with @RolesAllowed("uma_authorization")
and the server will automatically kick me over to Keycloak and then accept the token when I get back. Then, I can pick that up from the request attributes and use it for Domino data access. Keycloak is getting its user directory from Domino via LDAP in the same way as IAM usually would, but, like IAM with AD, it could be configured to use different user directories. I don't know that I'll want to do that, but it's good to know.
Conclusion
Like the original driver itself, this was mostly an educational exercise for me. I don't currently have any requirements to use the AppDev Pack or OIDC/Keycloak, but I'd wanted to dip my toes in both for a while now, and I'm pleased that I came out successful. I imagine that I'll have an occasion to implement something like this eventually. It may not be the same specific parts, but the core concepts are common, like in Keep's JWT and OAuth support. It's a neat setup, and it's definitely worth doing something similar if you have some experimentation time on your hands.