Setting up content staging
Some time ago Sitecore offered the staging module, but since 6.3 it isn’t available anymore. Recently, one of our clients asked for such functionality and I thought we could easily provide this by simply adding a publishing target. In general, this proved to be true and the solution works great, but there are some things you should know to create a fully independ content staging environment, in which the indexes play an important role.
Why content staging?
Having a preview function in Sitecore, some people wonder why you would need a separate staging environment. The answer is equally logical as surprising. Because Sitecore let’s you preview the page in the way it would be presented after a publish action, it respects the publish restrictions like date ranges and the non-publishable status of a version or an item. That’s nice, and by design, but we always teach content editors to make new content unpublishable until they want to publish it, to prevent colleague editors from accidentially publishing their unfinished content. Stalemate! You cannot preview your changes unless you’ll make your item publishable, but you do not want to yet, because it is still in preview..
Using a separate content staging environment, we let editors publish their content to a separate website (where it is publishable), without any publish restrictions or the risk of publishing content publicly too early in the process.
The solution
We will create a separate content staging site for content editors in this blog post, by adding a publishing target and a separate set of indexes for this publishing target.
Demarcation
We will only set this up for the production environment, so the test and acceptance environment will not be changed. They keep their single publishing target. Furthermore, having a content management server and multiple delivery servers, this content staging environment is only applicable for the content management server. So that’s our target environment: the content management server within the production environment needs an extra content staging website.
What it takes
1) Duplicate your web database and rename it with a _Staging postfix and add a new connection string to your ConnectionStrings.config file accordingly:
1 2 3 |
<add name="staging" connectionString="user id=sitecore;password=secret;Data Source=(local)\SQLEXPRESS;Database=DemoSitecore_Staging"/> |
2) Add a new database definition to your databases node in the Web.config. The id of this node must be equal to the name of your connection string node:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 |
<!-- Staging --> <database id="staging" singleInstance="true" type="Sitecore.Data.Database, Sitecore.Kernel"> <param desc="name">$(id)</param> <icon>Network/16x16/earth.png</icon> <securityEnabled>true</securityEnabled> <dataProviders hint="list:AddDataProvider"> <dataProvider ref="dataProviders/main" param1="$(id)"> <disableGroup>publishing</disableGroup> <prefetch hint="raw:AddPrefetch"> <sc.include file="/App_Config/Prefetch/Common.config" /> <sc.include file="/App_Config/Prefetch/Webdb.config" /> </prefetch> </dataProvider> </dataProviders> <proxiesEnabled>false</proxiesEnabled> <proxyDataProvider ref="proxyDataProviders/main" param1="$(id)" /> <archives hint="raw:AddArchive"> <archive name="archive" /> <archive name="recyclebin" /> </archives> <Engines.HistoryEngine.Storage> <obj type="Sitecore.Data.$(database).$(database)HistoryStorage, Sitecore.Kernel"> <param connectionStringName="$(id)" /> <EntryLifeTime>30.00:00:00</EntryLifeTime> </obj> </Engines.HistoryEngine.Storage> <cacheSizes hint="setting"> <data>20MB</data> <items>125MB</items> <paths>5MB</paths> <itempaths>100MB</itempaths> <standardValues>4MB</standardValues> </cacheSizes> </database> |
3) Add a publishing target item in your “system/Publishing targets” folder in the master database. You can point towards the actual database via the same id or name as mentioned in the previous steps:
4) Add an alternative site node to access the staging environment, be sure to change the database property. If you have more sub domains or websites you want to stage, you can include new entries for those sites as well:
1 2 3 |
<site name="website_demo_staging" hostName="www.staging.demo.local" targetHostName="www.staging.demo.local" virtualFolder="/" physicalFolder="/" rootPath="/sitecore/content" startItem="Demo/Homepage" database="staging" domain="extranet" allowDebug="true" cacheHtml="false" htmlCacheSize="10MB" registryCacheSize="0" viewStateCacheSize="0" xslCacheSize="5MB" filteredItemsCacheSize="2MB" enablePreview="true" enableWebEdit="true" enableDebugger="true" disableClientData="false" language="nl-NL" /> |
Of course, you can configure your HTML cache, HTML and data cache sizes the way you’re used to or like best; these examples only show what’s needed for the creation of the staging environment and use the basic Sitecore configuration as a starting point.
So, this basically creates the new publishing target and adds it to your Sitecore client in the following places. A new database:
And a new publishing target in all publishing dialogs:
But more importantly, when you click on “Change” in the Publish ribbon at the Restrictions chunk, you will see a list of checkboxes for each publishing target on the third tab:
Initially, those checkboxes are unchecked, meaning that the item will be published to all publishing targets by default. By selecting this publishing target, you will prevent other editors from accidentally publishing the item to the world wide web, without setting publish restrictions that prevent you from previewing the item.
If you now publish an item, the message you’ll see asks if you want to publish to “every publishing target”, but since this publishing action respects the publish restrictions, this actually means “to all allowed publishing targets”. Despite this message, the item will not be published to the Internet (web database):
You can now visit the staging website on your content delivery server and test your new content without any restrictions!
Indexability
But we have one more challenge, as I explained in the introduction of this article: indexes. Our site uses indexes for overviews of data sources (data types, like news or products), it uses indexes for populating dropdowns and for building up the FAQ section for example. So we need to separate them too. It might not be the case for your implementation, at least to this extent, but indexes sure are an integral part of your website and I think this solution comes in handy for almost all site implementations.
5) The following step consists of adding a configuration file for the new staging index:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 |
<configuration xmlns:patch="http://www.sitecore.net/xmlconfig/"> <sitecore> <contentSearch> <configuration type="Sitecore.ContentSearch.ContentSearchConfiguration, Sitecore.ContentSearch"> <indexes hint="list:AddIndex"> <index id="sitecore_staging_index" type="Sitecore.ContentSearch.LuceneProvider.SwitchOnRebuildLuceneIndex, Sitecore.ContentSearch.LuceneProvider"> <param desc="name">$(id)</param> <param desc="folder">$(id)</param> <!-- This initializes index property store. Id has to be set to the index id --> <param desc="propertyStore" ref="contentSearch/databasePropertyStore" param1="$(id)" /> <configuration ref="contentSearch/indexConfigurations/defaultLuceneIndexConfiguration" /> <strategies hint="list:AddStrategy"> <!-- NOTE: order of these is controls the execution order --> <strategy ref="contentSearch/indexUpdateStrategies/onPublishEndAsyncStaging" /> </strategies> <commitPolicyExecutor type="Sitecore.ContentSearch.CommitPolicyExecutor, Sitecore.ContentSearch"> <policies hint="list:AddCommitPolicy"> <policy type="Sitecore.ContentSearch.TimeIntervalCommitPolicy, Sitecore.ContentSearch" /> </policies> </commitPolicyExecutor> <locations hint="list:AddCrawler"> <crawler type="Sitecore.ContentSearch.SitecoreItemCrawler, Sitecore.ContentSearch"> <Database>staging</Database> <Root>/sitecore</Root> </crawler> </locations> </index> </indexes> </configuration> </contentSearch> </sitecore> </configuration> |
Let’s call this new file Sitecore.ContentSearch.Lucene.Index.Staging.config to accompany the already existing Core, Master and Web configs for Lucene.
6) … but as you can see, we’ve used a new publishing strategy in the above configuration, so this wouldn’t work without adding the following section to the Sitecore.ContentSearch.Lucene.DefaultIndexConfiguration.config as a child of the indexUpdateStrategies node:
1 2 3 4 5 6 7 8 |
<onPublishEndAsyncStaging type="Sitecore.ContentSearch.Maintenance.Strategies.OnPublishEndAsynchronousStrategy, Sitecore.ContentSearch"> <param desc="database">staging</param> <!-- Whether or not a full index rebuild should be triggered when the number of items in the EventQueue exceeds the number specified in Config.FullRebuildItemCountThreshold. --> <CheckForThreshold>true</CheckForThreshold> </onPublishEndAsyncStaging> |
You have to define a new strategy for your staging database, because the onPublishEndAsync strategy is only watching the event queue from a single database, being the web database by default. So you have to copy that strategy definition and define your own strategy that points towards the new staging database. If you would have multiple publishing targets or databases besides the default web database, mind that you should add a strategy for each of them.
7) Add a SearchContext helper class to retrieve the index that belongs to the current context programmatically. I quickly threw together a class and property that return the currently applicable index, which comes in handy when you query the index a lot from your code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 |
public static class SearchContext { public static string GetSearchIndex { get { var index = "sitecore_web_index"; // default if (Sitecore.Context.Database != null && !string.IsNullOrEmpty(Sitecore.Context.Database.Name)) { switch (Sitecore.Context.Database.Name.ToLowerInvariant()) { case "master": index = "sitecore_master_index"; break; case "staging": index = "sitecore_staging_index"; break; } } return index; } } } |
You can now easily get to the index that belongs to your current context by typing:
1 2 3 |
ContentSearchManager.GetIndex(SearchContext.GetSearchIndex); |
Targeting the content management server
8) As mentioned before, this staging environment shouldn’t be available on your delivery servers, nor should it be active on the test and acceptance environment (in our case). Since we are always using multiple specifically targeted builds in Visual Studio, we can add this line to build configurations of environments that do not need a staging version of the website:
1 2 3 |
<ExcludeFilesFromDeployment>App_Config\Include\Sitecore.ContentSearch.Lucene.Index.Staging.config</ExcludeFilesFromDeployment> |
Tip: we do not use the “Debug” and “Release” builds that VS defaults to, but we have a build type for each target enviroment, like “Development”, “Test”, “Acceptance”, “Production Management” for the content management server(s) and “Production Delivery” for all delivery servers. This way you cannot only maintain multiple Web.config (and other config) translations to alter your config files per target server (like the different URL’s in the sites section), but you can also keep all those deltas within your version control system of choice and you can also add specific project configurations for each environment.
You can now remove the new site node within the Web.config translations of the servers that do not need the staging environment:
1 2 3 |
<site name="website_demo_staging" xdt:Transform="Remove" xdt:Locator="Match(name)"/> |
This, of course, also applies to your connection string configuration.
Conclusion
As you can see, Sitecore is really flexible in setting up different content previewing strategies. A separate staging website, only available on your content management server, is easy to configure without having to setup a whole new server. This extensive blog post (it turned out a bit longer than I intended to..) should help you getting up and running pretty fast, but if you run into anything that isn’t complete, or compatible with your process, I would like to hear from you and I’ll see if I can improve my article.
Comments
Comments are disabled for this post