mirror of
				https://github.com/InsanusMokrassar/MicroUtils.git
				synced 2025-10-26 17:50:41 +00:00 
			
		
		
		
	add tests for smartrwlocker
This commit is contained in:
		| @@ -12,9 +12,9 @@ import kotlin.contracts.contract | ||||
|  * * [lockWrite] will lock [writeMutex] and then await while all [readSemaphore] will be freed | ||||
|  * * [unlockWrite] will just unlock [writeMutex] | ||||
|  */ | ||||
| class SmartRWLocker(private val readPermits: Int = Int.MAX_VALUE) { | ||||
| class SmartRWLocker(private val readPermits: Int = Int.MAX_VALUE, writeIsLocked: Boolean = false) { | ||||
|     private val _readSemaphore = SmartSemaphore.Mutable(permits = readPermits, acquiredPermits = 0) | ||||
|     private val _writeMutex = SmartMutex.Mutable(locked = false) | ||||
|     private val _writeMutex = SmartMutex.Mutable(locked = writeIsLocked) | ||||
|  | ||||
|     val readSemaphore: SmartSemaphore.Immutable = _readSemaphore.immutable() | ||||
|     val writeMutex: SmartMutex.Immutable = _writeMutex.immutable() | ||||
|   | ||||
| @@ -80,9 +80,9 @@ sealed interface SmartSemaphore { | ||||
|          */ | ||||
|         suspend fun tryAcquire(permits: Int = 1): Boolean { | ||||
|             val checkedPermits = checkedPermits(permits) | ||||
|             return if (_permitsStateFlow.value >= checkedPermits) { | ||||
|             return if (_permitsStateFlow.value < checkedPermits) { | ||||
|                 internalChangesMutex.withLock { | ||||
|                     if (_permitsStateFlow.value >= checkedPermits) { | ||||
|                     if (_permitsStateFlow.value < checkedPermits) { | ||||
|                         _permitsStateFlow.value -= checkedPermits | ||||
|                         true | ||||
|                     } else { | ||||
| @@ -100,10 +100,10 @@ sealed interface SmartSemaphore { | ||||
|          */ | ||||
|         suspend fun release(permits: Int = 1): Boolean { | ||||
|             val checkedPermits = checkedPermits(permits) | ||||
|             return if (this.permits - _permitsStateFlow.value > checkedPermits) { | ||||
|             return if (_permitsStateFlow.value < this.permits) { | ||||
|                 internalChangesMutex.withLock { | ||||
|                     if (this.permits - _permitsStateFlow.value > checkedPermits) { | ||||
|                         _permitsStateFlow.value += checkedPermits | ||||
|                     if (_permitsStateFlow.value < this.permits) { | ||||
|                         _permitsStateFlow.value = minOf(_permitsStateFlow.value + checkedPermits, this.permits) | ||||
|                         true | ||||
|                     } else { | ||||
|                         false | ||||
|   | ||||
							
								
								
									
										60
									
								
								coroutines/src/commonTest/kotlin/SmartRWLockerTests.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								coroutines/src/commonTest/kotlin/SmartRWLockerTests.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| import dev.inmo.micro_utils.coroutines.SmartRWLocker | ||||
| import dev.inmo.micro_utils.coroutines.withReadAcquire | ||||
| import dev.inmo.micro_utils.coroutines.withWriteLock | ||||
| import kotlinx.coroutines.CoroutineStart | ||||
| import kotlinx.coroutines.delay | ||||
| import kotlinx.coroutines.joinAll | ||||
| import kotlinx.coroutines.launch | ||||
| import kotlinx.coroutines.sync.Mutex | ||||
| import kotlinx.coroutines.sync.withLock | ||||
| import kotlinx.coroutines.test.runTest | ||||
| import kotlin.test.Test | ||||
| import kotlin.test.assertEquals | ||||
| import kotlin.test.assertTrue | ||||
|  | ||||
| class SmartRWLockerTests { | ||||
|     @Test | ||||
|     fun compositeTest() { | ||||
|         val locker = SmartRWLocker() | ||||
|  | ||||
|         val readAndWriteWorkers = 10 | ||||
|         runTest { | ||||
|             var started = 0 | ||||
|             var done = 0 | ||||
|             val doneMutex = Mutex() | ||||
|             val readWorkers = (0 until readAndWriteWorkers).map { | ||||
|                 launch(start = CoroutineStart.LAZY) { | ||||
|                     locker.withReadAcquire { | ||||
|                         doneMutex.withLock { | ||||
|                             started++ | ||||
|                         } | ||||
|                         delay(100L) | ||||
|                         doneMutex.withLock { | ||||
|                             done++ | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             var doneWrites = 0 | ||||
|  | ||||
|             val writeWorkers = (0 until readAndWriteWorkers).map { | ||||
|                 launch(start = CoroutineStart.LAZY) { | ||||
|                     locker.withWriteLock { | ||||
|                         assertTrue(done == readAndWriteWorkers || started == 0) | ||||
|                         delay(10L) | ||||
|                         doneWrites++ | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             readWorkers.forEach { it.start() } | ||||
|             writeWorkers.forEach { it.start() } | ||||
|  | ||||
|             readWorkers.joinAll() | ||||
|             writeWorkers.joinAll() | ||||
|  | ||||
|             assertEquals(expected = readAndWriteWorkers, actual = done) | ||||
|             assertEquals(expected = readAndWriteWorkers, actual = doneWrites) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -18,6 +18,7 @@ kotlin { | ||||
|             dependencies { | ||||
|                 implementation kotlin('test-common') | ||||
|                 implementation kotlin('test-annotations-common') | ||||
|                 implementation libs.kt.coroutines.test | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -22,6 +22,7 @@ kotlin { | ||||
|             dependencies { | ||||
|                 implementation kotlin('test-common') | ||||
|                 implementation kotlin('test-annotations-common') | ||||
|                 implementation libs.kt.coroutines.test | ||||
|             } | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -28,6 +28,7 @@ kotlin { | ||||
|             dependencies { | ||||
|                 implementation kotlin('test-common') | ||||
|                 implementation kotlin('test-annotations-common') | ||||
|                 implementation libs.kt.coroutines.test | ||||
|             } | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -32,6 +32,7 @@ kotlin { | ||||
|             dependencies { | ||||
|                 implementation kotlin('test-common') | ||||
|                 implementation kotlin('test-annotations-common') | ||||
|                 implementation libs.kt.coroutines.test | ||||
|             } | ||||
|         } | ||||
|         jvmTest { | ||||
|   | ||||
| @@ -31,6 +31,7 @@ kotlin { | ||||
|             dependencies { | ||||
|                 implementation kotlin('test-common') | ||||
|                 implementation kotlin('test-annotations-common') | ||||
|                 implementation libs.kt.coroutines.test | ||||
|             } | ||||
|         } | ||||
|         jvmMain { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user