mirror of
				https://github.com/InsanusMokrassar/MicroUtils.git
				synced 2025-10-26 17:50:41 +00:00 
			
		
		
		
	Compare commits
	
		
			16 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6a5ad4d728 | |||
| be4aa8daac | |||
| b5eac37782 | |||
| b1ad3c5a39 | |||
| ba16bad029 | |||
| ca8ae4cd72 | |||
| 53d35d74b3 | |||
| 49c139e235 | |||
| caf9c821f3 | |||
| ca4c6db96f | |||
| 6b2298c752 | |||
| a1bf43def9 | |||
| 15e9254e00 | |||
| afe5a72c6f | |||
| 750a8b9ecf | |||
| 27fc3f93e0 | 
							
								
								
									
										25
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,5 +1,30 @@ | |||||||
| # Changelog | # Changelog | ||||||
|  |  | ||||||
|  | ## 0.9.23 | ||||||
|  |  | ||||||
|  | * `Repos`: | ||||||
|  |     * `Exposed`: | ||||||
|  |         * New property `ExposedRepo#selectAll` to retrieve all the rows in the table | ||||||
|  |  | ||||||
|  | ## 0.9.22 | ||||||
|  |  | ||||||
|  | * `Ktor`: | ||||||
|  |     * `Server`: | ||||||
|  |         * Now `createKtorServer` fun is fully customizable | ||||||
|  |  | ||||||
|  | ## 0.9.21 | ||||||
|  |  | ||||||
|  | * `Repos`: | ||||||
|  |     * `Exposed`: | ||||||
|  |         * fixes in `AbstractExposedWriteCRUDRepo` | ||||||
|  |  | ||||||
|  | ## 0.9.20 | ||||||
|  |  | ||||||
|  | * `Repos`: | ||||||
|  |     * `Common`: | ||||||
|  |         * Fixes in `OneToManyAndroidRepo` | ||||||
|  |         * New `CursorIterator` | ||||||
|  |  | ||||||
| ## 0.9.19 | ## 0.9.19 | ||||||
|  |  | ||||||
| * `Versions`: | * `Versions`: | ||||||
|   | |||||||
| @@ -14,5 +14,5 @@ crypto_js_version=4.1.1 | |||||||
| # Project data | # Project data | ||||||
|  |  | ||||||
| group=dev.inmo | group=dev.inmo | ||||||
| version=0.9.19 | version=0.9.23 | ||||||
| android_code_version=109 | android_code_version=113 | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ package dev.inmo.micro_utils.ktor.server | |||||||
| import dev.inmo.micro_utils.ktor.server.configurators.KtorApplicationConfigurator | import dev.inmo.micro_utils.ktor.server.configurators.KtorApplicationConfigurator | ||||||
| import io.ktor.application.Application | import io.ktor.application.Application | ||||||
| import io.ktor.server.cio.CIO | import io.ktor.server.cio.CIO | ||||||
|  | import io.ktor.server.cio.CIOApplicationEngine | ||||||
| import io.ktor.server.engine.* | import io.ktor.server.engine.* | ||||||
| import kotlin.random.Random | import kotlin.random.Random | ||||||
|  |  | ||||||
| @@ -10,17 +11,21 @@ fun <TEngine : ApplicationEngine, TConfiguration : ApplicationEngine.Configurati | |||||||
|     engine: ApplicationEngineFactory<TEngine, TConfiguration>, |     engine: ApplicationEngineFactory<TEngine, TConfiguration>, | ||||||
|     host: String = "localhost", |     host: String = "localhost", | ||||||
|     port: Int = Random.nextInt(1024, 65535), |     port: Int = Random.nextInt(1024, 65535), | ||||||
|  |     additionalEngineEnvironmentConfigurator: ApplicationEngineEnvironmentBuilder.() -> Unit = {}, | ||||||
|  |     additionalConfigurationConfigurator: TConfiguration.() -> Unit = {}, | ||||||
|     block: Application.() -> Unit |     block: Application.() -> Unit | ||||||
| ): TEngine { | ): TEngine = embeddedServer( | ||||||
|     val env = applicationEngineEnvironment { |     engine, | ||||||
|  |     applicationEngineEnvironment { | ||||||
|         module(block) |         module(block) | ||||||
|         connector { |         connector { | ||||||
|             this@connector.host = host |             this.host = host | ||||||
|             this@connector.port = port |             this.port = port | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     return embeddedServer(engine, env) |  | ||||||
|         } |         } | ||||||
|  |         additionalEngineEnvironmentConfigurator() | ||||||
|  |     }, | ||||||
|  |     additionalConfigurationConfigurator | ||||||
|  | ) | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Create server with [CIO] server engine without starting of it |  * Create server with [CIO] server engine without starting of it | ||||||
| @@ -30,18 +35,31 @@ fun <TEngine : ApplicationEngine, TConfiguration : ApplicationEngine.Configurati | |||||||
| fun createKtorServer( | fun createKtorServer( | ||||||
|     host: String = "localhost", |     host: String = "localhost", | ||||||
|     port: Int = Random.nextInt(1024, 65535), |     port: Int = Random.nextInt(1024, 65535), | ||||||
|  |     additionalEngineEnvironmentConfigurator: ApplicationEngineEnvironmentBuilder.() -> Unit = {}, | ||||||
|  |     additionalConfigurationConfigurator: CIOApplicationEngine.Configuration.() -> Unit = {}, | ||||||
|     block: Application.() -> Unit |     block: Application.() -> Unit | ||||||
| ): ApplicationEngine = createKtorServer(CIO, host, port, block) | ): CIOApplicationEngine = createKtorServer( | ||||||
|  |     CIO, | ||||||
|  |     host, | ||||||
|  |     port, | ||||||
|  |     additionalEngineEnvironmentConfigurator, | ||||||
|  |     additionalConfigurationConfigurator, | ||||||
|  |     block | ||||||
|  | ) | ||||||
|  |  | ||||||
| fun <TEngine : ApplicationEngine, TConfiguration : ApplicationEngine.Configuration> createKtorServer( | fun <TEngine : ApplicationEngine, TConfiguration : ApplicationEngine.Configuration> createKtorServer( | ||||||
|     engine: ApplicationEngineFactory<TEngine, TConfiguration>, |     engine: ApplicationEngineFactory<TEngine, TConfiguration>, | ||||||
|     host: String = "localhost", |     host: String = "localhost", | ||||||
|     port: Int = Random.nextInt(1024, 65535), |     port: Int = Random.nextInt(1024, 65535), | ||||||
|  |     additionalEngineEnvironmentConfigurator: ApplicationEngineEnvironmentBuilder.() -> Unit = {}, | ||||||
|  |     additionalConfigurationConfigurator: TConfiguration.() -> Unit = {}, | ||||||
|     configurators: List<KtorApplicationConfigurator> |     configurators: List<KtorApplicationConfigurator> | ||||||
| ): TEngine = createKtorServer( | ): TEngine = createKtorServer( | ||||||
|     engine, |     engine, | ||||||
|     host, |     host, | ||||||
|     port |     port, | ||||||
|  |     additionalEngineEnvironmentConfigurator, | ||||||
|  |     additionalConfigurationConfigurator | ||||||
| ) { | ) { | ||||||
|     configurators.forEach { it.apply { configure() } } |     configurators.forEach { it.apply { configure() } } | ||||||
| } | } | ||||||
| @@ -54,5 +72,7 @@ fun <TEngine : ApplicationEngine, TConfiguration : ApplicationEngine.Configurati | |||||||
| fun createKtorServer( | fun createKtorServer( | ||||||
|     host: String = "localhost", |     host: String = "localhost", | ||||||
|     port: Int = Random.nextInt(1024, 65535), |     port: Int = Random.nextInt(1024, 65535), | ||||||
|     configurators: List<KtorApplicationConfigurator> |     configurators: List<KtorApplicationConfigurator>, | ||||||
| ): ApplicationEngine = createKtorServer(CIO, host, port, configurators) |     additionalEngineEnvironmentConfigurator: ApplicationEngineEnvironmentBuilder.() -> Unit = {}, | ||||||
|  |     additionalConfigurationConfigurator: CIOApplicationEngine.Configuration.() -> Unit = {}, | ||||||
|  | ): ApplicationEngine = createKtorServer(CIO, host, port, additionalEngineEnvironmentConfigurator, additionalConfigurationConfigurator, configurators) | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ import dev.inmo.micro_utils.pagination.* | |||||||
|  * Example: |  * Example: | ||||||
|  * |  * | ||||||
|  * * `|__f__l_______________________|` will be transformed to `|_______________________f__l__|` |  * * `|__f__l_______________________|` will be transformed to `|_______________________f__l__|` | ||||||
|  * * `|__f__l_|` will be transformed to `|__f__l_|` |  * * `|__f__l_|` will be transformed to `|_f__l__|` | ||||||
|  * |  * | ||||||
|  * @return Reversed version of this [Pagination] |  * @return Reversed version of this [Pagination] | ||||||
|  */ |  */ | ||||||
|   | |||||||
| @@ -0,0 +1,27 @@ | |||||||
|  | package dev.inmo.micro_utils.repos | ||||||
|  |  | ||||||
|  | import android.database.Cursor | ||||||
|  |  | ||||||
|  | class CursorIterator( | ||||||
|  |     private val c: Cursor | ||||||
|  | ) : Iterator<Cursor> { | ||||||
|  |     private var i = 0 | ||||||
|  |  | ||||||
|  |     init { | ||||||
|  |         c.moveToFirst() | ||||||
|  |     } | ||||||
|  |     override fun hasNext(): Boolean { | ||||||
|  |         return i < c.count | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun next(): Cursor { | ||||||
|  |         i++ | ||||||
|  |         return if (c.moveToNext()) { | ||||||
|  |             c | ||||||
|  |         } else { | ||||||
|  |             throw NoSuchElementException() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | operator fun Cursor.iterator(): CursorIterator = CursorIterator(this) | ||||||
| @@ -143,7 +143,12 @@ class OneToManyAndroidRepo<Key, Value>( | |||||||
|     }.toLong() |     }.toLong() | ||||||
|  |  | ||||||
|     override suspend fun count(k: Key): Long = helper.blockingReadableTransaction { |     override suspend fun count(k: Key): Long = helper.blockingReadableTransaction { | ||||||
|         selectDistinct(tableName, columns = valueColumnArray, selection = "$idColumnName=?", selectionArgs = arrayOf(k.keyAsString()), limit = FirstPagePagination(1).limitClause()).use { |         selectDistinct( | ||||||
|  |             tableName, | ||||||
|  |             columns = valueColumnArray, | ||||||
|  |             selection = "$idColumnName=?", | ||||||
|  |             selectionArgs = arrayOf(k.keyAsString()) | ||||||
|  |         ).use { | ||||||
|             it.count |             it.count | ||||||
|         } |         } | ||||||
|     }.toLong() |     }.toLong() | ||||||
|   | |||||||
| @@ -17,13 +17,19 @@ abstract class AbstractExposedWriteCRUDRepo<ObjectType, IdType, InputValueType>( | |||||||
|     ExposedCRUDRepo<ObjectType, IdType>, |     ExposedCRUDRepo<ObjectType, IdType>, | ||||||
|     WriteStandardCRUDRepo<ObjectType, IdType, InputValueType> |     WriteStandardCRUDRepo<ObjectType, IdType, InputValueType> | ||||||
| { | { | ||||||
|     protected val newObjectsChannel = MutableSharedFlow<ObjectType>(replyCacheInFlows, flowsChannelsSize) |     protected val _newObjectsFlow = MutableSharedFlow<ObjectType>(replyCacheInFlows, flowsChannelsSize) | ||||||
|     protected val updateObjectsChannel = MutableSharedFlow<ObjectType>(replyCacheInFlows, flowsChannelsSize) |     protected val _updatedObjectsFlow = MutableSharedFlow<ObjectType>(replyCacheInFlows, flowsChannelsSize) | ||||||
|     protected val deleteObjectsIdsChannel = MutableSharedFlow<IdType>(replyCacheInFlows, flowsChannelsSize) |     protected val _deletedObjectsIdsFlow = MutableSharedFlow<IdType>(replyCacheInFlows, flowsChannelsSize) | ||||||
|  |     @Deprecated("Renamed", ReplaceWith("_newObjectsFlow")) | ||||||
|  |     protected val newObjectsChannel = _newObjectsFlow | ||||||
|  |     @Deprecated("Renamed", ReplaceWith("_updatedObjectsFlow")) | ||||||
|  |     protected val updateObjectsChannel = _updatedObjectsFlow | ||||||
|  |     @Deprecated("Renamed", ReplaceWith("_deletedObjectsIdsFlow")) | ||||||
|  |     protected val deleteObjectsIdsChannel = _deletedObjectsIdsFlow | ||||||
|  |  | ||||||
|     override val newObjectsFlow: Flow<ObjectType> = newObjectsChannel.asSharedFlow() |     override val newObjectsFlow: Flow<ObjectType> = _newObjectsFlow.asSharedFlow() | ||||||
|     override val updatedObjectsFlow: Flow<ObjectType> = updateObjectsChannel.asSharedFlow() |     override val updatedObjectsFlow: Flow<ObjectType> = _updatedObjectsFlow.asSharedFlow() | ||||||
|     override val deletedObjectsIdsFlow: Flow<IdType> = deleteObjectsIdsChannel.asSharedFlow() |     override val deletedObjectsIdsFlow: Flow<IdType> = _deletedObjectsIdsFlow.asSharedFlow() | ||||||
|  |  | ||||||
|     protected abstract fun InsertStatement<Number>.asObject(value: InputValueType): ObjectType |     protected abstract fun InsertStatement<Number>.asObject(value: InputValueType): ObjectType | ||||||
|     abstract val selectByIds: SqlExpressionBuilder.(List<IdType>) -> Op<Boolean> |     abstract val selectByIds: SqlExpressionBuilder.(List<IdType>) -> Op<Boolean> | ||||||
| @@ -43,7 +49,7 @@ abstract class AbstractExposedWriteCRUDRepo<ObjectType, IdType, InputValueType>( | |||||||
|         return transaction(db = database) { |         return transaction(db = database) { | ||||||
|             values.map { value -> createWithoutNotification(value) } |             values.map { value -> createWithoutNotification(value) } | ||||||
|         }.onEach { |         }.onEach { | ||||||
|             newObjectsChannel.emit(it) |             _newObjectsFlow.emit(it) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -74,7 +80,7 @@ abstract class AbstractExposedWriteCRUDRepo<ObjectType, IdType, InputValueType>( | |||||||
|         onBeforeUpdate(listOf(id to value)) |         onBeforeUpdate(listOf(id to value)) | ||||||
|         return updateWithoutNotification(id, value).also { |         return updateWithoutNotification(id, value).also { | ||||||
|             if (it != null) { |             if (it != null) { | ||||||
|                 updateObjectsChannel.emit(it) |                 _updatedObjectsFlow.emit(it) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -85,16 +91,25 @@ abstract class AbstractExposedWriteCRUDRepo<ObjectType, IdType, InputValueType>( | |||||||
|                 values.map { (id, value) -> updateWithoutNotification(id, value) } |                 values.map { (id, value) -> updateWithoutNotification(id, value) } | ||||||
|             }.filterNotNull() |             }.filterNotNull() | ||||||
|         ).onEach { |         ).onEach { | ||||||
|             updateObjectsChannel.emit(it) |             _updatedObjectsFlow.emit(it) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     protected open suspend fun onBeforeDelete(ids: List<IdType>) {} |     protected open suspend fun onBeforeDelete(ids: List<IdType>) {} | ||||||
|     override suspend fun deleteById(ids: List<IdType>) { |     override suspend fun deleteById(ids: List<IdType>) { | ||||||
|         onBeforeDelete(ids) |         onBeforeDelete(ids) | ||||||
|         transaction(db = database) { |         transaction(db = database) { | ||||||
|             deleteWhere(null, null) { |             val deleted = deleteWhere(null, null) { | ||||||
|                 selectByIds(ids) |                 selectByIds(ids) | ||||||
|             } |             } | ||||||
|  |             if (deleted == ids.size) { | ||||||
|  |                 ids | ||||||
|  |             } else { | ||||||
|  |                 ids.filter { | ||||||
|  |                     select { selectById(it) }.limit(1).none() | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }.forEach { | ||||||
|  |             _deletedObjectsIdsFlow.emit(it) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,8 +1,10 @@ | |||||||
| package dev.inmo.micro_utils.repos.exposed | package dev.inmo.micro_utils.repos.exposed | ||||||
|  |  | ||||||
| import dev.inmo.micro_utils.repos.Repo | import dev.inmo.micro_utils.repos.Repo | ||||||
| import org.jetbrains.exposed.sql.Database | import org.jetbrains.exposed.sql.* | ||||||
|  |  | ||||||
| interface ExposedRepo : Repo { | interface ExposedRepo : Repo, FieldSet { | ||||||
|     val database: Database |     val database: Database | ||||||
|  |     val selectAll: Transaction.() -> Query | ||||||
|  |         get() = { (this@ExposedRepo as FieldSet).selectAll() } | ||||||
| } | } | ||||||
		Reference in New Issue
	
	Block a user