开源软件名称(OpenSource Name):andrewoma/kwery开源软件地址(OpenSource Url):https://github.com/andrewoma/kwery开源编程语言(OpenSource Language):Kotlin 98.0%开源软件介绍(OpenSource Introduction):Kwery OverviewKwery is an SQL library for Kotlin. Kwery consists of three major modules (core, mapper and fetcher) that when combined provide similar functionality to a traditional ORM. Kwery's manifesto:
CoreThe core module is a fairly thin wrapper over JDBC, providing support for named parameters, logging and transactions. class Actor(val firstName: String, val lastName: String, val lastUpdate: Timestamp)
val session = DefaultSession(connection, HsqlDialect()) // Standard JDBC connection
val sql = "select * from actor where first_name = :first_name"
val actors = session.select(sql, mapOf("first_name" to "Brad")) { row ->
Actor(row.string("first_name"), row.string("last_name"), row.timestamp("last_update"))
} MapperThe mapper module module builds on core to provide typical DAO (Data Access Object) functionality. As Kwery believes your domain model shouldn't be tainted by mapping annotations,
it uses a // We'll map to standard immutable classes, grouping name fields into a class
class Name(val firstName: String, val lastName: String)
class Actor(val id: Int, val name: Name, val lastUpdate: LocalDateTime)
// A table object defines the mapping between columns and models
// Conversions default to those defined in the configuration but may be overridden
object actorTable : Table<Actor, Int>("actor"), VersionedWithTimestamp {
val ActorId by col(Actor::id, id = true)
val FirstName by col(Name::firstName, Actor::name)
val LastName by col(Name::lastName, Actor::name)
val LastUpdate by col(Actor::lastUpdate, version = true)
override fun idColumns(id: Int) = setOf(ActorId of id)
override fun create(value: Value<Actor>) = Actor(value of ActorId,
Name(value of FirstName, value of LastName), value of LastUpdate)
}
// Given a table object, a generic dao is a one-liner, including standard CRUD operations
class ActorDao(session: Session) : AbstractDao<Actor, Int>(session, actorTable, Actor::id)
// Now we can use the DAO
val dao = ActorDao(session)
val inserted = dao.insert(Actor(1, Name("Kate", "Beckinsale"), LocalDateTime.now()))
val actors = dao.findAll() See Graph FetcherDAOs only fetch data from their linked table by default. To fetch an object graph, using a graph fetcher is the recommended method. Given a graph specification, the fetcher attempts to fetch the graph in the minimum number of queries possible. It does this by batching together requests for the same type into a single query. As it fetches by ids, it also provides an ideal mechanism to insert a cache layer. // Given the following domain model
data class Actor(val id: Int, val firstName: String, val lastName: String)
data class Language(val id: Int, val name: String)
data class Film(val id: Int, val language: Language, val actors: Set<Actor>,
val title: String, val releaseYear: Int)
// Define types with functions describing how to fetch a batch by ids
val language = Type(Language::id, { languageDao.findByIds(it) })
val actor = Type(Actor::id, { actorDao.findByIds(it) })
// For types that reference other types describe how to apply fetched values
val film = Type(Film::id, { filmDao.findByIds(it) }, listOf(
// 1 to 1
Property(Film::language, language, { it.language.id }, { f, l -> f.copy(language = l) }),
// 1 to many requires a function to describe how to fetch the related objects
CollectionProperty(Film::actors, actor, Film::id,
{ f, a -> f.copy(actors = a.toSet()) },
{ actorDao.findByFilmIds(it) })
))
val fetcher = GraphFetcher(setOf(language, actor, film))
// Extension function to fetch the graph for any List using fetcher defined above
fun <T> Collection<T>.fetch(node: Node) = fetcher.fetch(this, Node(node))
// We can now efficiently fetch various graphs for any list of films
// The following fetches the films with actors and languages in 3 queries
val filmsWithAll = filmDao.findFilmsReleasedAfter(2010).fetch(Node.all)
// The graph specification can also be built using properties
val filmsWithActors = filmDao.findFilmsReleasedAfter(2010).fetch(Film::actors.node()) DAOs and graph fetching aim to cover 95% of a typical application data retrievals. For the remaining performance critical sections, use specialised methods on the DAOs using partial selects and joins as required. ExampleThe example module demonstrates using Kwery to expose a simple model via RESTful web services via Dropwizard. TransactionalThe transactional module adds general purpose transaction interceptors. e.g. @Transactional open class MyService(val session: Session) {
open fun foo() {}
}
val session = ManagedThreadLocalSession(dataSource, HsqlDialect())
val service = transactionalFactory.fromClass(MyService(session), MyService::session)
service.foo() // Now calls to service automatically occur within a transaction See the readme for more information. Transactional for JerseyThe transactional-jersey module adds transaction annotations for Jersey. Registering Path("/films")
@Transactional class FilmResource : Resource {
GET fun find(): List<Film> {
...
}
} See the readme for more information. StatusKwery is unstable. It's currently being developed for a side project, so features are added as required. Kwery is available in Maven Central
Buildinggit clone https://github.com/andrewoma/kwery.git
cd kwery
./gradlew check install Note: The tests require a local postgres and mysql database named
To open in IntelliJ, just open the RoadmapCore:
DAO:
Fetcher:
Modules:
Robustness/Performance:
Misc:
LicenseThis project is licensed under a MIT license. |
2023-10-27
2022-08-15
2022-08-17
2022-09-23
2022-08-13
请发表评论