A few weeks ago, Optimizely released their new Headless CMS product: SaaS CMS. With a new Visual Builder for content authors and edge delivery of content with Optimizely Graph, Optimizely's new SaaS offering is fully embracing modern web architecture.
I've written at length about how headless opens the door to leveraging modern web architecture like static generation, edge functions, serverless and more. Whether you target PaaS or SaaS, taking a headless approach will allow you to target these modern practices, make it easier to achieve great performance and scalability over other architectures.
Optimizely SaaS CMS offering also aligns well with MACH principles and would probably be a good option for composable solutions leveraging commerce capabilities as well. It's important to note that Optimizely appears to be positioning this offering as an alternative to their traditional robust PaaS offering. It provides a migration path to PaaS if a client needs more capabilities than are available in SaaS, since customizations will be limited given the SaaS nature of the solution.
.png)
Source: Optimizely
It's important to note that you can build headless applications against either PaaS Core or SaaS core. Both solutions can publish to Optimizely Graph, though the Visual Builder only currently supports SaaS.
My hands-on experience with Optimizely SaaS CMS
I was recently given access to my own Optimizely SaaS CMS environment and spent the past couple of weekends going through sample code and tutorials to better understand the capabilities of the offering. The rest of the blog post documents my impressions, opinions and thoughts. I'll caveat all of this with the fact that I don't have a ton of Optimizely PaaS experience, and most of my viewpoints are shaped through my knowledge of other CMS Platforms like Sitecore, Acquia and Contentful.
Visual Builder
At a glance, the new Visual Builder looks familiar. The navigation and UX are very similar to the traditional page editor Optimizely has for PaaS. But the biggest difference is the editing experience. Visual Builder supports side by side editing of content, as clicking on a page in editing mode will automatically scroll to the form fields associated with the component you clicked.
.png)
This editing experience seems very intuitive. The rest of the interface, navigating pages and images and other shared content should look very familiar and intuitive for content authors already familiar with Optimizely CMS.
.png) 
 
It's also pretty easy to change the look and behavior of components through the styles interface. In the visual builder you can access page and element by clicking on the three dots and selecting style.
.png)
The style options can actually be configured in code (more on this later) and you can even provide a list of variants that actual map to different react components. In the Mosey bank example, the Heading component has both a default style and an animated style that will load a different component to render the heading with an animated effect.
.png)
Content Modeling
Modeling content is also quite different in Optimizely Saas CMS. Optimizely PaaS relied heavily on .Net Attributes and custom properties in .Net classes to define page and component content types. In contrast, the SaaS version includes a content type modeling interface in the "Settings" areas of the Admin interface.
.png)
Creating and editing content types is pretty straightforward. You can differentiate between page types, element (component) types and reusable types (block types) and add all the regular field types to easily define restrictions and display options right in the editor.
Additional style options may be added to support components in Json (more on this later) but defining a robust page driven content model should be achievable by most business users familiar with content modeling concepts.
.png)
I suspect that this content modeling approach is a lot more efficient than defining similar structures through .net attributes for PaaS codebases and is probably a big driver for time savings in development, not to mention the convenience of managing it in a web interface.
Note that if you're modifying your content model, you may need to run a graph full synchronization to ensure your code can consume the updated model. This is easily done from the "Scheduled Jobs" section of settings and only takes about a minute to run.
.png)
Learn how Optimizely empowers content creators in digital transformation.
Sample Projects
It's important to note that the two Github repositories are provided as samples. There is no real SDK for building a new site against the SaaS CMS. Of the two, the ‘hello world’ sample probably isn't the greatest of starting points. It will walk you through the creation of a basic component including modeling a component (element) content type) and probably won't take you more than 30 minutes.
But if you look at the codebase, it's a Next.js application implemented with Pages Router. Even the component example implementation looks like it's not there to show best practices (setting the markup using "dangerously set innerHTML").
.png)
The other sample from Vercel though is a much more robust implementation. It uses the more modern Next.js App Router, includes a small library of components and demonstrates how to properly structure a full project.
That being said, it's important to note that this repository is a demo site implementation for Mosey Bank. It is not a starter kit. There are currently no SDK's or starter kits to use as a starting point for a new project. This means that your best bet today is to start with the Mosey Bank demo codebase and remove the things you don't want or need and build from there.
There are also no other samples outside of Next.js as far as I'm aware. The Mosey Bank demo also leverages Tailwind, with a tailwind specific grid implementation, which should be probably straightforward to modify to support another design framework.
There should also be no issues building with Nuxt, Angular or any other framework, but you may need to spend some time building out the low-level code to render pages, manage rendering the grid, and more. The path of least resistance, at least for now, is Next.js.
Environments
Optimizely Saas CMS provisioned three environments for my team at Altudo. When you set up your local development environment, you actually need to configure the website to use your localhost address.
.png)
Given this setup, I am left to wonder whether each developer would get their own SaaS instance to develop in, or whether developers would share environments. I can imagine them stepping on each other, trying to work in a shared environment, but depending on the cost, not sure how feasible it would be to have every dev have their own environment. Although I can see this may not be an issue for front end developers who probably won't need to modify content models.
Security
I have not been able to add users to my instance but think that's a permissions issue as I don't have access to the admin center for my organization. Outside of that, setting permissions looks relatively straightforward.
.png)
Tracking Changes
All the changes you make to content types and content itself can be exported and imported from one instance to another.
.png)
This gives you flexibility to create a package that contains only the configurations you wish to export. The file format can easily be renamed as a zip file, which can then be opened and inspected. Inside, you will find that the definitions are stored in XML files.
.png)
I'm left considering what the proper way is to source control this data. Whether to track the exported file on a periodic basis or to unzip the file and source control those (at least you'd be able to easily do diffs and roll back). It may be worthwhile to provide another mechanism to serialize this structure for source control purposes.
Solution Structure
The solution follows the typical Next.js app router project structure. It's relatively easy to figure out where things are. I'm not 100% clear on the component folder structure (what goes in the root vs. cms folder vs. component folder, but there are clearly some low-level components under shared for rendering content, images, links and more that should probably be part of a starter kit.
.png)
There is also built-in code generation for GraphQL. While you still need to construct your queries and fragments within your component structure, a watch command is available that will take those files, validate them against the Opti Graph endpoints, and then generate code in the gql folder. To run this watch command, I copied the codegen script from the "package.json" configuration file of the hello world project into the "package.json" of the Vercel project and executed the script from the command line.
.png)
Components
At first glance, the component structure seems a little overwhelming. Looking at just the existing heading component, which has a single text field, there are 8 different files.
.png)
It's not as bad as it looks through. The index is really just there to look at the style settings and decide which variant to use: the normal one, or the animated version.
.png)
The "default_Heading.tsx" and "animated_heading.tsx" are straight forward implementations of the actual components. The "Types.ts" file just includes definitions for the Layoutprops and Component base types used by all the variants. The GraphQL file contains the fragment for the component. Other components had actual queries.
.png)
If your GraphQL watcher is active, every time you modify this file, the details will be validated against your Opti instance. If you've recently changed your content model, you may need to run the scheduled job sooner, and then the code will be generated into the gql folder.
The "opti-type" json file contains basic details about the component and its properties, most of which are part of the content definition itself. There should only be one of these per component.
.png)
Each variant can have its own "style.json" file to define additional rendering parameters which can be used to customize that variant even further. For example, the heading component's default variant gave the option to select which heading tag to use to render the content.
.png)
How this gets wired up in the editor is unclear. In my attempts to create a hero component, I was unable to get this to work properly even though it worked fine for existing components. It would be helpful to have better logging and information about how these files are interpreted, including details on any errors that occur.
My other thought is that most of the information needed for these additional files (including the layout props and GraphQL) is already defined in the Content type. Assuming this meta data can be fetched from the server, there should be an approach that can be built to scaffold out a component and make it easier to just focus on the component functionality instead of the plumbing.
Creating a New Component
I was able to successfully create a new element content type for a new Hero component I wanted to add. I followed the same pattern as the heading component and was able to lay things out and update the textual fields of the component with no issues.
.png)
For image fields, I'm not sure if it's a problem with my instance or my code, but I don't have the ability to select any of the images in my media library. I suspect it's my instance since I get a similar experience trying to add out of the box image components as well.
.png)
The other problem I had was in getting it to recognize my style options for the new component. As I mentioned there's not any real feedback or logs that I'm aware of that tell me what's wrong with it. I also noticed that you could export "styles" from the export settings, but I'm not sure if there's another place I need to define them other than in the code.
What's Next?
Despite a few bumps, I'm actually really impressed with Optimizely SaaS CMS. The Visual Builder interface is fast and easy to use. Creating and configuring content types is straight forward. The fact that it’s SaaS and will improve without the need for an upgrade is another huge plus.
If you’re interested in learning more about it and seeing a demo of SaaS CMS in action, I’ve put together a good overview deck that we can run through in a short call. Write to me at marketing@altudo.co if you’d like to see it.
.png)
I believe headless CMS architecture is the future, and we'll see more and more use of this platform over time. At Altudo we made the jump to headless, Next.js and React driven architectures for clients on other CMS platforms a few years ago. With these approaches we've been able to create websites that have amazing performance, targeting near perfect lighthouse scores. We can do the same for Optimizely SaaS CMS.
We even have headless accelerators that include libraries of readymade components. I'll be working with my team to start figuring out how to support Opti with these same components and get one of our demo site implementations up and running on the platform. If you have a POC need, please reach out as I'd love to leverage a real-world use case for this demo.


.jpg?h=319&iar=0&w=640&hash=E4F291C554275FF48409FBCAA1F4C92B)