* Only prevent the unpublish or delete of a related item when configured to do so if it is related as a child, not as a parent (#18886) * Only prevent the unpubkish or delete of a related item when configured to do so if it is related as a child, not as a parent. * Fixed incorect parameter names. * Fixed failing integration tests. * Use using variable instead to reduce nesting * Applied suggestions from code review. * Used simple using statement throughout RelationService for consistency. * Applied XML header comments consistently. --------- Co-authored-by: mole <nikolajlauridsen@protonmail.ch> * Feature: highlight invariant doc with variant blocks is unsupported (#18806) * mark variant blocks in invariant docs as invalid * implement RTE Blocks * Fix pagination for users restricted by start nodes (#18907) * Fix pagination for users restricted by start nodes * Default implementation to avoid breakage * Review comments * Fix failing test * Add media start node tests * Fix issue preventing blueprint derived values from being scaffolded (#18917) * Fix issue preventing blueprint derived values from being scaffolded. * fix manipulating frooen array * compare with variantId as well --------- Co-authored-by: Niels Lyngsø <niels.lyngso@gmail.com> * ci: add Azure Static Web Apps workflow file on-behalf-of: @Azure opensource@microsoft.com * ci: add Azure Static Web Apps workflow file on-behalf-of: @Azure opensource@microsoft.com * ci: add Azure Static Web Apps workflow file on-behalf-of: @Azure opensource@microsoft.com * Remove admin permission on user configuration, allowing users with user section access only to manaage users and groups. (#18848) * Tiptap RTE: Style Menu extension kind (#18918) * Adds 'styleMenu' Tiptap toolbar extension kind * Adds icons for `<h4>` and `<p>` tags * Adds commands to HTML Global Attributes extension for setting the `class` and `id` attributes. * Renamed "default-tiptap-toolbar-element.api.ts" file The "element" part was confusing. * Toolbar Menu: uses correct `item` value * Cascading Menu: adds localization for the label * Adds `label` attribute to UUI components for accessibility. * Toolbar Menu: uses correct `appearance` value * Removed unrequired `api` from Style Select * Destructs the `item.data` object * Ensure has children reflects only items with folder children when folders only are queried. (#18790) * Ensure has children reflects only items with folder children when folders only are queried. * Added supression for change to integration test public code. --------- Co-authored-by: Migaroez <geusens@gmail.com> * Only apply validation on content update to variant cultures where the editor has permission for the culture (#18778) * Only apply validation on content update to variant cultures where the editor has permission for the culture. * Remove inadvertent comment updates. * Fixed failing integration test. * Adds ancestor ID details on document tree and collection responses (#18909) * Populate ancestor keys on document tree response items. * Populate ancestor keys on document collection response items. * Update OpenApi.json * Use array of objects rather than Ids for the ancestor collection. * Update OpenApi.json. * Move publish with descendants to a background task with polling (#18497) * Use background queue for database cache rebuild and track rebuilding status. * Updated OpenApi.json and client-side types. * Updated client to poll for completion of database rebuild. * Move IBackgroundTaskQueue to core and prepare publish branch to run as background task. * Endpoints for retrieval of status and result from branch publish operations. * Poll and retrieve result for publish with descendants. * Handled issues from testing. * Rework to single controller for status and result. * Updated client side sdk. * OpenApi post dev merge gen --------- Co-authored-by: Migaroez <geusens@gmail.com> * Clear roots before rebuilding navigation dictionary (#18766) * Clear roots before rebuilding navigation dictionary. * Added tests to verify fix. * Correct test implementation. * Convert integration tests with method overloads into test cases. * Integration test compatibility supressions. * Fixes save of empty, invariant block list on variant content. (#18932) * remove unnecessary code (#18927) * V15/bugfix/fix route issue from 18859 (#18931) * unique check * unique for workspace empty path * more unique routes * Bump vite from 6.2.3 to 6.2.4 in /src/Umbraco.Web.UI.Client Bumps [vite](https://github.com/vitejs/vite/tree/HEAD/packages/vite) from 6.2.3 to 6.2.4. - [Release notes](https://github.com/vitejs/vite/releases) - [Changelog](https://github.com/vitejs/vite/blob/v6.2.4/packages/vite/CHANGELOG.md) - [Commits](https://github.com/vitejs/vite/commits/v6.2.4/packages/vite) --- updated-dependencies: - dependency-name: vite dependency-version: 6.2.4 dependency-type: direct:development ... Signed-off-by: dependabot[bot] <support@github.com> * removes autogenerated workflows * make getHasUnpersistedChanges public (#18929) * Added management API endpoint, service and repository for retrieval of references from the recycle bin (#18882) * Added management API endpoint, service and repository for retrieval of references from the recycle bin. * Update src/Umbraco.Cms.Api.Management/Controllers/Document/RecycleBin/ReferencedByDocumentRecycleBinController.cs Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Removed unused code. --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Updated management API endpoint and model for data type references to align with that used for documents, media etc. (#18905) * Updated management API endpoint and model for data type references to align with that used for documents, media etc. * Refactoring. * Update src/Umbraco.Core/Constants-ReferenceTypes.cs Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Fixed typos. * Added id to tracked reference content type response. * Updated OpenApi.json. * Added missing updates. * Renamed model and constants from code review feedback. * Fix typo * Fix multiple enumeration --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: mole <nikolajlauridsen@protonmail.ch> * Skip lock tests * Look-up redirect in content finder for multi-lingual sites using path and legacy route prefixed with the integer ID of the node with domains defined (#18763) * Look-up redirect in content finder for multi-lingual sites using path and legacy route prefixed with the integer ID of the node with domains defined. * Added tests to verify functionality. * Added reference to previous PR. * Referenced second PR. * Assemble URLs for all cultures, not just the default. * Revert previous update. * Display an original URL if we have one. * Bump vite from 6.2.4 to 6.2.5 in /src/Umbraco.Web.UI.Client Bumps [vite](https://github.com/vitejs/vite/tree/HEAD/packages/vite) from 6.2.4 to 6.2.5. - [Release notes](https://github.com/vitejs/vite/releases) - [Changelog](https://github.com/vitejs/vite/blob/v6.2.5/packages/vite/CHANGELOG.md) - [Commits](https://github.com/vitejs/vite/commits/v6.2.5/packages/vite) --- updated-dependencies: - dependency-name: vite dependency-version: 6.2.5 dependency-type: direct:development ... Signed-off-by: dependabot[bot] <support@github.com> * Add raw value validation to multiple text strings property editor (#18936) * Add raw value validation to multiple text strings property editor * Added additional assert on unit test and comment on validation logic. * Don't remove items to obtain a valid value --------- Co-authored-by: Andy Butland <abutland73@gmail.com> * Integration tests for content publishing with ancestor unpublished (#18941) * Resolved warnings in test class. * Refactor regions into partial classes. * Aligned test names. * Variable name refactoring. * Added tests for unpublished paths. * Adjust tests to verify current behaviour. * Cleaned up project file. * fix circular icon import (#18952) * remove segment toggle for elements (#18949) * Fix modal route registration circular import (#18953) * fix modal route registration circular import * Update modal-route-registration.controller.ts * V15/fix/18595 (#18925) * fix for #18595 * updates the en.ts * Avoid unneeded Dictionary operations (#18890) * Avoid some heap allocations * Remove unneeded double seek * Avoid allocating new empty arrays, reuse existing empty array * Avoid allocating strings for parsing comma separated int values (#18199) * Data type References UI: Workspace + Delete (#18914) * Updated management API endpoint and model for data type references to align with that used for documents, media etc. * Refactoring. * Update src/Umbraco.Core/Constants-ReferenceTypes.cs Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Fixed typos. * generate server models * add extension slot * register data type reference info app * add reference data mappers * Added id to tracked reference content type response. * Updated OpenApi.json. * Added missing updates. * generate new models * update models * register ref item * remove debugger * render types * register member type property type ref * register media type property type ref * Renamed model and constants from code review feedback. * register reference workspace info app kind * use kind for document references * use kind for media references * use kind for member references * use deleteWithRelation kind when deleting data types * fix manifest types * fix types * Update types.gen.ts * update code to fit new server models --------- Co-authored-by: Andy Butland <abutland73@gmail.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Feature: discard changes for block workspace (#18930) * make getHasUnpersistedChanges public * Discard changes impl for Block Workspace * fix 18367 (#18956) * Merge commit from fork * Prevent path traveral vulnerability with upload of temporary files. * Used BadRequest instead of NotFound for invalid file name response. * V15 QA Fixing the failing media acceptance tests (#18881) * Fixed the function name due to test helper changes * Updated assertion steps due to UI changes * Added more waits * Bumped version * Increase timeout * Reverted --------- Co-authored-by: Andreas Zerbst <73799582+andr317c@users.noreply.github.com> * V15 QA added clipboard test for not being able to copy to root when block is not allowed at root (#18937) * Added clipboard test * Bumped version * Updated to use the name * Run all tests on the pipeline * Reverted command * build: adjusts circular ref number to 4 --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Andy Butland <abutland73@gmail.com> Co-authored-by: mole <nikolajlauridsen@protonmail.ch> Co-authored-by: Niels Lyngsø <nsl@umbraco.dk> Co-authored-by: Niels Lyngsø <niels.lyngso@gmail.com> Co-authored-by: Jacob Overgaard <752371+iOvergaard@users.noreply.github.com> Co-authored-by: Lee Kelleher <leekelleher@users.noreply.github.com> Co-authored-by: Migaroez <geusens@gmail.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Nikolaj Geisle <70372949+Zeegaan@users.noreply.github.com> Co-authored-by: Mads Rasmussen <madsr@hey.com> Co-authored-by: Jacob Welander Jensen <64834767+Welander1994@users.noreply.github.com> Co-authored-by: Henrik <hg@impact.dk> Co-authored-by: Sebastiaan Janssen <sebastiaan@umbraco.com> Co-authored-by: Nhu Dinh <150406148+nhudinh0309@users.noreply.github.com> Co-authored-by: Andreas Zerbst <73799582+andr317c@users.noreply.github.com>
664 lines
20 KiB
C#
664 lines
20 KiB
C#
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
using Microsoft.Extensions.Logging;
|
|
using NPoco;
|
|
using NUnit.Framework;
|
|
using Umbraco.Cms.Core;
|
|
using Umbraco.Cms.Core.DistributedLocking.Exceptions;
|
|
using Umbraco.Cms.Infrastructure.Persistence.Dtos;
|
|
using Umbraco.Cms.Persistence.Sqlite.Interceptors;
|
|
using Umbraco.Cms.Tests.Common.Attributes;
|
|
using Umbraco.Cms.Tests.Common.Testing;
|
|
using Umbraco.Cms.Tests.Integration.Testing;
|
|
|
|
namespace Umbraco.Cms.Tests.Integration.Umbraco.Infrastructure.Persistence;
|
|
|
|
[TestFixture]
|
|
[Timeout(60000)]
|
|
[UmbracoTest(Database = UmbracoTestOptions.Database.NewSchemaPerTest, Logger = UmbracoTestOptions.Logger.Console)]
|
|
internal sealed class LocksTests : UmbracoIntegrationTest
|
|
{
|
|
[SetUp]
|
|
protected void SetUp()
|
|
{
|
|
// create a few lock objects
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
var database = ScopeAccessor.AmbientScope.Database;
|
|
database.Insert("umbracoLock", "id", false, new LockDto { Id = 1, Name = "Lock.1" });
|
|
database.Insert("umbracoLock", "id", false, new LockDto { Id = 2, Name = "Lock.2" });
|
|
database.Insert("umbracoLock", "id", false, new LockDto { Id = 3, Name = "Lock.3" });
|
|
scope.Complete();
|
|
}
|
|
}
|
|
|
|
protected override void ConfigureTestServices(IServiceCollection services) =>
|
|
// SQLite + retry policy makes tests fail, we retry before throwing distributed locking timeout.
|
|
services.RemoveAll(x => !x.IsKeyedService && x.ImplementationType == typeof(SqliteAddRetryPolicyInterceptor));
|
|
|
|
[Test]
|
|
public void SingleReadLockTest()
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
scope.EagerReadLock(Constants.Locks.Servers);
|
|
scope.Complete();
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void ConcurrentReadersTest()
|
|
{
|
|
const int threadCount = 8;
|
|
var threads = new Thread[threadCount];
|
|
var exceptions = new Exception[threadCount];
|
|
var locker = new object();
|
|
var acquired = 0;
|
|
var m2 = new ManualResetEventSlim(false);
|
|
var m1 = new ManualResetEventSlim(false);
|
|
|
|
for (var i = 0; i < threadCount; i++)
|
|
{
|
|
var ic = i; // capture
|
|
threads[i] = new Thread(() =>
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
try
|
|
{
|
|
scope.EagerReadLock(Constants.Locks.Servers);
|
|
lock (locker)
|
|
{
|
|
acquired++;
|
|
if (acquired == threadCount)
|
|
{
|
|
m2.Set();
|
|
}
|
|
}
|
|
|
|
m1.Wait();
|
|
lock (locker)
|
|
{
|
|
acquired--;
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
exceptions[ic] = e;
|
|
}
|
|
|
|
scope.Complete();
|
|
}
|
|
});
|
|
}
|
|
|
|
// ensure that current scope does not leak into starting threads
|
|
using (ExecutionContext.SuppressFlow())
|
|
{
|
|
foreach (var thread in threads)
|
|
{
|
|
thread.Start();
|
|
}
|
|
}
|
|
|
|
m2.Wait();
|
|
// all threads have locked in parallel
|
|
var maxAcquired = acquired;
|
|
m1.Set();
|
|
|
|
foreach (var thread in threads)
|
|
{
|
|
thread.Join();
|
|
}
|
|
|
|
Assert.AreEqual(threadCount, maxAcquired);
|
|
Assert.AreEqual(0, acquired);
|
|
|
|
for (var i = 0; i < threadCount; i++)
|
|
{
|
|
Assert.IsNull(exceptions[i]);
|
|
}
|
|
}
|
|
|
|
[NUnit.Framework.Ignore("We currently do not have a way to force lazy locks")]
|
|
[Test]
|
|
public void GivenNonEagerLocking_WhenNoDbIsAccessed_ThenNoSqlIsExecuted()
|
|
{
|
|
var sqlCount = 0;
|
|
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
var db = ScopeAccessor.AmbientScope.Database;
|
|
try
|
|
{
|
|
db.EnableSqlCount = true;
|
|
|
|
// Issue a lock request, but we are using non-eager
|
|
// locks so this only queues the request.
|
|
// The lock will not be issued unless we resolve
|
|
// scope.Database
|
|
scope.WriteLock(Constants.Locks.Servers);
|
|
|
|
sqlCount = db.SqlCount;
|
|
}
|
|
finally
|
|
{
|
|
db.EnableSqlCount = false;
|
|
}
|
|
}
|
|
|
|
Assert.AreEqual(0, sqlCount);
|
|
}
|
|
|
|
[NUnit.Framework.Ignore("We currently do not have a way to force lazy locks")]
|
|
[Test]
|
|
public void GivenNonEagerLocking_WhenDbIsAccessed_ThenSqlIsExecuted()
|
|
{
|
|
var sqlCount = 0;
|
|
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
var db = ScopeAccessor.AmbientScope.Database;
|
|
try
|
|
{
|
|
db.EnableSqlCount = true;
|
|
|
|
// Issue a lock request, but we are using non-eager
|
|
// locks so this only queues the request.
|
|
// The lock will not be issued unless we resolve
|
|
// scope.Database
|
|
scope.WriteLock(Constants.Locks.Servers);
|
|
|
|
scope.Database.ExecuteScalar<int>("SELECT 1");
|
|
|
|
sqlCount = db.SqlCount;
|
|
}
|
|
finally
|
|
{
|
|
db.EnableSqlCount = false;
|
|
}
|
|
}
|
|
|
|
Assert.AreEqual(2,sqlCount);
|
|
}
|
|
|
|
[Test]
|
|
[LongRunning]
|
|
public void ConcurrentWritersTest()
|
|
{
|
|
const int threadCount = 8;
|
|
var threads = new Thread[threadCount];
|
|
var exceptions = new Exception[threadCount];
|
|
Lock locker = new();
|
|
var acquired = 0;
|
|
var entered = 0;
|
|
var ms = new AutoResetEvent[threadCount];
|
|
for (var i = 0; i < threadCount; i++)
|
|
{
|
|
ms[i] = new AutoResetEvent(false);
|
|
}
|
|
|
|
var m1 = new ManualResetEventSlim(false);
|
|
|
|
for (var i = 0; i < threadCount; i++)
|
|
{
|
|
var ic = i; // capture
|
|
threads[i] = new Thread(() =>
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
try
|
|
{
|
|
lock (locker)
|
|
{
|
|
entered++;
|
|
if (entered == threadCount)
|
|
{
|
|
m1.Set();
|
|
}
|
|
}
|
|
|
|
ms[ic].WaitOne();
|
|
scope.EagerWriteLock(Constants.Locks.Servers);
|
|
lock (locker)
|
|
{
|
|
acquired++;
|
|
}
|
|
|
|
ms[ic].WaitOne();
|
|
lock (locker)
|
|
{
|
|
acquired--;
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
exceptions[ic] = e;
|
|
}
|
|
|
|
scope.Complete();
|
|
}
|
|
});
|
|
}
|
|
|
|
// ensure that current scope does not leak into starting threads
|
|
using (ExecutionContext.SuppressFlow())
|
|
{
|
|
foreach (var thread in threads)
|
|
{
|
|
thread.Start();
|
|
}
|
|
}
|
|
|
|
m1.Wait();
|
|
// all threads have entered
|
|
ms[0].Set(); // let 0 go
|
|
// TODO: This timing is flaky
|
|
Thread.Sleep(100);
|
|
for (var i = 1; i < threadCount; i++)
|
|
{
|
|
ms[i].Set(); // let others go
|
|
}
|
|
|
|
// TODO: This timing is flaky
|
|
Thread.Sleep(500);
|
|
// only 1 thread has locked
|
|
Assert.AreEqual(1, acquired);
|
|
for (var i = 0; i < threadCount; i++)
|
|
{
|
|
ms[i].Set(); // let all go
|
|
}
|
|
|
|
foreach (var thread in threads)
|
|
{
|
|
thread.Join();
|
|
}
|
|
|
|
Assert.AreEqual(0, acquired);
|
|
|
|
for (var i = 0; i < threadCount; i++)
|
|
{
|
|
Assert.IsNull(exceptions[i]);
|
|
}
|
|
}
|
|
|
|
[Retry(10)] // TODO make this test non-flaky.
|
|
[Test]
|
|
public void DeadLockTest()
|
|
{
|
|
if (BaseTestDatabase.IsSqlite())
|
|
{
|
|
Assert.Ignore("This test doesn't work with Microsoft.Data.Sqlite - SELECT * FROM sys.dm_tran_locks;");
|
|
return;
|
|
}
|
|
|
|
Exception e1 = null, e2 = null;
|
|
AutoResetEvent ev1 = new(false), ev2 = new(false);
|
|
|
|
// testing:
|
|
// two threads will each obtain exclusive write locks over two
|
|
// identical lock objects deadlock each other
|
|
|
|
var thread1 = new Thread(() => DeadLockTestThread(1, 2, ev1, ev2, ref e1));
|
|
var thread2 = new Thread(() => DeadLockTestThread(2, 1, ev2, ev1, ref e2));
|
|
|
|
// ensure that current scope does not leak into starting threads
|
|
using (ExecutionContext.SuppressFlow())
|
|
{
|
|
thread1.Start();
|
|
thread2.Start();
|
|
}
|
|
|
|
ev2.Set();
|
|
|
|
thread1.Join();
|
|
thread2.Join();
|
|
|
|
//Assert.IsNotNull(e1);
|
|
if (e1 != null)
|
|
{
|
|
AssertIsDistributedLockingTimeoutException(e1);
|
|
}
|
|
|
|
// the assertion below depends on timing conditions - on a fast enough environment,
|
|
// thread1 dies (deadlock) and frees thread2, which succeeds - however on a slow
|
|
// environment (CI) both threads can end up dying due to deadlock - so, cannot test
|
|
// that e2 is null - but if it's not, can test that it's a timeout
|
|
//
|
|
//Assert.IsNull(e2);
|
|
if (e2 != null)
|
|
{
|
|
AssertIsDistributedLockingTimeoutException(e2);
|
|
}
|
|
}
|
|
|
|
private void AssertIsDistributedLockingTimeoutException(Exception e)
|
|
{
|
|
var sqlException = e as DistributedLockingTimeoutException;
|
|
Assert.IsNotNull(sqlException);
|
|
}
|
|
|
|
private void DeadLockTestThread(int id1, int id2, EventWaitHandle myEv, WaitHandle otherEv, ref Exception exception)
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
try
|
|
{
|
|
otherEv.WaitOne();
|
|
Console.WriteLine($"[{id1}] WAIT {id1}");
|
|
scope.EagerWriteLock(id1);
|
|
Console.WriteLine($"[{id1}] GRANT {id1}");
|
|
WriteLocks(ScopeAccessor.AmbientScope.Database);
|
|
myEv.Set();
|
|
|
|
if (id1 == 1)
|
|
{
|
|
otherEv.WaitOne();
|
|
}
|
|
else
|
|
{
|
|
Thread.Sleep(5200); // wait for deadlock...
|
|
}
|
|
|
|
Console.WriteLine($"[{id1}] WAIT {id2}");
|
|
scope.EagerWriteLock(id2);
|
|
Console.WriteLine($"[{id1}] GRANT {id2}");
|
|
WriteLocks(ScopeAccessor.AmbientScope.Database);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
exception = e;
|
|
}
|
|
finally
|
|
{
|
|
scope.Complete();
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void NoDeadLockTest()
|
|
{
|
|
if (BaseTestDatabase.IsSqlite())
|
|
{
|
|
Assert.Ignore("This test doesn't work with Microsoft.Data.Sqlite - SELECT * FROM sys.dm_tran_locks;");
|
|
return;
|
|
}
|
|
|
|
|
|
Exception e1 = null, e2 = null;
|
|
AutoResetEvent ev1 = new(false), ev2 = new(false);
|
|
|
|
// testing:
|
|
// two threads will each obtain exclusive write lock over two
|
|
// different lock objects without blocking each other
|
|
|
|
var thread1 = new Thread(() => NoDeadLockTestThread(1, ev1, ev2, ref e1));
|
|
var thread2 = new Thread(() => NoDeadLockTestThread(2, ev2, ev1, ref e1));
|
|
|
|
// ensure that current scope does not leak into starting threads
|
|
using (ExecutionContext.SuppressFlow())
|
|
{
|
|
thread1.Start();
|
|
thread2.Start();
|
|
}
|
|
|
|
ev2.Set();
|
|
|
|
thread1.Join();
|
|
thread2.Join();
|
|
|
|
Assert.IsNull(e1);
|
|
Assert.IsNull(e2);
|
|
}
|
|
|
|
[Test]
|
|
public void Throws_When_Lock_Timeout_Is_Exceeded_Read()
|
|
{
|
|
if (BaseTestDatabase.IsSqlite())
|
|
{
|
|
// Reader reads snapshot, isolated from the writer.
|
|
Assert.Ignore("Doesn't apply to SQLite with journal_mode=wal");
|
|
}
|
|
|
|
using (ExecutionContext.SuppressFlow())
|
|
{
|
|
var t1 = Task.Run(() =>
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
Console.WriteLine("Write lock A");
|
|
// This will acquire right away
|
|
scope.EagerWriteLock(TimeSpan.FromMilliseconds(2000), Constants.Locks.ContentTree);
|
|
Thread.Sleep(6000); // Wait longer than the Read Lock B timeout
|
|
scope.Complete();
|
|
Console.WriteLine("Finished Write lock A");
|
|
}
|
|
});
|
|
|
|
Thread.Sleep(500); // 100% sure task 1 starts first
|
|
|
|
var t2 = Task.Run(() =>
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
Console.WriteLine("Read lock B");
|
|
|
|
// This will wait for the write lock to release but it isn't going to wait long
|
|
// enough so an exception will be thrown.
|
|
Assert.Throws<DistributedReadLockTimeoutException>(() =>
|
|
scope.EagerReadLock(TimeSpan.FromMilliseconds(3000), Constants.Locks.ContentTree));
|
|
scope.Complete();
|
|
Console.WriteLine("Finished Read lock B");
|
|
}
|
|
});
|
|
|
|
Task.WaitAll(t1, t2);
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
[LongRunning]
|
|
public void Throws_When_Lock_Timeout_Is_Exceeded_Write()
|
|
{
|
|
var counter = 0;
|
|
var gate = new ManualResetEventSlim(false);
|
|
var logger = GetRequiredService<ILogger<LocksTests>>();
|
|
|
|
using (ExecutionContext.SuppressFlow())
|
|
{
|
|
var t1 = Task.Run(() =>
|
|
{
|
|
using var scope = ScopeProvider.CreateScope();
|
|
|
|
_ = scope.Database; // Begin transaction
|
|
Interlocked.Increment(ref counter);
|
|
gate.Wait();
|
|
|
|
logger.LogInformation("t1 - Attempting to acquire write lock");
|
|
Assert.DoesNotThrow(() =>
|
|
{
|
|
// ReSharper disable once AccessToDisposedClosure
|
|
// This will acquire right away
|
|
scope.EagerWriteLock(TimeSpan.FromMilliseconds(1000), Constants.Locks.ContentTree);
|
|
});
|
|
|
|
logger.LogInformation("t1 - Acquired write lock, sleeping");
|
|
Thread.Sleep(1500); // Wait longer than the Read Lock B timeout
|
|
|
|
scope.Complete();
|
|
logger.LogInformation("t1 - Complete transaction");
|
|
});
|
|
|
|
var t2 = Task.Run(() =>
|
|
{
|
|
using var scope = ScopeProvider.CreateScope();
|
|
|
|
_ = scope.Database; // Begin transaction
|
|
Interlocked.Increment(ref counter);
|
|
gate.Wait();
|
|
Thread.Sleep(100); // Let other transaction obtain write lock first.
|
|
|
|
logger.LogInformation("t2 - Attempting to acquire write lock");
|
|
var ex = Assert.Throws<DistributedWriteLockTimeoutException>(() =>
|
|
{
|
|
// ReSharper disable once AccessToDisposedClosure
|
|
scope.EagerWriteLock(TimeSpan.FromMilliseconds(1000), Constants.Locks.ContentTree);
|
|
logger.LogInformation("t2 - Acquired write lock, something has gone wrong.");
|
|
});
|
|
|
|
if (ex != null)
|
|
{
|
|
logger.LogInformation("t2 - Failed to acquire write lock in time, all is well.");
|
|
}
|
|
|
|
scope.Complete();
|
|
});
|
|
|
|
while (counter < 2)
|
|
{
|
|
Thread.Sleep(10);
|
|
}
|
|
|
|
gate.Set();
|
|
Task.WaitAll(t1, t2);
|
|
}
|
|
}
|
|
|
|
[NUnit.Framework.Ignore("This test is very flaky, and is stopping our nightlys")]
|
|
[Test]
|
|
public void Read_Lock_Waits_For_Write_Lock()
|
|
{
|
|
if (BaseTestDatabase.IsSqlite())
|
|
{
|
|
// Reader reads snapshot, isolated from the writer.
|
|
Assert.Ignore("Doesn't apply to SQLite with journal_mode=wal");
|
|
}
|
|
|
|
var locksCompleted = 0;
|
|
|
|
using (ExecutionContext.SuppressFlow())
|
|
{
|
|
var t1 = Task.Run(() =>
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
Console.WriteLine("Write lock A");
|
|
// This will acquire right away
|
|
scope.EagerWriteLock(TimeSpan.FromMilliseconds(2000), Constants.Locks.ContentTree);
|
|
Thread.Sleep(4000); // Wait less than the Read Lock B timeout
|
|
scope.Complete();
|
|
Interlocked.Increment(ref locksCompleted);
|
|
Console.WriteLine("Finished Write lock A");
|
|
}
|
|
});
|
|
|
|
Thread.Sleep(500); // 100% sure task 1 starts first
|
|
|
|
var t2 = Task.Run(() =>
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
Console.WriteLine("Read lock B");
|
|
|
|
// This will wait for the write lock to release
|
|
Assert.DoesNotThrow(() =>
|
|
scope.EagerReadLock(TimeSpan.FromMilliseconds(6000), Constants.Locks.ContentTree));
|
|
|
|
Assert.GreaterOrEqual(locksCompleted, 1);
|
|
|
|
scope.Complete();
|
|
Interlocked.Increment(ref locksCompleted);
|
|
Console.WriteLine("Finished Read lock B");
|
|
}
|
|
});
|
|
|
|
|
|
var t3 = Task.Run(() =>
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
Console.WriteLine("Read lock C");
|
|
|
|
// This will wait for the write lock to release
|
|
Assert.DoesNotThrow(() =>
|
|
scope.EagerReadLock(TimeSpan.FromMilliseconds(6000), Constants.Locks.ContentTree));
|
|
|
|
Assert.GreaterOrEqual(locksCompleted, 1);
|
|
|
|
scope.Complete();
|
|
Interlocked.Increment(ref locksCompleted);
|
|
Console.WriteLine("Finished Read lock C");
|
|
}
|
|
});
|
|
|
|
Task.WaitAll(t1, t2, t3);
|
|
}
|
|
|
|
Assert.AreEqual(3, locksCompleted);
|
|
}
|
|
|
|
[Test]
|
|
public void Lock_Exceeds_Command_Timeout()
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
var realDb = (Database)ScopeAccessor.AmbientScope.Database;
|
|
realDb.CommandTimeout = 1000;
|
|
|
|
Console.WriteLine("Write lock A");
|
|
// TODO: In theory this would throw
|
|
scope.EagerWriteLock(TimeSpan.FromMilliseconds(3000), Constants.Locks.ContentTree);
|
|
scope.Complete();
|
|
Console.WriteLine("Finished Write lock A");
|
|
}
|
|
}
|
|
|
|
|
|
private void NoDeadLockTestThread(int id, EventWaitHandle myEv, WaitHandle otherEv, ref Exception exception)
|
|
{
|
|
using (var scope = ScopeProvider.CreateScope())
|
|
{
|
|
try
|
|
{
|
|
otherEv.WaitOne();
|
|
Console.WriteLine($"[{id}] WAIT {id}");
|
|
scope.EagerWriteLock(id);
|
|
Console.WriteLine($"[{id}] GRANT {id}");
|
|
WriteLocks(ScopeAccessor.AmbientScope.Database);
|
|
myEv.Set();
|
|
otherEv.WaitOne();
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
exception = e;
|
|
}
|
|
finally
|
|
{
|
|
scope.Complete();
|
|
myEv.Set();
|
|
}
|
|
}
|
|
}
|
|
|
|
private void WriteLocks(IDatabaseQuery database)
|
|
{
|
|
Console.WriteLine("LOCKS:");
|
|
var info = database.Query<dynamic>("SELECT * FROM sys.dm_tran_locks;").ToList();
|
|
var sb = new StringBuilder("> ");
|
|
foreach (var row in info)
|
|
{
|
|
if (row is IDictionary<string, object> values)
|
|
{
|
|
sb.AppendJoin(", ", values);
|
|
}
|
|
|
|
sb.AppendLine(string.Empty);
|
|
}
|
|
|
|
Console.WriteLine(sb.ToString());
|
|
}
|
|
}
|