In this article we learn about some challenges I faced while working with both Unicorn and Sitecore CLI.
On my project we are at a stage in which a wholesale change from Unicorn to Sitecore CLI is not an option. There is however an advantage of updating the deployment pipeline to leverage the IAR plugin for the Sitecore CLI. Our current setup takes upwards of 30 minutes to complete the deployment. Imagine a folder of 40k Unicorn items and trying to zip that up, cleaning the old ones, extracting on the destination server, and then syncing items after every deployment. Deployment nights are exhausting.
Here is a brief list of issues I uncovered along the way.
- Minor issues in YAML compatibility between Unicorn/Rainbow and Sitecore Rainbow.
- Content of YAML may cause the creation of Items in Sitecore without a language version.
YAML Compatibility
In the following partial example, if this was generated by the Sitecore CLI the database name would be missing which could cause an issue with Unicorn.
--- | |
ID: "9c7e7d60-bfa0-4fec-b55c-462b3efdd545" | |
Parent: "839b77db-6040-4f00-8771-8e96fd37aba2" | |
Template: "854ba861-63ea-4a0c-8c7b-541e9a7ec4c1" | |
Path: /sitecore/system/Settings/Foundation/Scms/Search/Search Query Rules Context/Tags/Default | |
DB: master |
In the following partial example, if this was generated by the Sitecore CLI the Multilist field type would be missing which causes Unicorn to sync the fields without the pipe delimiter.
SharedFields: | |
- ID: "42f77151-098f-496a-94cf-590b7edeeabe" | |
Hint: Tags | |
Type: Multilist | |
Value: | | |
{B5BAE47B-7C89-46AE-A367-B3F5027E8D18} | |
{83E58187-D1E5-4FB8-953E-EE89816EC0B5} | |
{D8933FCB-48F3-468E-8FB6-8F2B5CFAF404} | |
{09340671-13CF-4181-A706-BE150B00D735} | |
{CCDEDBA3-3677-433E-BA09-64A3094808C9} | |
{007842BF-6DD7-464F-AD3E-559DEEA38CDA} |
Recommendation: If you are going to also use the Sitecore CLI to serialize items, take note of what fields go missing and make sure they are added back.
The command used with the Sitecore CLI to generate IAR items would be one of the following:
- Sitecore CLI Module ->
dotnet sitecore itemres create -o _out/scms -i Scms.* --overwrite
- Unicorn Folder ->
dotnet sitecore itemres unicorn -o _out/scms -p "../Serialization" --overwrite
Language Versions
Most installations of Unicorn will exclude several fields such as __Revision. This missing from the serialized items (both in IAR form and YAML) may cause the Sitecore Publishing Service to improperly delete items during publishing. Due to this issue both SXA and SPE now include the revisions in the IAR files.
Another interesting discovery is how few fields you may see listed under each language version. In some files generated by Unicorn I noticed no fields listed under the version. Not sure if this was a mistake during developers merging or something else.
- Language: en | |
Versions: | |
- Version: 1 |
In other cases I found by excluding fields in the sitecore.json as seen in the official docs the YAML serialization and IAR generation will result in fields missing as well. The one time I actually read the manual I end up introducing an issue. (sitecore.json fields removed for brevity)
{ | |
"serialization": { | |
"excludedFields": [ | |
{ | |
"fieldId": "badd9cf9-53e0-4d0c-bcc0-2d784c282f6a", | |
"description": "__Updated by" | |
}, | |
{ | |
"fieldId": "d9cf14b1-fa16-4ba6-9288-e8a174d4d522", | |
"description": "__Updated" | |
}, | |
{ | |
"fieldId": "5dd74568-4d4b-44c1-b513-0af5f4cda34f", | |
"description": "__Created by" | |
}, | |
{ | |
"fieldId": "25bed78c-4957-4165-998a-ca1b52f67497", | |
"description": "__Created" | |
}, | |
{ | |
"fieldId": "{52807595-0F8F-4B20-8D2A-CB71D28C6103}", | |
"description": "__Owner" | |
}, | |
{ | |
"fieldId": "{001DD393-96C5-490B-924A-B0F25CD9EFD8}", | |
"description": "__Lock" | |
} | |
] | |
} | |
} |
With the above configuration all the fields serialized are excluded when interacting with the Sitecore CLI. This certainly becomes an issue for items without other fields (like folders).
Languages: | |
- Language: en | |
Versions: | |
- Version: 1 | |
Fields: | |
- ID: "25bed78c-4957-4165-998a-ca1b52f67497" | |
Hint: __Created | |
Value: 20230111T165326Z | |
- ID: "52807595-0f8f-4b20-8d2a-cb71d28c6103" | |
Hint: __Owner | |
Value: | | |
sitecore\admin | |
- ID: "5dd74568-4d4b-44c1-b513-0af5f4cda34f" | |
Hint: __Created by | |
Value: | | |
sitecore\admin | |
- ID: "8cdc337e-a112-42fb-bbb4-4143751e123f" | |
Hint: __Revision | |
Value: "12a9573b-0656-47f2-9614-b06aa437eb58" | |
- ID: "badd9cf9-53e0-4d0c-bcc0-2d784c282f6a" | |
Hint: __Updated by | |
Value: | | |
sitecore\admin | |
- ID: "d9cf14b1-fa16-4ba6-9288-e8a174d4d522" | |
Hint: __Updated | |
Value: 20230111T165326Z |
No fields equals no language version. No Language version equals a broken SPS. Kittens will cry. Ice cream will melt.
Recommendation: Update Unicorn.config to serialize with the __Revision field and then reserialize everything. An issue was opened with Unicorn to alter the patching behavior but for now you may wish to make the change directly for now by manually commenting out the line in the file here.
Run the following SPE report for finding problematic items. Uncomment the section related to language versions missing.
#https://www.maartenwillebrands.nl/2022/02/15/sitecore-removing-iar-items-from-the-database/ | |
function Get-ModifiedItem { | |
param( | |
$databaseName, | |
$filename | |
) | |
$resourceLoaderType = ([System.Type]::GetType("Sitecore.Data.DataProviders.ReadOnly.Protobuf.IResourceLoader, Sitecore.Data.ResourceItems.ProtobufNet")) | |
$resourceLoader = [Sitecore.DependencyInjection.ServiceLocator]::ServiceProvider.GetService($resourceLoaderType) | |
$paths = [System.Collections.Generic.List[String]]@() | |
$paths.Add([Sitecore.MainUtil]::MapPath("/App_Data/items/$($databaseName)/$($filename)")) > $null | |
$paths.Add([Sitecore.MainUtil]::MapPath("/sitecore modules/items/$($databaseName)/$($filename)")) > $null | |
$defaultFieldValues = New-Object -TypeName 'System.Collections.Generic.Dictionary[[guid], [string]]' | |
$database = [Sitecore.Configuration.Factory]::GetDatabase($databaseName) | |
$connectionString = [System.Configuration.ConfigurationManager]::ConnectionStrings[$databaseName] | |
$callContext = New-Object -TypeName "Sitecore.Data.DataProviders.CallContext, Sitecore.Kernel" -ArgumentList @($database.DataManager, $database.DataProviders.Count) | |
$sqlDataProvider = New-Object -TypeName "Sitecore.Data.SqlServer.SqlServerDataProvider, Sitecore.Kernel" -ArgumentList @($connectionString) | |
$itemDataSet = $resourceLoader.LoadFromFiles($paths, "dat", $defaultFieldValues) | |
foreach($itemRecord in $itemDataSet.Definitions.Values) { | |
<# Find Items in database and IAR file | |
$id = New-Object -TypeName "Sitecore.Data.ID, Sitecore.Kernel" -ArgumentList @($itemRecord.ID) | |
$itemDefinition = $sqlDataProvider.GetItemDefinition($id, $callContext) | |
if($itemDefinition) { | |
Get-Item -Path "$($databaseName):" -ID ([ID]::Parse($itemRecord.ID)) | |
} | |
#> | |
<# Find items where a language version is missing | |
$item = Get-Item -Path "$($databaseName):" -ID ([ID]::Parse($itemRecord.ID)) | |
if($item.Versions.GetVersions($true).Count -eq 0) { | |
$item | |
#Add-ItemVersion -Item $item -TargetLanguage "en" -IfExist Skip -IfNoSourceVersion Add | |
} else { | |
#$item | |
} | |
#> | |
<# Update revision on items which use the Standard Value | |
$id = New-Object -TypeName "Sitecore.Data.ID, Sitecore.Kernel" -ArgumentList @($itemRecord.ID) | |
$itemDefinition = $sqlDataProvider.GetItemDefinition($id, $callContext) | |
if(!$itemDefinition) { | |
$item = Get-Item -Path "$($databaseName):" -ID ([ID]::Parse($itemRecord.ID)) | |
if($item.Fields["__Revision"].ContainsStandardValue) { | |
Write-Host "Saving revision for $($item.Paths.Path)" | |
$item.Editing.BeginEdit() | |
$item["__Updated by"] = "sitecore\admin" | |
$item.Editing.EndEdit($true, $false) > $null | |
$item | |
} | |
} | |
#> | |
} | |
} | |
$iarItemNames = Get-ChildItem -Path "$($AppPath)/sitecore modules/items/" -File -Filter *.dat -Recurse | Select-Object -ExpandProperty Name | Sort-Object | |
$options = [ordered]@{} | |
foreach($iarItemName in $iarItemNames) { | |
$options[$iarItemName] = $iarItemName | |
} | |
$props = @{ | |
Parameters = @( | |
@{Name="filename"; Title="Choose an option"; Tooltip="Additional details about the option"; Options=$options; } | |
) | |
Title = "Option Selector" | |
Icon = "OfficeWhite/32x32/question.png" | |
Description = "Choose an option." | |
Width = 450 | |
Height = 300 | |
ShowHints = $true | |
} | |
$result = Read-Variable @props | |
if($result -eq "cancel"){ | |
exit | |
} | |
$databaseName = $filename.Split(".")[1] | |
Get-ModifiedItem -Database $databaseName -Filename $filename | Show-ListView -ViewName IARCleanup -ActionData @{"databaseName"=$databaseName;"filename"=$filename} -Property ID, Name, ItemPath, TemplateName, Database, Language, Version, @{Label="Revision";Expression={$_.__Revision}} |
No comments:
Post a Comment