<html><head><meta http-equiv="Content-Type" content="text/html charset=utf-8"></head><body style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class="">We’re using Keycloak on production, stage/QA, development environments and every developer’s workstation / laptop.<div class=""><br class=""></div><div class="">While there will always be differing options on how to successfully do change management, we’ve found a very effective method for handling Keycloak provisioning in all environments so that developers don’t need to mess around with. We’re a continuous integration / deployment shop using micro services and everything has to “just work” … I’ll give an overview of our process here but please keep in mind a few things:</div><div class=""><br class=""></div><div class="">1. This approach works for us, I’m not saying it’s the best way</div><div class="">2. We do _not_ allow production config changes to be automated due to security implications </div><div class="">3. We're very opinionated in our approach to configuration management and we don’t ever modify 3rd party software databases directly. We always use APIs.</div><div class=""><br class=""></div><div class="">We deploy Keycloak to all environments using Docker images. On developer workstations we use Docker Compose to orchestrate bringing up all services a developer may need, including Keycloak.</div><div class=""><br class=""></div><div class="">We have 4 docker images for Keycloak:</div><div class="">- Keycloak Base</div><div class=""> \- Keycloak HA</div><div class=""> \- Keycloak Dev</div><div class="">- Keycloak config manager*</div><div class=""><br class=""></div><div class="">The base image includes all customizations necessary to bring up a Keycloak instance configured with our modules and themes installed.</div><div class="">The HA instance builds off base and configures Keycloak to run as a cluster node. This is used on stage and prod.</div><div class="">The dev instance builds off base and includes our realm file. On startup, this instance loads our realm configuration if it’s not already loaded.</div><div class=""><br class=""></div><div class=""><div class="">All docker images are built and published by the CI server and Keycloak HA can be deployed to stage and prod after a clean CI build.</div></div><div class=""><br class=""></div><div class="">Developers are free to add clients for testing, do whatever they want, etc. to their running dev instance. If they want to get back to our stock build, they pull the latest Docker image from our private Docker repo and restart it.</div><div class=""><br class=""></div><div class="">Adding clients to stage and prod requires approval and is done by a hand. This is for security reasons. Once a configuration change is detected on stage - say a client is added - our CI server exports the realm from stage, changes the realm keys, and creates a new Keycloak Dev instance with the updated realm file.</div><div class=""><br class=""></div><div class="">*A word about configuration management:</div><div class=""><br class=""></div><div class="">Obviously, the realm file we generate knows the URLs of staging services, not local or development environment URLs. To overcome this we introduced another Docker based service called the Keycloak configuration manger. It runs on development environments and workstations. It’s responsible for discovering running services and updating Keycloak via its admin endpoints to reflect the proper configuration for the given environment.</div><div class=""><br class=""></div><div class="">That’s it. The whole process is automated with the exception of configuration changes to stage and prod which require a security review.</div><div class=""><br class=""></div><div class="">Hope this helps. Let me know if you’d like me to elaborate on anything.</div><div class=""><br class=""></div><div class="">Best,</div><div class="">Scott</div><div class=""><br class=""><div class="">
<div style="color: rgb(0, 0, 0); letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><div style="color: rgb(0, 0, 0); font-family: Helvetica; font-size: 12px; font-style: normal; font-variant: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class="">Scott Rossillo</div><div style="color: rgb(0, 0, 0); font-family: Helvetica; font-size: 12px; font-style: normal; font-variant: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class="">Smartling | Senior Software Engineer</div><div style="color: rgb(0, 0, 0); font-family: Helvetica; font-size: 12px; font-style: normal; font-variant: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><a href="mailto:srossillo@smartling.com" class="">srossillo@smartling.com</a></div><div style="color: rgb(0, 0, 0); font-family: Helvetica; font-size: 12px; font-style: normal; font-variant: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class="">
</div>
</div></div><br class=""><div><blockquote type="cite" class=""><div class="">On May 20, 2016, at 1:46 AM, Stian Thorgersen <<a href="mailto:sthorger@redhat.com" class="">sthorger@redhat.com</a>> wrote:</div><br class="Apple-interchange-newline"><div class=""><div dir="ltr" class="">Firstly, just wanted to highlight that core Keycloak team are devs, not sysadmins/ops guys, so we have limited experience in <span style="font-size:12.8px" class="">continuous delivery and maintenance of real production systems. Hence, we'd love input from the community on this.</span><div class=""><span style="font-size:12.8px" class=""><br class=""></span></div><div class=""><span style="font-size:12.8px" class="">As it stands we don't really have a proper solution. I believe the best you can do at the moment is either using import feature, partial import or admin rest endpoints. Import is not going to work IMO as it requires re-creating the whole realm. Partial import may work, but would work best for new resources rather than modifying existing resources as it does a delete/create operation rather than attempt to modify. With the admin rest endpoints you'd get the best control of what's going on, but obviously that leaves a fair amount of the work.</span></div><div class=""><span style="font-size:12.8px" class=""><br class=""></span></div><div class=""><span style="font-size:12.8px" class="">In the future we have an idea of introducing an "import directory" it would be possible to drop json files in here that would add, modify or delete resources (realms, clients, roles, users, whatever). This would allow dropping json files before the server starts and the server would then import on startup. It would also be possible to do this at runtime and new files would be detected at runtime. Finally, we also had an idea of an offline mode to run import of this (it would basically start the server without http listener, import files, then stop, so it could be used in a script/tool). Import is probably not the best name for it, as it would support modify and delete as well as "importing" new things.</span></div></div><div class="gmail_extra"><br class=""><div class="gmail_quote">On 19 May 2016 at 19:53, Jesse Chahal <span dir="ltr" class=""><<a href="mailto:jessec@stytch.com" target="_blank" class="">jessec@stytch.com</a>></span> wrote:<br class=""><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">Following some of the best practices for continuous Integration and<br class="">
continuous delivery there needs to be environments for build, test,<br class="">
and production. This would mean that following these practices would<br class="">
require you to have multiple versions of keycloak at different stages<br class="">
of development cycle. Some of these environments might not have<br class="">
important persistent data while others might. In order to have builds<br class="">
transition from one environment to another there may be configuration<br class="">
changes required for a build to be valid. This is especially true when<br class="">
new services (openid clients) are being added or "default" accounts.<br class="">
I'm trying to come up with a scripted way of updating keycloak<br class="">
instances that are backed up by an RDMS. This may include adding new<br class="">
clients, adding new users, updating realm config, etc... Originally I<br class="">
was planning on simply exporting the realm config and importing it<br class="">
every time keycloak starts. If I enabled the OVERWRITE option I might<br class="">
overwrite things that I do not want overridden. This is especially<br class="">
true if there is some config that differ's based on whether it is a<br class="">
build, test, or production instance. If I don't enable it then it is<br class="">
only useful for new/blank keycloak environments. I considered using<br class="">
liquibase but since I do not have control of schema changes created by<br class="">
the keycloak team I might run into issues with my liquibase file not<br class="">
being valid after a migration/liquibase update by the keycloak team as<br class="">
my liquibase file would run after keycloak's does. There might also be<br class="">
some other unknown issues our liquibase changes conflicting somehow<br class="">
with keycloak's liquibase changes. I've also considered writing my own<br class="">
updater tool using a scripting language (python/ruby) that calls<br class="">
keycloak's rest api. The issues with this mechanism is it feels like I<br class="">
am recreating the wheel as well as not being able to find good<br class="">
documentation on keycloak's openid endpoints/url's used for different<br class="">
oauth2 flows. Even if I did find this documentation it would also<br class="">
require me to find a good openid client for the scripting language.<br class="">
This doesn't matter for our normal clients as they simply use the<br class="">
keycloak subsystems and adapters instead. I've also looked at commonly<br class="">
used server configuration software such as chef, puppet, and ansible.<br class="">
I don't see a good solution using any of those tools yet either. What<br class="">
have other people done for cases like this? Please don't tell me there<br class="">
is someone who is doing this all manually because that doesn't work in<br class="">
modern software development.<br class="">
<br class="">
- doesn't accidentally delete users<br class="">
- doesn't accidentally delete clients<br class="">
- doesn't invalidate sessions (optional)<br class="">
- works to bring up new, correctly configured, keycloak instances<br class="">
- handles applying updates to existing keycloak instances<br class="">
- can handle minor differences between keycloak instances (build,<br class="">
test, production) when updating<br class="">
- preferably can work well in rolling deployment scenario's.<br class="">
-- I hope the keycloak team is taking these into consideration when<br class="">
doing database migration between 1-2 releases. It would be nice if<br class="">
they set some specific rules for rolling updates between versions (aka<br class="">
backwards breaking changes)<br class="">
_______________________________________________<br class="">
keycloak-user mailing list<br class="">
<a href="mailto:keycloak-user@lists.jboss.org" class="">keycloak-user@lists.jboss.org</a><br class="">
<a href="https://lists.jboss.org/mailman/listinfo/keycloak-user" rel="noreferrer" target="_blank" class="">https://lists.jboss.org/mailman/listinfo/keycloak-user</a><br class="">
</blockquote></div><br class=""></div>
_______________________________________________<br class="">keycloak-user mailing list<br class=""><a href="mailto:keycloak-user@lists.jboss.org" class="">keycloak-user@lists.jboss.org</a><br class="">https://lists.jboss.org/mailman/listinfo/keycloak-user</div></blockquote></div><br class=""></div></body></html>