-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.xml
1821 lines (1355 loc) · 141 KB
/
index.xml
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
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
<channel>
<title>Ian's Blog on Ian's Blog</title>
<link>https://bigredmachine.github.io/</link>
<description>Recent content in Ian's Blog on Ian's Blog</description>
<generator>Hugo -- gohugo.io</generator>
<language>en-us</language>
<lastBuildDate>Sun, 14 Feb 2021 00:00:00 +0000</lastBuildDate>
<atom:link href="/" rel="self" type="application/rss+xml" />
<item>
<title>Session state and Xdb tracking no longer required for rendering personalisation</title>
<link>https://bigredmachine.github.io/post/session-state-and-xdb-tracking-no-longer-required-for-rendering-personalisation/</link>
<pubDate>Sun, 14 Feb 2021 00:00:00 +0000</pubDate>
<guid>https://bigredmachine.github.io/post/session-state-and-xdb-tracking-no-longer-required-for-rendering-personalisation/</guid>
<description><h2 id="happy-valentines-day">Happy Valentine&rsquo;s day</h2>
<p> </p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/session-state-and-xdb-tracking-no-longer-required-for-rendering-personalisation/images/iheart_hua7cb347cab19114bde7edf1f069cf370_21611_495x0_resize_q100_box.jpg">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/session-state-and-xdb-tracking-no-longer-required-for-rendering-personalisation/images/iheart_hua7cb347cab19114bde7edf1f069cf370_21611_465x0_resize_q100_box.jpg">
<img
src="https://bigredmachine.github.io/post/session-state-and-xdb-tracking-no-longer-required-for-rendering-personalisation/images/iheart_hua7cb347cab19114bde7edf1f069cf370_21611_300x0_resize_q100_box.jpg"
alt="I heart Sitecore Support">
</picture>
<p> </p>
<h3 id="background">Background</h3>
<p>So for quite sometime in order to be able to personalise a rendering on Content Delivery you&rsquo;ve had to turn on Xdb tracking, even if you don&rsquo;t want to use Xdb. <br />
And by turning on Xdb Tracking, you have to turn on ASP.NET Session State, or it blows up.</p>
<ul>
<li>&ldquo;Xdb.Tracking.Enabled&rdquo; = true</li>
<li>&ldquo;Xdb.Endabled&rdquo; = false</li>
<li>ASP.NET Seession State - Enabled</li>
</ul>
<h3 id="personalisation-without-xdb">Personalisation without Xdb?</h3>
<p>What can you personalise with if you aren&rsquo;t using Xdb?
Well certainly you are going to be limited what you can personalise on, but you can still personalise on</p>
<ul>
<li>Request Url</li>
<li>Request Headers</li>
<li>Request Cookies</li>
<li>Geo Location of Request</li>
</ul>
<p>Think anything in the current request, but not their History like you would if had an Xdb profile, or current browsing session for previous pages visited available with session state.</p>
<h3 id="whats-the-problem-with-turning-on-session-state-and-xdb-tracking">What&rsquo;s the problem with turning on Session State and Xdb Tracking</h3>
<p>Well other than having to enable features that aren&rsquo;t required, and it not being tidy.</p>
<p>Both of these two features cause cookies to be issued.</p>
<ul>
<li>SC_ANALYTCS_GLOBAL_COOKIE</li>
<li>ASP.NET_SessionId</li>
</ul>
<p>If you want to cache the response in a CDN then you can&rsquo;t emit these request specific cookies at origin, the CDN won&rsquo;t cache these responses. (Maybe with some custom logic to strip them?)</p>
<p>And trying to remove these cookies from being sent before the ASP.NET pipeline has flushed the response headers (after which can&rsquo;t remove the headers as the client has already been sent the headers), can be quite tricky,
especially for the ASP.NET Session State cookie which gets added to the response when the ASP.NET Session State object is accessed if it doesn&rsquo;t already exist.</p>
<p>So trying to find the sweet spot in the ASP.NET pipeline when Sitecore has finished accessing the Session State object, and before ASP.NET has flushed the response headers, can be quite tricky.
Or actually impossible to guarantee 100% of the time.</p>
<h3 id="solution---hotfix-for-sitecore-93">Solution - Hotfix for Sitecore 9.3</h3>
<p>Sitecore Hotfix 448700 for Sitecore 9.3 fixes this.</p>
<p>As I understand it this issue might have been fixed in Sitecore 10 possibly, and has been backported, based on conversations, but I&rsquo;ve not tested if this is fixed in Sitecore 10.</p>
<p>With this hotfix, gone is the need for Session State on Content Delivery and Xdb.Tracking.Enabled. <br />
Gone is the need for the work arounds to remove these cookies from the response before they are sent. <br />
And can still have renderings personalised based on the current request.</p>
<p> </p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/session-state-and-xdb-tracking-no-longer-required-for-rendering-personalisation/images/cake_hueaac250c617d8a711f7ccc022cea4f61_50947_642x0_resize_q100_box.jpg">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/session-state-and-xdb-tracking-no-longer-required-for-rendering-personalisation/images/cake_hueaac250c617d8a711f7ccc022cea4f61_50947_465x0_resize_q100_box.jpg">
<img
src="https://bigredmachine.github.io/post/session-state-and-xdb-tracking-no-longer-required-for-rendering-personalisation/images/cake_hueaac250c617d8a711f7ccc022cea4f61_50947_300x0_resize_q100_box.jpg"
alt="Can have cake and eat it">
</picture>
<p> </p>
</description>
</item>
<item>
<title>Sitecore Log4Net Lockup - Hotfix</title>
<link>https://bigredmachine.github.io/post/sitecore-log4net-lock-up/</link>
<pubDate>Sun, 31 Jan 2021 00:00:00 +0000</pubDate>
<guid>https://bigredmachine.github.io/post/sitecore-log4net-lock-up/</guid>
<description><h2 id="sitecore-log4net-lockup---hotfix--dont-share-appenders">Sitecore Log4Net Lockup - Hotfix &amp; Don&rsquo;t Share Appenders</h2>
<p>This post is about a very rare elusive bug that caused the affected Sitecore instance to freeze up, as well not log anything which gave a clue what the problem was.</p>
<p>It was a combination of a bug in the Sitecore Log4Net code and additional Log4Net configuration that had been added (not OOTB) that caused Sitecore to lock up.</p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/padlock_hud18f49ec515f36941c728dfeefc35a24_662916_650x0_resize_q100_box.jpg">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/padlock_hud18f49ec515f36941c728dfeefc35a24_662916_465x0_resize_q100_box.jpg">
<img
src="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/padlock_hud18f49ec515f36941c728dfeefc35a24_662916_300x0_resize_q100_box.jpg"
alt="Lock">
</picture>
<p>If you haven&rsquo;t seen this issue, where a Sitecore instance locks up and you needed to restart IIS/Server, you probably won&rsquo;t need this hotfix, but hopefully will get this hotfix rolled into a future version of Sitecore, so you&rsquo;ll get the benefit then of hopefully never seeing this bug. <br />
The Log4Net configuration recommendations are worth following though should you add any bespoke configuration.</p>
<p>As I mentioned it was a rare bug, and could go months without seeing the issue surface.</p>
<p>The affected Sitecore instance performed publishing and indexing operations, so was quite busy with lots of threads. <br />
This issue has been seen in multiple Sitecore versions, up to at least Sitecore 9.3.</p>
<p>To get to the bottom of this elusive bug required a memory dump of the affected server. Sometimes restoring service is more important so not always possible to get a memory dump.</p>
<p>Even once that was done, required diagnostics to be added to work out what was causing the lock up/what was null, and more waiting for the issue to re-occur.</p>
<h3 id="what-was-locking-upwhat-was-null">What was locking up/What was null?</h3>
<p>Looking in the Sitecore.Logging.dll <br /></p>
<p>And the &ldquo;PatternParser&rdquo; class, you can see if you pass in any of %C, %F, %L, %M, %l in your log4net conversionPattern configuration, it will call either the &ldquo;ClassNamePatternConverter&rdquo; or the &ldquo;LocationPatternConverter&rdquo;:</p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/PatternParser_hu8aac3e27f9f65af552070b966547a495_37309_650x0_resize_q100_box_2.PNG">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/PatternParser_hu8aac3e27f9f65af552070b966547a495_37309_465x0_resize_q100_box_2.PNG">
<img
src="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/PatternParser_hu8aac3e27f9f65af552070b966547a495_37309_300x0_resize_q100_box_2.PNG"
alt="Pattern Parser">
</picture>
<p>Both of these then make calls to get the Location information, which constructs a new &ldquo;LocationInfo&rdquo; instance.</p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/LocationInformation_huc911b9f510ab5985018c99d64b4502eb_14562_650x0_resize_q100_box_2.PNG">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/LocationInformation_huc911b9f510ab5985018c99d64b4502eb_14562_465x0_resize_q100_box_2.PNG">
<img
src="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/LocationInformation_huc911b9f510ab5985018c99d64b4502eb_14562_300x0_resize_q100_box_2.PNG"
alt="Location Information">
</picture>
<p>The constructor of the &ldquo;LocationInfo&rdquo; loops through the stack trace.</p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/DeclaringType_hu5629a8b18b739bf4b270958ed048c863_52098_650x0_resize_q100_box_2.PNG">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/DeclaringType_hu5629a8b18b739bf4b270958ed048c863_52098_465x0_resize_q100_box_2.PNG">
<img
src="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/DeclaringType_hu5629a8b18b739bf4b270958ed048c863_52098_300x0_resize_q100_box_2.PNG"
alt="Declaring Type">
</picture>
<p>This code assumes that the DeclaringType is always going to be available, however it is possible that the stackFrame.GetMethod().DeclaringType might be null for lambda, which it was in our cases.</p>
<h3 id="possible-workaround">Possible workaround</h3>
<p>One possible workaround before getting a hotfix would have been to not pass in these patterns which were causing the issues. However having the class name and location info of where an issue occurred is useful information.</p>
<h3 id="dont-share-appenders">Don&rsquo;t share appenders</h3>
<p>So Sitecore/Log4Net was failing to log when the stack trace had a Lambda which had a null DeclaringType. <br />
Why was this causing Sitecore to hang? <br />
Shouldn&rsquo;t logging fail safe?</p>
<p>Well when a logger fails, it tries to log the issue to the root logger.</p>
<p>Our root logger was setup to also log to the same appender instance, also with the same ConversionPattern.</p>
<p>Due to the shared appender between loggers it was possible for deadlock to occur.</p>
<p>Setup/Key Facts</p>
<ol>
<li>RootLogger and PublishingLogger use the same appender instance</li>
<li>DoAppend locks the appender which invoked it</li>
<li>CallAppenders locks the logger which invoked it.</li>
</ol>
<p>Scenario</p>
<ol>
<li>Publishing Logger call CallAppenders and DoAppend =&gt; gets the lock of Appender instance and PublishingLoggerObject</li>
<li>RootLogger call CallAppeners and locks RootLoggerObject</li>
<li>RootLogger calls DoAppender for Appender and tries to get the lock of the Appender Instance. But is already locked by PublshingLogger so has to wait until finishes.</li>
<li>Whilst the PublishingLogger is trying to log, it gets an exception in LocationInfo, and so calls RootLogger to log the exception.</li>
<li>RootLogger calls CallAppender and tries to get the lock of RootLoggerObject but is already locked</li>
</ol>
<p>Deadlock</p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/deadlock_hu45f59038fbd33db2bad2749093e160b8_104660_500x0_resize_q100_box.jpg">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/deadlock_hu45f59038fbd33db2bad2749093e160b8_104660_465x0_resize_q100_box.jpg">
<img
src="https://bigredmachine.github.io/post/sitecore-log4net-lock-up/images/deadlock_hu45f59038fbd33db2bad2749093e160b8_104660_300x0_resize_q100_box.jpg"
alt="Deadlock">
</picture>
<h3 id="azurefallbackappender">AzureFallbackAppender</h3>
<p>Looking at
App_Config\Sitecore\ExperienceContentManagement.Administration\Sitecore.ExperienceContentManagement.Administration.config</p>
<p>The AzureFallbackAppender is setup the same, for all the loggers to share the same appender.</p>
<p>&ldquo;AzureFallbackAppender was designed specifically for Sitcore Support Package generator and it allowed us to collect the log files while generating the package. It was decided that only one AzureFallbackAppender remains for all kinds of logs, because of performance reasons.&rdquo;</p>
<p>So whilst this isn&rsquo;t recommended, the AzureFallbackAppender is a limited specific scenario, and issues haven&rsquo;t been encountered with this.</p>
<h3 id="application-insights-paas-appender">Application Insights PAAS appender</h3>
<p>Looking at the file in the Azure Websites project:
App_Config\Sitecore\Azure\Sitecore.Cloud.ApplicationInsights.config</p>
<p>can see has the patch swaps out the default appends for the ApplicationInsights Log4NetAppender, so can see has a separate appender per logger as recommended.</p>
<p>So following the recommended practice.</p>
<h3 id="summary">Summary</h3>
<ul>
<li>DeclaringType can be null for Lambdas</li>
<li>Check for nulls</li>
<li>Don&rsquo;t share Log4Net appenders between loggers</li>
</ul>
<h3 id="sitecore-hotfix-reference-number">Sitecore hotfix reference number</h3>
<p>To monitor the progress of the related issue, please use reference number [437603]</p>
</description>
</item>
<item>
<title>Sitecore Incremental Publishing Edge Case</title>
<link>https://bigredmachine.github.io/post/incremental-publishing-edge-case/</link>
<pubDate>Thu, 13 Aug 2020 00:00:00 +0000</pubDate>
<guid>https://bigredmachine.github.io/post/incremental-publishing-edge-case/</guid>
<description><h2 id="what-kind-of-publishing-is-this-post-about">What kind of publishing is this post about?</h2>
<p>This post is about Incremental Publishing using the out of the box publishing mechanism, not the new Sitecore Publishing Service. There <a href="https://mikael.com/2019/07/learnings-from-a-year-of-implementing-sitecore-publishing-service/">maybe reasons why you can&rsquo;t use SPS just yet</a> for your project.</p>
<p>Again if you are using a Smart or Full site publish this article won&rsquo;t apply.
But it&rsquo;s <a href="https://blog.rauljimenez.co.uk/sitecore-publishing-tips/">recommended to setup a scheduled incremental publish</a>.</p>
<p>This post relates to Sitecore 9.3, not reviewed if applicable for Sitecore 10 yet.</p>
<p>Great, so I know this is about Sitecore OOTB Incremental Publishing, and why it&rsquo;s recommended to use this if SPS isn&rsquo;t a good fit, tell me more about this edge case.</p>
<h2 id="edge-case">Edge Case</h2>
<p>Disclaimer, this is an Incremental Publishing Edge Case issue where an item which should be published never gets published, most of the time you probably wouldn&rsquo;t see this issue occur, and if you did spot something wasn&rsquo;t published as it should, you&rsquo;d just smart publish/republish the unpublished items &amp; possibly their children to fix the issue.</p>
<p>You might possibly be thinking this could be a user error, something might not have been pushed through workflow, or not created until very recently (Always good to explore for technical issues before jumping to conclusions).</p>
<p>However if you have ruled out any user issues, and reviewed the logs and confirmed that the item should have been picked up in an Incremental Publish, you need to start looking for the Ghost in the Machine.</p>
<h2 id="ghost-in-the-machine">Ghost in the Machine</h2>
<p><img src="https://bigredmachine.github.io/post/incremental-publishing-edge-case/images/ghost.gif" alt="Ghost"></p>
<p>Working closely with Sitecore Support and with enough diagnostics data from the rare occurrences when this issue would rear it&rsquo;s head, it was spotted that these items which wouldn&rsquo;t get picked up by any incremental publish were saved around the time of the scheduled publishes.</p>
<p>Let&rsquo;s explore what&rsquo;s going on, and why an item saved very near an incremental publish operation on a rare occurrence might not get picked up.</p>
<h2 id="sql-column-specifications--rounding">Sql Column Specifications &amp; Rounding</h2>
<p>The &ldquo;Properties&rdquo; table stores the Last Publish date as a &ldquo;String&rdquo; including the ticks.</p>
<p>The &ldquo;PublishQueue&rdquo; table &ldquo;Date&rdquo; column stores the date in a <a href="https://docs.microsoft.com/en-us/sql/t-sql/data-types/datetime-transact-sql?view=sql-server-ver15#rounding-of-datetime-fractional-second-precision">Sql &ldquo;Datetime&rdquo; column which isn&rsquo;t as precise</a> and results in rounding.</p>
<p>As does the &ldquo;Item&rdquo; table &ldquo;Created&rdquo; and &ldquo;Updated&rdquo; columns which both use the same Sql &ldquo;datetime&rdquo; column, with the same rounding issue.</p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/incremental-publishing-edge-case/images/DatetimeRounding_hudedd1ac0b6b650abbd8018f2506def47_25522_650x0_resize_q100_box_2.PNG">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/incremental-publishing-edge-case/images/DatetimeRounding_hudedd1ac0b6b650abbd8018f2506def47_25522_465x0_resize_q100_box_2.PNG">
<img
src="https://bigredmachine.github.io/post/incremental-publishing-edge-case/images/DatetimeRounding_hudedd1ac0b6b650abbd8018f2506def47_25522_300x0_resize_q100_box_2.PNG"
alt="Datetime Rounding">
</picture>
<p>Due to these differences, say a publishing job was started with value From 14:00:00.017545Z, it would be rounded to 14:00:00.003ms, and would miss an item added at 14:00:00.000ms.</p>
<p>These millisecond round issues could cause an issue with an item not getting published, however that&rsquo;s very small window for the issue to occur.</p>
<h2 id="what-happens-when-you-save-an-item">What happens when you save an item?</h2>
<p>When you save an item, the items statistics get updated with the timestamp from C#/.NET in memory before getting persisted to the Database.</p>
<p>So the latency to get the record with the timestamp into the database could certainly be an issue/factor.</p>
<p>The datetime value used defaults to not storing the ticks/milliseconds on the item, and just defaults to <code>datetime.ToString(&quot;yyyyMMddTHHmmss&quot;)</code></p>
<p>Truncating any milliseconds or ticks.
(See <code>ItemStatistics.UpdateRevision</code>, <code>DateUtil.IsoNow</code>, <code>DateUtil.ToIsoDate</code>)</p>
<p>So that could certainly be an issue, if saving an item towards the end of the Second (with a high millisecond value).</p>
<p>On save items are added to the <code>publishqueue</code>, with the date field values Updated, PublishDate, UnpublishDate, ValidFrom, ValidTo.
(See <code>DefaultPublishManager.DataEngine_SavedItem</code>, <code>DefaultPublishManager.AddToPublishQueue</code>, <code>DefaultPublishManager.GetActionDateFields</code>)</p>
<p>So that truncated second value gets copied across to the <code>publishqueue</code> table.</p>
<h2 id="example-of-an-edge-case">Example of an Edge Case</h2>
<p>If I save an item at <code>2020-08-13T18:00:00.9881779+00:00</code></p>
<p>It will get saved as being created/modified at <code>2020-08-13T18:00:00.000</code>,
and added to the publish queue</p>
<p>And I have an incremental publish kick off at 18:00, and picks up from the last publish 17:00:00.103-18:00:00.103 as a slight delay in starting.
This doesn&rsquo;t pick up the item which hasn&rsquo;t been created yet at 18:00:00.9881779.</p>
<p>And the future publish operation won&rsquo;t pick it up either as will look for items between 18:00:00.103-19:00:00.103, when it&rsquo;s value will be saved as 18:00:00.000ms.</p>
<h2 id="work-around">Work around</h2>
<p>A work around would be to fix the query to include records affected by this precision/latency issue, at the expense of reprocessing some items more than once.</p>
<p>To do this can customise <code>GetPublishQueueEntries</code> method on the <code>SqlDataProvider</code></p>
<div class="highlight"><pre style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4"><code class="language-csharp" data-lang="csharp"><span style="color:#66d9ef">public</span> <span style="color:#66d9ef">override</span> List&lt;PublishQueueEntry&gt;GetPublishQueueEntries(DateTime <span style="color:#66d9ef">from</span>, DateTime to, CallContext context)
{
<span style="color:#66d9ef">from</span> = <span style="color:#66d9ef">from</span>.AddSecond(-<span style="color:#ae81ff">1</span>);
<span style="color:#66d9ef">return</span> <span style="color:#66d9ef">base</span>.GetPublishQueueEntries(<span style="color:#66d9ef">from</span>, to, context);
}
</code></pre></div><div class="highlight"><pre style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4"><code class="language-xml" data-lang="xml"><span style="color:#f92672">&lt;configuration</span> <span style="color:#a6e22e">xmlns:patch=</span><span style="color:#e6db74">&#34;http://www.sitecore.net/xmlconfig/&#34;</span> <span style="color:#a6e22e">xmlns:set=</span><span style="color:#e6db74">&#34;http://www.sitecore.net/xmlconfig/set/&#34;</span><span style="color:#f92672">&gt;</span>
<span style="color:#f92672">&lt;sitecore&gt;</span>
...
<span style="color:#f92672">&lt;dataProviders&gt;</span>
<span style="color:#f92672">&lt;main&gt;</span>
<span style="color:#f92672">&lt;patch:attribute</span> <span style="color:#a6e22e">name=</span><span style="color:#e6db74">&#34;type&#34;</span> <span style="color:#a6e22e">value=</span><span style="color:#e6db74">&#34;YourNamespace.CustomSqlServerDataProvider, YourAssembly&#34;</span> <span style="color:#f92672">/&gt;</span>
<span style="color:#f92672">&lt;/main&gt;</span>
<span style="color:#f92672">&lt;/dataProviders&gt;</span>
...
<span style="color:#f92672">&lt;/sitecore&gt;</span>
<span style="color:#f92672">&lt;/configuration&gt;</span>
</code></pre></div><h2 id="worst-case">Worst Case</h2>
<p>Item statistics get updated at 17:59:59.999ms in C#</p>
<p>Item gets saved as 17:59:59.000ms but some latency before added to the database so doesn&rsquo;t get added until after 18:00:00.103ms
Item gets added to the publish queue with value 17:59:59.000ms</p>
<p>Publish Window by default queries for
<code>18:00:00.103-19:00:00.103</code></p>
<p>with the 1 second fix would query for
<code>17:59:59.103-19:00:00.103</code></p>
<p>Both of these would still miss the item at 17:59:59.000ms.</p>
<h2 id="improved-work-around">Improved Work Around</h2>
<p>To round the start time query down to 0 milliseconds, and 0 ticks, can add the following line <a href="https://stackoverflow.com/questions/1004698/how-to-truncate-milliseconds-off-of-a-net-datetime">Credit Stack Overflow</a>:</p>
<div class="highlight"><pre style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4"><code class="language-csharp" data-lang="csharp"><span style="color:#66d9ef">public</span> <span style="color:#66d9ef">override</span> List&lt;PublishQueueEntry&gt;GetPublishQueueEntries(DateTime <span style="color:#66d9ef">from</span>, DateTime to, CallContext context)
{
<span style="color:#66d9ef">from</span> = <span style="color:#66d9ef">from</span>.AddSecond(-<span style="color:#ae81ff">1</span>);
<span style="color:#66d9ef">from</span> = <span style="color:#66d9ef">from</span>.AddTicks(-(<span style="color:#66d9ef">from</span>.Ticks % TimeSpan.TicksPerSecond));
<span style="color:#66d9ef">return</span> <span style="color:#66d9ef">base</span>.GetPublishQueueEntries(<span style="color:#66d9ef">from</span>, to, context);
}
</code></pre></div><p>So the new query would pick up even this more extreme edge case, with the query
<code>17:59:59.000-19:00:00.103</code>.</p>
<h2 id="summary">Summary</h2>
<p>TL;DR; OOTB if you save an item at the same time an incremental publish kicks off, depending on the exact timing there is an edge case that it may never get published.</p>
<p>With a small work around in code you can ensure items saved around the time of a scheduled publish get picked up by the next incremental publish. At the expense of reprocessing some items/events.
However this workaround isn&rsquo;t in the form of an official hotfix, and would require extensive testing. No guarantees. Only consider this if are experiencing the issues described, and of course speak with Sitecore Support first.</p>
<h2 id="do-i-need-this">Do I need this?</h2>
<p>Well firstly you have to be using incremental publishing and scheduled publishing.
And secondly unless you&rsquo;ve spotted items not getting published, have enough editor activity that keep on encountering this issue on rare occasions, you probably don&rsquo;t need it.</p>
<p>However if you are using incremental and scheduled publishing, and have spotted odd issues with items not getting published, and can&rsquo;t explain them due to user error. This might help explain the Ghost in the machine and a work around. But don&rsquo;t apply this fix if you don&rsquo;t need it.</p>
<p>And if you do implement this fix you&rsquo;ll have to test this customisation, as not an official hotfix, and no guarantees. Just a suggestion that this could help if suffering from this particular issue.
And if was implemented incorrectly could lead big problems, e.g. re-processing of events.
So you&rsquo;ve been warned.</p>
<h2 id="longer-term">Longer term</h2>
<p>Longer term I&rsquo;m sure SPS will further improve, and will be able to be used by more customers. At which point Smart Publishing becomes fast enough, we don&rsquo;t need the old incremental publishing.</p>
<p>If there was to be a fix for incremental publishing, the window could be reduced by calculating and storing a last modified time at Sql Server level with millisecond/timestamp precision, rather than using C# and truncating down to the second level with the latency to insert it into the database.
As well as using the DateTime2 column format rather than DateTime with it&rsquo;s millisecond rounding precision issues.
However, this is quite a schema change for an edge case, so the work around will suffice for those that are affected.</p>
<h2 id="sitecore-bug-tracking-reference">Sitecore Bug Tracking Reference</h2>
<p>Sitecore Bug Tracking Reference #397573.</p>
<h2 id="final-disclaimer">Final Disclaimer</h2>
<p>Don&rsquo;t use this work around unless you are experiencing the issues described, and even then perform your own testing, no guarantees.</p>
<p>These code samples haven&rsquo;t been tested, and just give the gist of the fix required.</p>
<p>Work with Sitecore Support to confirm you are experiencing the issue described, as this customisation has a cost to it, and could have negative implications to your solution even if implemented correctly, and even more negative implication if incorrectly implemented.</p>
</description>
</item>
<item>
<title>Precompiled Razor Views</title>
<link>https://bigredmachine.github.io/post/sitecore-precompiled-razor-views-fix/</link>
<pubDate>Sun, 02 Feb 2020 00:00:00 +0000</pubDate>
<guid>https://bigredmachine.github.io/post/sitecore-precompiled-razor-views-fix/</guid>
<description><h2 id="precompiled-razor-views---sitecore-9---issue-192470-amp-119596">Precompiled Razor Views - Sitecore 9 - Issue 192470 &amp; 119596</h2>
<p>There are already some posts on this topic of using Precompiled Razor Views, so I won&rsquo;t duplicate that content.</p>
<p>Kam&rsquo;s blog - <a href="https://kamsar.net/index.php/2016/09/Precompiled-Views-with-Sitecore-8-2/">Precompiled Views supported by Sitecore since Sitecore 8.2</a></p>
<p>Jeremy&rsquo;s blog - <a href="https://jermdavis.wordpress.com/2018/04/02/an-interesting-side-effect-of-compiled-views/">possible side affect of Precompiled Views</a></p>
<p>Chris&rsquo;s blog - <a href="https://chrisperks.co/2018/03/22/hundreds-of-renderings-your-first-page-load-could-be-sloooow/">Slow compilation when using MVC areas/Helix folder structure</a></p>
<p>Matt&rsquo;s blog - <a href="https://www.mattfletcher.co.uk/2019/01/20/beware-of-view-compilation-with-helix-features/">ensuring your razor view paths are unique</a></p>
<p>And probably others, but these one&rsquo;s stood out to me, and relevant for this post.</p>
<p>Read these articles first, I&rsquo;ll wait.</p>
<h2 id="quick-summary">Quick summary</h2>
<p>So you&rsquo;ve read the above articles,
here&rsquo;s a quick recap:</p>
<ul>
<li>
<p>we know Precompiled Razor Views are supported by Sitecore since Sitecore 8.2.</p>
</li>
<li>
<p>We want to measure/benchmark before and after applying this performance fix, how else would you know it&rsquo;s had the desired effect. And watch out for this Disk IO calls, which seem to be occurring even if have the Sitecore setting disabled &ldquo;UsePhysicalViewsIfNewer&rdquo; which seems counterintuitive.</p>
</li>
<li>
<p>If you are following a MVC areas/Helix style structure with multiple folders for your views, it looks like using precompiled razor view is going to be really beneficial in your case.</p>
</li>
<li>
<p>Ensure you are using unique MVC area/Helix folder names &amp; unique razor file paths across projects to avoid issues, just like you would if they were physical files in one project.</p>
</li>
</ul>
<h2 id="ok---ive-read-the-above-articles-what-more-are-you-writing-about">Ok - I&rsquo;ve read the above articles, what more are you writing about?</h2>
<p>Well first I want to draw attention to &ldquo;UsePhysicalViewsIfNewer&rdquo; setting.</p>
<p>Check in your Sitecore setup what this value is set to.</p>
<p>If for example you have received a Sitecore Support patch/hotfix with a new .cshtml file</p>
<blockquote>
<p>E.g. Issue 214239 for &ldquo;\sitecore\shell\client\Business Components Library\version 2\Layouts\Renderings\ListsAndGrids\Grid.cshtml&rdquo;</p>
</blockquote>
<p>Which includes a config patch which changes &ldquo;UsePhysicalViewsIfNewer&rdquo; to true to load the .cshtml file with the fix, rather than the precompiled view in the assembly.</p>
<p>Those disk IO calls which seem to occur either way (more on that later), are defiantly going to occur if you tell it to.</p>
<blockquote>
<p>This also explains the guidance you might see where the .cshtml file last modified data must be more recent than the .dll file, otherwise it wouldn&rsquo;t be loaded.</p>
</blockquote>
<p>So possibly a Sitecore Support patch to a .cshtml file might undo some of the performance gains we are trying to achieve. As with everything measure.</p>
<p>If you don&rsquo;t have &ldquo;UsePhysicalViewsIfNewer&rdquo; set to true, can skip this next section, and skip to the hotfix, otherwise read on.</p>
<h2 id="i-want-a-support-fix-and-performance-settings">I want a support fix and performance settings</h2>
<blockquote>
<p>Disclaimer, I&rsquo;ve only tried this on a dev machine. And not completed benchmarking/testing</p>
</blockquote>
<p>Maybe if you are only worried about the Sitecore MVC .cshtml hotfix for Content Authoring, and only want the compilation benefit on Content Delivery so could change this setting &ldquo;UsePhysicalViewsIfNewer&rdquo; to true just for Authoring?</p>
<p>However, if you want to have the compilation benefit on Content Authoring as well as the MVC .cshtml fix. One possible way could be to create a new project named &ldquo;ZZZZZ&hellip;&rdquo; with Razor compilation enabled and place the fixed .cshtml file in that project with the matching path of the file you want to override from Sitecore.</p>
<p>And add to the pre-compilation configuration a reference to this new assembly.</p>
<div class="highlight"><pre style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4"><code class="language-xml" data-lang="xml"><span style="color:#f92672">&lt;configuration</span> <span style="color:#960050;background-color:#1e0010">xmlns:patch&#34;http://www.sitecore.net/xmlconfig/&#34;</span><span style="color:#f92672">&gt;</span>
<span style="color:#f92672">&lt;sitecore&gt;</span>
<span style="color:#f92672">&lt;mvc&gt;</span>
<span style="color:#f92672">&lt;precompilation&gt;</span>
<span style="color:#f92672">&lt;assemblies&gt;</span>
<span style="color:#f92672">&lt;assemblyIdentity</span> <span style="color:#a6e22e">name=</span><span style="color:#e6db74">&#34;ZZZZZ......AssemblyName&#34;</span> <span style="color:#f92672">/&gt;</span>
<span style="color:#f92672">&lt;/assemblies&gt;</span>
<span style="color:#f92672">&lt;/precompilation&gt;</span>
<span style="color:#f92672">&lt;/mvc&gt;</span>
<span style="color:#f92672">&lt;/sitecore&gt;</span>
<span style="color:#f92672">&lt;/configuration&gt;</span>
</code></pre></div><h3 id="why-call-the-project-zzzzz">Why call the project ZZZZZ&hellip;.</h3>
<p>We are going to turn the bug from Matt&rsquo;s blog - <a href="https://www.mattfletcher.co.uk/2019/01/20/beware-of-view-compilation-with-helix-features/">ensuring your razor view paths are unique</a>
into a solution.</p>
<blockquote>
<p>&ldquo;The problem becomes apparent when two assemblies have the same generated virtual view path, <strong>the assembly that comes alphabetically last wins out</strong> and overwrites the value corresponding to the virtual path key string in the _views dictionary.&rdquo;</p>
</blockquote>
<p>We are going to deliberately overwrite the Sitecore Pre-compiled view, with the fixed Pre-compiled view, but naming the assembly with the fixed .cshtml file with a name Alphabetically after the original Sitecore Assembly.</p>
<p>If this is a good solution without drawback, perhaps support could do this, to save us the time of doing it ourselves?</p>
<p>Or actually Sitecore could re-issue the original .dll with the fixed Razor view, so don&rsquo;t need to overwrite it.</p>
<p>And now with the new Hotfix only model, they probably would do this.</p>
<p>Still this might be relevant for an existing patch you have, prior to the new Hotfix model.</p>
<h3 id="best-of-both">Best of both</h3>
<p>With this technique, can leave &ldquo;UsePhysicalViewsIfNewer&rdquo; set to false, and override a Sitecore precompiled view.</p>
<p>More testing required to verify this works without drawback.</p>
<h2 id="great-so-ive-definitely-got-usephysicalviewsifnewer-set-to-false-whats-this-hotfix">Great so I&rsquo;ve definitely got &ldquo;UsePhysicalViewsIfNewer&rdquo; set to false, what&rsquo;s this Hotfix</h2>
<blockquote>
<p>Disclaimer, I&rsquo;ve only tried this on a dev machine. And not completed benchmarking/testing</p>
</blockquote>
<p>Going back to Jeremy&rsquo;s blog <a href="https://jermdavis.wordpress.com/2018/04/02/an-interesting-side-effect-of-compiled-views/">possible side affect of Precompiled Views</a>. If look at the comments section, can see I enquired about if this was still an issue in Sitecore 9, and had a little more success from Sitecore Support who confirmed this was a bug in Sitecore and not the RazorGenerator project.</p>
<p>Looking at the contents of Sitecore.Support.192470.dll, can see it removes the first MVC view engine from the list, I asked Sitecore Support why and got this answer:</p>
<blockquote>
<p>The setting UsePhysicalViewsIfNewer works as expected for all precompiled views except the ones that come from a couple of SPEAK dlls (Sitecore.Speak.Web.dll and Sitecore.Speak.Components.Web.dll).
The UsePhysicalViewsIfNewer setting value is used to initialize PrecompiledViewAssembly objects which will be a part of the RazorGenerator.Mvc.CompositePrecompiledMvcEngine. Sitecore initializes an instance of Sitecore.Mvc.SitecoreRazorViewEngine derived from the CompositePrecompiledMvcEngine. And further the instantiated engine is encapsulated in the Sitecore.Mvc.InstrumentendViewEngine type.</p>
</blockquote>
<p>So far so good.</p>
<blockquote>
<p>The problem is that there is an extra PrecompiledMvcEngine instance that comes from SPEAK. This particular engine instance ignores the UsePhysicalViewsIfNewer setting. The engine instance gets placed at the very first position of the ViewEngines.Engines list. And the issue happens when this &ldquo;incorrect&rdquo; engine is taken via a FirstOrDefault LINQ func by Sitecore MVC.
Please also notice that with the current architecture the only possible way to fix this is to remove the engine from the list.</p>
</blockquote>
<p>This would seem to explain the issue Jeremy was having.</p>
<p>However I&rsquo;ve yet to benchmark this solution, to see if solves the disk IO issue, but got a logical explanation from Sitecore Support.</p>
<p>I&rsquo;m not sure if this is still an issue or fixed in later version of Sitecore after Sitecore 9. But a quick scan through the recent version release notes I can&rsquo;t see either of the public reference numbers 192470 or 119596.</p>
<h2 id="summary">Summary</h2>
<blockquote>
<p>Disclaimer, I&rsquo;ve only tried this on a dev machine. And not completed benchmarking/testing</p>
</blockquote>
<p>If you are experiencing issues with Compiled Razor views and disk activity even with &ldquo;UsePhysicalViewsIfNewer&rdquo; set to false, perhaps check if this fix is applicable for the version of Sitecore you are on, and measure if it has the desired impact.</p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/sitecore-precompiled-razor-views-fix/images/measure_everything_hu69f3d958114e641daa238781f031e139_62737_430x0_resize_q100_box.jpg">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/sitecore-precompiled-razor-views-fix/images/measure_everything_hu69f3d958114e641daa238781f031e139_62737_430x0_resize_q100_box.jpg">
<img
src="https://bigredmachine.github.io/post/sitecore-precompiled-razor-views-fix/images/measure_everything_hu69f3d958114e641daa238781f031e139_62737_300x0_resize_q100_box.jpg"
alt="Measure Everything">
</picture>
</description>
</item>
<item>
<title>Sitecore MVP 2020 - Technology</title>
<link>https://bigredmachine.github.io/post/mvp-2020/</link>
<pubDate>Sat, 01 Feb 2020 00:00:00 +0000</pubDate>
<guid>https://bigredmachine.github.io/post/mvp-2020/</guid>
<description><h2 id="sitecore-mvp-2020---technology">Sitecore MVP 2020 - Technology</h2>
<p>I&rsquo;m thankful to receive my 1st Sitecore MVP award in the Technology category for 2020.</p>
<a href="https://mvp.sitecore.com/">
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/mvp-2020/images/Sitecore_MVP_logo_2020_hu639937a7b071c4c109d7139ebd34e829_75255_300x0_resize_q100_box.jpg">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/mvp-2020/images/Sitecore_MVP_logo_2020_hu639937a7b071c4c109d7139ebd34e829_75255_300x0_resize_q100_box.jpg">
<img
src="https://bigredmachine.github.io/post/mvp-2020/images/Sitecore_MVP_logo_2020_hu639937a7b071c4c109d7139ebd34e829_75255_300x0_resize_q100_box.jpg"
alt="MVP">
</picture>
</a>
<a href="https://mvp.sitecore.com/">
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/mvp-2020/images/Sitecore_MVP_Technology_2020_hu546c55212ec4b38f28e19b98886d9195_37580_300x0_resize_q100_box.jpg">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/mvp-2020/images/Sitecore_MVP_Technology_2020_hu546c55212ec4b38f28e19b98886d9195_37580_300x0_resize_q100_box.jpg">
<img
src="https://bigredmachine.github.io/post/mvp-2020/images/Sitecore_MVP_Technology_2020_hu546c55212ec4b38f28e19b98886d9195_37580_300x0_resize_q100_box.jpg"
alt="MVP">
</picture>
</a>
<p>I&rsquo;ve been using Sitecore since Sitecore 6 in 2011.</p>
<p>Keep sharing and trying if didn&rsquo;t make it this year, this is my first year I&rsquo;ve been fortunate enough to be selected for the MVP award after several years of applying. <br />
Although it&rsquo;s not enough to just keep on applying, you can&rsquo;t get the award if you don&rsquo;t apply, and you might get some feedback what you need to improve upon.</p>
<p>A lot of my activity with Sitecore might not be publicly visible, and probably explains why I&rsquo;ve not received this award previously.<br />
Digging out dotpeek, finding the root cause of an issue, and possibly supplying a fix on occasion to Sitecore support for review (of course would always prefer to use an official hotfix, but to prove have found the problem &amp; working solution I might get there first). <br />
Or again using dotpeek finding a way to add in support for a feature that isn&rsquo;t currently supported, and sharing the approach with Sitecore to make it easier to extend in future upgrades, or be integrated into the product in the future.<br />
This might not be visible to the community straight away, but hopefully makes the product better in the long run.</p>
<p>Now with the new single Hotfix model per version of Sitecore, these identified issues fixes are more likely to get into the product quicker, as well as making upgrades easier.</p>
<p>I&rsquo;ve been trying to share what I&rsquo;m up to, either blogging, asking/answering questions on Stack Exchange, or presenting when I can.</p>
<p>I&rsquo;ve also been on a train the trainer course on .net core, Docker, K8s and AKS, and delivered this training.</p>
<p>As well as attended a course on improving my public speaking skills.</p>
<p>All of which has helped me practice my public speaking.</p>
<p>Special mentions to <a href="https://scug.co.uk/">Sitecore User Group UK</a> and <a href="http://sitecore.events/">Sitecore Discussion Club</a> for letting me present.<br />
And <a href="https://twitter.com/steviemcgill">Steve McGill</a> and <a href="https://twitter.com/SitecoreMartin">Martin Miles</a> for running them.</p>
<p>As well as to the Sitecore Support &amp; Product teams for collaborating on tickets, and working on getting the fixes/enhancements into the product.</p>
<p>Plus of course the awesome Sitecore community, whether that&rsquo;s Twitter, Sitecore Slack, Blogs, <a href="https://sitecore.stackexchange.com/">Sitecore StackExchange</a>, <a href="https://scug.co.uk/">SCUG Meetups</a> &amp; <a href="http://sitecore.events/">Sitecore Discussion Club</a>, SUGCON, Symposium etc.</p>
<p>As well as the awesome team I work with, but as this is my personal blog I&rsquo;ll leave it at that.</p>
<p>Thanks for reading my blog, more to follow!</p>
</description>
</item>
<item>
<title>Improving the Siteore Solr Content Search Provider</title>
<link>https://bigredmachine.github.io/post/improving-the-solr-content-search-provider/</link>
<pubDate>Tue, 26 Nov 2019 00:00:00 +0000</pubDate>
<guid>https://bigredmachine.github.io/post/improving-the-solr-content-search-provider/</guid>
<description><h2 id="improving-the-siteore-solr-content-search-provider">Improving the Siteore Solr Content Search Provider</h2>
<p>Back in February this year I gave a presentation at the <a href="https://www.meetup.com/sug-uk/events/258462625/">London Sitecore User Group</a> &ldquo;Improving the Solr Content Search Provider&rdquo;.</p>
<p>My first talk at a Sitecore User group.</p>
<p>Although I regularly attend the Sitecore discussion club, and have presented there on several occasions.</p>
<h2 id="known-bugs">Known bugs</h2>
<p>At the presentation I included a link to a <a href="https://github.com/bigredmachine/sitecore-solr">github repository</a> with a list of known issues and their public reference numbers.</p>
<p>The list of issues will vary between Sitecore versions, I&rsquo;ve included the list of issues I&rsquo;ve come across between Sitecore 8.2 up to Sitecore 9.0 update 2.</p>
<p>Including:</p>
<ul>
<li>Partially rebuilt index going live - 96016 - IIS recycle - switches Alias although indexing job cancelled/incomplete - fixed in Sitecore 9 initial release</li>
<li>OptimizeOnRebuildEnabled setting not used - was in bug fix 96016 - introduced in Sitecore 9 initial release</li>
<li>Patch for IsSolrAliveAgent to update SolrState and process reinitialisation correctly - 163850.171950</li>
<li>Incorrect data being indexed, if have ContentSearch.Indexing.DisableDatabaseCaches setting set to true - 96740.127177.155383</li>
<li>If IndexAllFields=false the IncludedFields are indexed as string values - New bug in Sitecore 9 - 252532</li>
<li><a href="https://bigredmachine.github.io/post/sitecore-9-index-slow-down/">Index rebuild slow down after Sitecore 9 upgrade</a>.</li>
</ul>
<h2 id="common-patterns-for-distributed-computing">Common Patterns for distributed computing</h2>
<p>The github repostiory also included an example implementation of patterns to address some of the <a href="https://en.wikipedia.org/wiki/Fallacies_of_distributed_computing">fallacies of distributed computing</a>.</p>
<ul>
<li>Circuit Breaker Pattern
<ul>
<li>Don&rsquo;t make every request wait for a timeout exception, use a circuit breaker to fail fast, so can handle load under a degraded scenario</li>
</ul>
</li>
<li>Not swallowing errors
<ul>
<li>We want to know the difference between no results, and an error.</li>
<li>The circuit breaker can&rsquo;t work if we don&rsquo;t know what&rsquo;s an error.</li>
<li>If we don&rsquo;t know the difference then we might cache there are no results, when there are results, but just couldn&rsquo;t retrieve the results.</li>
</ul>
</li>
<li>Shorter timeouts for queries
<ul>
<li>Again, we want queries to fail fast, and for the circuit breaker to kick in. But we can allow longer for indexing operations to take place, especially as there is no retry mechanism on the crawling/indexing side.</li>
</ul>
</li>
</ul>
<h2 id="comparison-of-the-other-options-out-of-the-box">Comparison of the other options out of the box</h2>
<ul>
<li>Sitecore Query</li>
<li>Sitecore Fast Query</li>
<li>Links Database</li>
</ul>
<p>Related blog posts:</p>
<ul>
<li><a href="http://intothecore.cassidy.dk/2016/05/sitecore-decennial-series-2-dont.html">http://intothecore.cassidy.dk/2016/05/sitecore-decennial-series-2-dont.html</a></li>
<li><a href="https://intothecloud.blog/2018/10/21/It-s-time-to-put-fast-query-to-rest/">https://intothecloud.blog/2018/10/21/It-s-time-to-put-fast-query-to-rest/</a></li>
<li><a href="https://bigredmachine.github.io/post/what-is-wrong-with-the-link-database/">What&rsquo;s wrong with the links database</a></li>
</ul>
<p>On reasons not to use each of them, at least on Content Delivery, or when querying over lots of items.</p>
<h2 id="a-list-of-other-alternatives">A list of other alternatives</h2>
<ul>
<li>Lucene
<ul>
<li>Per server copy to maintain</li>
<li>Removed in Sitecore 9.3</li>
</ul>
</li>
<li>Elastic Search
<ul>
<li>No Sitecore Search Provider</li>
</ul>
</li>
<li>Coveo
<ul>
<li>Not a total replacement, would need Coveo + another search provider, so would need considering after chosen initial search provider.</li>
<li>Commercial Cost</li>
<li>Supported Sitecore Search Provider</li>
</ul>
</li>
<li>Algolia
<ul>
<li>Hosted search</li>
<li>No official Sitecore Search Provider, but used in community.</li>
<li>Not a total replacement, would need Algolia + another search provider, so would need considering after chosen initial search provider.</li>
</ul>
</li>
<li>Azure Search
<ul>
<li>At the time there wasn&rsquo;t an offical Sitecore Azure Search provider, there now is.</li>
<li>However it has some limitations such as <a href="https://doc.sitecore.com/developers/92/platform-administration-and-architecture/en/sitecore-azure-search-overview.html">An Azure Search index can only contain up to 1000 fields.</a>
Which could be an issue for the master search index.</li>
<li>In this particular benchmark the crawling rate was slower</li>
<li><a href="https://grantkillian.wordpress.com/2017/03/09/azure-search-compared-to-solr-for-sitecore-paas-chapter-1-ingestion/">https://grantkillian.wordpress.com/2017/03/09/azure-search-compared-to-solr-for-sitecore-paas-chapter-1-ingestion/</a></li>
<li>But the querying rate was similar, although I do question the cost/limits on querying Azure Search if you&rsquo;ve got a high traffic site. And I&rsquo;ve read some people have had to put a Redis cache in front, to not go over the query limit in Azure Search.</li>
<li><a href="https://grantkillian.wordpress.com/2017/03/20/azure-search-compared-to-solr-for-sitecore-paas-chapter-2-querying/">https://grantkillian.wordpress.com/2017/03/20/azure-search-compared-to-solr-for-sitecore-paas-chapter-2-querying/</a></li>
</ul>
</li>
</ul>
<h2 id="examples-of-customisations">Examples of Customisations</h2>
<p>Changing MyItems &amp; Unlock all to use Solr, rather than a Sitecore Query.</p>
<p>I told you Sitecore.Query was slow, and Memory Intensive.</p>
<h2 id="since-giving-this-presentation">Since giving this presentation</h2>
<p>Infact since giving this talk, one of the issues that got me to customise the Sitecore Solr Search provider in the first place reared its head again. Sitecore not being able to startup when Solr is down.</p>
<h2 id="i-thought-wed-fixed-that-already-why-cant-sitecore-startup-when-solr-is-down-anymore">I thought we&rsquo;d fixed that already, why can&rsquo;t Sitecore startup when Solr is down anymore?</h2>
<p>Well the number of Sitecore indexes had grown over time.</p>
<p>On startup each index was doing it&rsquo;s own check to see if Solr was available, and waiting for a timeout/exception. This was happening in series. Depending on the type of error you get, if it&rsquo;s a timeout, this can result in Sitecore taking longer to startup than IIS allows. Causing IIS to recycle the application and try again, in an infinite loop. Oh dear!</p>
<p>So I&rsquo;ve added to the github repository a new issue <a href="https://github.com/bigredmachine/sitecore-solr">314454 - Sitecore doesn&rsquo;t startup when Solr Down.</a></p>
<h2 id="simulating-solr-being-downlatency">Simulating Solr being down/latency</h2>
<p>You can simulate on your local PC latency to Solr using Fiddler.
If you change your solr connection string to</p>
<p><code>http://localhost.:8983/solr;solrCloud=true</code></p>
<p>Notice the &ldquo;.&rdquo;, will ensure the traffic goes through Fiddler</p>
<p>Change your Sitecore IIS Application Pool to run as your local user, as Fiddler automatically picks up traffic running as your local user. (Or you can change the proxy settings)</p>
<p>Then run Fiddler.</p>
<p>Under &ldquo;AutoResponder&rdquo; tab, you can add a Rule for requests matching</p>
<p><code>http://localhost.8983/solr/.....</code></p>
<p>The URL you set could be specific to a particular index, or request, depending on what you want to test.</p>
<p>Then on the response for that URL you can set</p>
<p><code>*delay:10000</code></p>
<p>with how many milliseconds you want to delay the request with.</p>
<p>Using this technique you can simulate a network timeout,
rather than a quicker port not listening/service not running error.</p>
<p>And from this can replicate the issue of Sitecore note being able to start up, if you have enough indexes which all timeout on startup.</p>
<h2 id="whats-in-314454---sitecore-doesnt-startup-when-solr-down">What&rsquo;s in 314454 - Sitecore doesn&rsquo;t startup when Solr Down</h2>
<p>Sitecore doesn&rsquo;t issue patches anymore, so instead you get a hotfix, which include all of the issue fixes for that version of Sitecore you are on.</p>
<p>For Sitecore 9.0 update 2, this was included in hotfix 323662-1.</p>
<ol>
<li>If you have enough indexes, the timeout on initialisation from each index run in sequence can result in Sitecore not being starting up in the allowed time.</li>
<li>Retry logic for SolrCloud aliases.</li>
<li>Retry logic for initialising Indexes</li>
<li>Exception handling in IsOnline index check.</li>
<li>Initialisation of indexes to not be interupted if Solr is unavailable, to initialise what can and retry later.</li>
</ol>
<p>And included changes to dlls</p>
<ul>
<li>Sitecore.ContentSearch.Client.dll</li>
<li>Sitecore.ContentSearch.Data.dll</li>
<li>Sitecore.ContentSearch.dll</li>
<li>Sitecore.ContentSearch.Linq.dll</li>
<li>Sitecore.ContentSearch.Linq.Lucene.dll</li>
<li>Sitecore.ContentSearch.LuceneProvider.dll</li>
<li>Sitecore.ContentSearch.SolrNetExtension.dll</li>
<li>Sitecore.ContentSearch.SolrProvider.dll</li>
</ul>
<p>There were so many changes it felt like a mini Sitecore Upgrade.
And the contract changed, had to rewrite our customisations for them to continue to work.</p>
<p>This hotfix fixes the slow startup issue, but only checking once if Solr is available, rather than per index.
So make Sitecore much faster to startup when Solr is down.</p>
<p>There was also some work to make the initialising of the indexes to work in parallel, to further speed up the time for initialisation.</p>
<h2 id="customisations-and-upgrades">Customisations and Upgrades</h2>
<p>Each time we upgrade, we have to see if our customisations are still possible.
Often the extension points/hooks we have used have gone, and we have to find new ones.</p>
<p>Even on this update to the hotfix which address 314454, we had to do a lot of rework.</p>
<p>Hopefully in a future version of the product the extensions points we require will be included in the product, and we won&rsquo;t have to get out dotpeek to find a place to override, and use reflection/so many custom classes and overrides to change the behaviour to what we need.</p>
<h2 id="update-the-repository-examples">Update the repository examples</h2>
<p>As the hotfix isn&rsquo;t publicly available in the nuget feeds, and the contract has changed.</p>
<p>I can&rsquo;t update the example code to reference the hotfix, and for it to still compile.</p>
<p>If anyone is interested to see the examples updated to use the hotfix, reach out to me on twitter, and I can see about creating a feature branch.</p>
<p>Otherwise, I&rsquo;m going to wait for a version of Sitecore which includes these fixes/contract changes, which is available in the nuget feed so I can update the examples and still have it compiling.</p>
</description>
</item>
<item>
<title>Federated Authentication Single Sign Out</title>
<link>https://bigredmachine.github.io/post/federated-authentication-single-sign-out/</link>
<pubDate>Sun, 24 Nov 2019 00:00:00 +0000</pubDate>
<guid>https://bigredmachine.github.io/post/federated-authentication-single-sign-out/</guid>
<description><h2 id="federated-authentication-single-sign-out">Federated Authentication Single Sign Out</h2>
<p>By default when you sign out of Sitecore, you don&rsquo;t get signed out of your Federated Authentication Provider (Tested against Sitecore 9.0). So if after you sign out, you try to sign in again, your Federated Authentication Provider still recognises you and doesn&rsquo;t challenge you to sign back in again, and lets you into the system.</p>
<p>So have you really signed out, if you haven&rsquo;t signed out of the Federated Authentication provider as well? IMHO - no.</p>
<h2 id="whats-the-behaviour-of-other-services-using-federated-authentication">What&rsquo;s the behaviour of other services using Federated Authentication?</h2>
<p>If you log out of Microsoft Office 365, it also logs you out of Azure AD.</p>
<p>If you log out of Azure Portal, it also logs you out of Azure AD.</p>
<p>Also note, on the Azure AD sign out page, it also signs you out of Office 365, if you are signing out from another system. (That&rsquo;s going even further, to an even more true single sign out where sign out of every system at once, but the sign out page has to know about all of the systems to sign you out of, beyond the scope of this post)</p>
<p>So can see it&rsquo;s pretty standard behaviour when you sign out of a system, to also sign you out of the Federated Authentication provider as well.</p>
<h2 id="so-how-do-you-configure-sitecore-to-sign-out-of-a-federated-authentication-provider-as-well-as-sitecore">So how do you configure Sitecore to sign out of a Federated Authentication provider as well as Sitecore? </h2>
<p>Pipelines/Processor + Reflector = Win!</p>
<p> </p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/federated-authentication-single-sign-out/images/pipelines_hu736a386aecced7e0b41b74d134483cb0_84525_650x0_resize_q100_box.jpg">
<source
media="(min-width: 465px)"
srcset="https://bigredmachine.github.io/post/federated-authentication-single-sign-out/images/pipelines_hu736a386aecced7e0b41b74d134483cb0_84525_465x0_resize_q100_box.jpg">
<img
src="https://bigredmachine.github.io/post/federated-authentication-single-sign-out/images/pipelines_hu736a386aecced7e0b41b74d134483cb0_84525_300x0_resize_q100_box.jpg"
alt="Pipelines Everywhere">
</picture>
<p> </p>
<h3 id="for-the-launchpad-using-speak">For the LaunchPad using Speak</h3>
<p>Starting with the Sitecore Launch pad.</p>
<p>The &ldquo;Log out&rdquo; button has some JS that fires &ldquo;-/speak/v1/business/AccountInformation.js&rdquo;.
Which does an AJAX POST request to &ldquo;/sitecore/shell/api/sitecore/Authentication/Logout?sc_database=master&rdquo;.
It then tries to parse the JSON result, and read the &ldquo;Redirect&rdquo; value.
And then redirects the user client side to the specified url.</p>
<p>Serverside this &ldquo;AuthenticationController&rdquo; can be found in &ldquo;Sitecore.Speak.Client.dll&rdquo; &ldquo;Sitecore.Controllers.AuthenticationController&rdquo; &ldquo;Logout&rdquo; HttpPost method.</p>
<p>This in turn calls &ldquo;Sitecore.Shell.Security().Logout&rdquo; passing in an &ldquo;Action<UrlString>&rdquo;, to capture the RedirectUrl for the JSON result.</p>
<p>&ldquo;Sitecore.Shell.Security().Logout&rdquo; calls a pipeline &ldquo;speak.logout&rdquo;.
This pipeline takes an argument of type &ldquo;LogoutArgs&rdquo;.
&ldquo;LogoutArgs&rdquo; has a property &ldquo;RedirectUrl&rdquo; of type &ldquo;UrlString&rdquo;.</p>
<p>&ldquo;RedirectUrl&rdquo; is initialised before the pipeline &ldquo;speak.logout&rdquo; is called with &ldquo;Context.Site.LoginPage&rdquo;.
And after the pipleline is called, the &ldquo;RedirectUrl&rdquo; value is used to send the client to the specified page.</p>
<h3 id="for-the-content-editor">For the Content Editor</h3>
<p>The content editor on the other hand the &ldquo;Log out&rdquo; button calls onclick &ldquo;scForm.postEvent(this, event, &lsquo;system.logout&rsquo;)&rdquo;.</p>
<p>This triggers the processor &ldquo;logout&rdquo;.
This also takes the same argument of type &ldquo;LogoutArgs&rdquo;, as the &ldquo;speak.logout&rdquo; pipeline.</p>
<p>If the &ldquo;LogoutArgs&rdquo; &ldquo;RedirectUrl&rdquo; property has not been set, it triggers a server side redirect to the &ldquo;Client.Site.LoginPage&rdquo;.
See &ldquo;Sitecore.Kernel.dll&rdquo;, &ldquo;Sitecore.Pipelines.Logout.GotoLogin.Process&rdquo; method.</p>
<h3 id="pipeline">Pipeline</h3>
<p>So to override the behaviour of logout going to the Sitecore login page.
Adding a pipeline to &ldquo;Speak.Logout&rdquo; to set the &ldquo;RedirectUrl&rdquo;.</p>
<p>And adding a processor to &ldquo;logout&rdquo; before the processor &ldquo;Sitecore.Pipelines.Logout.GotoLogin, Sitecore.Kernel&rdquo;, to set the &ldquo;RedirectUrl&rdquo;.</p>
<p>We can control where users are redirected to on logout.</p>
<h3 id="federated-authentication-provider-settings-not-used">Federated Authentication Provider settings not used</h3>
<p>Interestingly when configuring a Federated Authentication provider, you specify on the &ldquo;OpenIdConnectAuthenticationOptions&rdquo; class the &ldquo;PostLogoutRedirectUri&rdquo;, but this isn&rsquo;t used.
Still we can reuse this configuration setting in our pipeline.</p>
<h3 id="example-settings-to-change-redirecturl-to">Example settings to change RedirectUrl to</h3>
<p>If you are using Azure AD, you can set the &ldquo;RedirectUrl&rdquo; to something like &ldquo;<a href="https://login.microsoft.com/common/oauth2/logout?post_logout_redirect_uri=https%3A%2F%2fsitecore.local%2Fsitecore%2Flogin%22">https://login.microsoft.com/common/oauth2/logout?post_logout_redirect_uri=https%3A%2F%2fsitecore.local%2Fsitecore%2Flogin&quot;</a></p>
<p>So it will log you out of Azure AD, then redirect you back to &ldquo;<a href="https://sitecore.local/sitecore/login%22">https://sitecore.local/sitecore/login&quot;</a></p>
<p>Obviously change the &ldquo;post_logout_redirect_uri&rdquo; parameter to match your environment.</p>
<h2 id="summary">Summary</h2>
<p>With a few extra pipelines, and with the help of reflector, can achieve single sign out with Sitecore.</p>
<h2 id="additional-notes">Additional Notes</h2>
<p>I&rsquo;ve only tried this with Sitecore 9.0, and not with the new Sitecore Identity Server.</p>
</description>
</item>
<item>
<title>Sitecore 9.3 - media library blob storage</title>
<link>https://bigredmachine.github.io/post/sitecore-9-dot-3-media-library-blob-storage/</link>
<pubDate>Sun, 24 Nov 2019 00:00:00 +0000</pubDate>
<guid>https://bigredmachine.github.io/post/sitecore-9-dot-3-media-library-blob-storage/</guid>
<description><h2 id="sitecore-93---media-library-azure-blob-storage">Sitecore 9.3 - Media Library Azure Blob Storage</h2>
<p>Pieter Brinkman put out this tweet after Sitecore Symposium, with some information about Sitecore 9.3, for those of us not able to attend Symposium and see his session.</p>
<blockquote class="twitter-tweet"><p lang="en" dir="ltr"><a href="https://twitter.com/hashtag/SitecoreSYM?src=hash&amp;ref_src=twsrc%5Etfw">#SitecoreSYM</a> is a wrap, now time to fire-up that blog engine. A lot to share around the upcoming <a href="https://twitter.com/hashtag/Sitecore?src=hash&amp;ref_src=twsrc%5Etfw">#Sitecore</a> 9.3 release. As a gesture of good will -&gt; the deck is now available on <a href="https://t.co/B9DqZ9CLot">https://t.co/B9DqZ9CLot</a> <a href="https://t.co/B8Qy9kzLF0">pic.twitter.com/B8Qy9kzLF0</a></p>&mdash; Pieter Brinkman (@pieterbrink123) <a href="https://twitter.com/pieterbrink123/status/1194279592447660032?ref_src=twsrc%5Etfw">November 12, 2019</a></blockquote>
<script async src="https://platform.twitter.com/widgets.js" charset="utf-8"></script>
<p>I also saw him present the at the London Sitecore User Group where he explained this slide deck in some more detail.</p>
<p>Some highlights for me:</p>
<h3 id="slide-12---top-5-core-platform-enhancements">Slide 12 - Top 5 core platform enhancements</h3>
<ul>
<li>Removing obsolete functionality &ldquo;Lucene Search Provider&rdquo; &amp; &ldquo;Sitecore.Nvelocity&rdquo;</li>
<li>Experience Editor Improvements</li>
<li>Performance Improvements &ldquo;Reducing number of calls&rdquo;, &ldquo;Reduce the amount of data saved&rdquo;</li>
</ul>
<p>I look forward to seeing some benchmarks about the performance improvements, and what areas this covers. When I first read this I had thought this might mean Sql calls, but he explained at the User Group was referring to personalisation/tracking.
Still he only listed out his top 5, and hopefully the performance improvements extends to other areas.</p>
<h3 id="slide-35---paas---media-library-in-azure-blob-storage">Slide 35 - PaaS - Media Library in Azure Blob Storage</h3>
<p>There is a footnote on Slide 35 &amp; 36</p>
<p>&ldquo;Media Library in Azure Blob Storage: Reduce the cost and increase the performance of storing media in Sitecore&rdquo;</p>
<p>This is only listed for &ldquo;PaaS&rdquo; and the new &ldquo;Managed Cloud&rdquo;.</p>
<p>However I&rsquo;d like to explore using Azure Blob Storage on Azure IaaS.
Let&rsquo;s see if this is possible when Sitecore 9.3 comes out.</p>
<p>I&rsquo;d also like to find out if the Azure Blob Storage implementation just replaces the backing storage.</p>
<p>Or if it&rsquo;s possible to serve the files direct from Blob Storage, bypassing the Sitecore Media library handler/disk caching on the Content Delivery servers. (Possibly with some other technology/thin API layer to provider resizing functionality)</p>
<h3 id="summary">Summary</h3>
<p>Using Azure Blob Storage for the Sitecore Media Library for me is the most exciting development of Sitecore 9.3 that I&rsquo;ve heard about so far, as could significantly reduce the size of the Sql databases.</p>
<h3 id="notes">Notes</h3>
<p>Jammy Kam has customised Sitecore do use Azure Blob Storage for it&rsquo;s Media Library before
<a href="https://jammykam.wordpress.com/2015/12/01/sitecore-media-library-in-azure-cloud-storage-part-1/">https://jammykam.wordpress.com/2015/12/01/sitecore-media-library-in-azure-cloud-storage-part-1/</a>
<a href="https://jammykam.wordpress.com/2015/12/03/sitecore-media-library-in-azure-cloud-storage-part-2/">https://jammykam.wordpress.com/2015/12/03/sitecore-media-library-in-azure-cloud-storage-part-2/</a></p>
<p>I wonder how similar the Sitecore implementation will be.</p>
</description>
</item>
<item>
<title>Java goes commercial, what does this mean for Sitecore - Solr?</title>
<link>https://bigredmachine.github.io/post/java-comercial-sitecore-solr-future/</link>
<pubDate>Sat, 15 Dec 2018 00:00:00 +0000</pubDate>
<guid>https://bigredmachine.github.io/post/java-comercial-sitecore-solr-future/</guid>
<description><h1 id="oracle-to-start-charging-for-java">Oracle to start charging for Java</h1>
<p>I&rsquo;ve been following this update for Java for many months since initally seeing the popup warning when updating Java on my local machine. And now finally getting around to blog about it, now the options have become clearer. And finding that some people still aren&rsquo;t aware about the pending update in January 2019 that will mean will require a commercial Java licence if you want any further security updates from the Oracle version of Java.</p>
<p>I brought this topic up at the Sitecore Discussion Club<br>
<a href="http://sitecore.events/">http://sitecore.events/</a></p>
<p> </p>
<picture>
<source
media="(min-width: 650px)"
srcset="https://bigredmachine.github.io/post/java-comercial-sitecore-solr-future/images/JavaPublicUpdates_huea38cc4076a499c2d07c53f82223e548_19941_650x0_resize_q100_box_2.PNG">
<source