Create documentation
This commit is contained in:
@@ -5,3 +5,7 @@ _Kotlin library to request postgres with native SQL queries and return JSON_
|
|||||||
[](https://sonarcloud.io/dashboard?id=postgres-json)
|
[](https://sonarcloud.io/dashboard?id=postgres-json)
|
||||||
[](https://sonarcloud.io/dashboard?id=postgres-json)
|
[](https://sonarcloud.io/dashboard?id=postgres-json)
|
||||||
[](https://sonarcloud.io/dashboard?id=postgres-json)
|
[](https://sonarcloud.io/dashboard?id=postgres-json)
|
||||||
|
|
||||||
|
* [Installation](./docs/installation.md)
|
||||||
|
* [Migrations](./docs/migrations/migrations.md)
|
||||||
|
* [Usage](./docs/usage/usage.md)
|
||||||
|
|||||||
9
docs/installation.md
Normal file
9
docs/installation.md
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
# Installation
|
||||||
|
|
||||||
|
## Gradle
|
||||||
|
```kotlin
|
||||||
|
// build.gradle.kts
|
||||||
|
dependencies {
|
||||||
|
implementation("com.github.flecomte:postgres-json:+")
|
||||||
|
}
|
||||||
|
```
|
||||||
16
docs/migrations/migrations-application.md
Normal file
16
docs/migrations/migrations-application.md
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
# Execute migration in application
|
||||||
|
```kotlin
|
||||||
|
import fr.postgresjson.migration.Migrations
|
||||||
|
import fr.postgresjson.connexion.Connection
|
||||||
|
|
||||||
|
val conn: Connection = TODO()
|
||||||
|
val migrations = Migrations(
|
||||||
|
conn,
|
||||||
|
this::class.java.getResource("/sql/migrations")?.toURI() ?: error("No migrations found"),
|
||||||
|
this::class.java.getResource("/sql/functions")?.toURI() ?: error("No sql function found")
|
||||||
|
)
|
||||||
|
|
||||||
|
migrations.status() // Show executed and not executed migrations
|
||||||
|
migrations.runDry() // Execute migration in transaction and rollback at the end
|
||||||
|
migrations.run() // Execute migration in transaction and commit if no error
|
||||||
|
```
|
||||||
37
docs/migrations/migrations-gradle.md
Normal file
37
docs/migrations/migrations-gradle.md
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
# Execute Migrations with Gradle
|
||||||
|
|
||||||
|
You can execute migration with a Gradle task like this:
|
||||||
|
|
||||||
|
```kotlin
|
||||||
|
// build.gradle.kts
|
||||||
|
import fr.postgresjson.connexion.Connection
|
||||||
|
import fr.postgresjson.connexion.Requester
|
||||||
|
import fr.postgresjson.migration.Migrations
|
||||||
|
|
||||||
|
buildscript {
|
||||||
|
dependencies {
|
||||||
|
classpath("com.github.flecomte:postgres-json:+")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val migration by tasks.registering {
|
||||||
|
doLast {
|
||||||
|
val connection = Connection(
|
||||||
|
host = "localhost",
|
||||||
|
port = 5432,
|
||||||
|
database = "database",
|
||||||
|
username = "username",
|
||||||
|
password = "password"
|
||||||
|
)
|
||||||
|
Migrations(
|
||||||
|
connection,
|
||||||
|
file("$buildDir/resources/main/sql/migrations").toURI(),
|
||||||
|
file("$buildDir/resources/main/sql/functions").toURI()
|
||||||
|
).run()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
```shell
|
||||||
|
$ gradle migration
|
||||||
|
```
|
||||||
71
docs/migrations/migrations.md
Normal file
71
docs/migrations/migrations.md
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
# Migration
|
||||||
|
## Schemas migration
|
||||||
|
Migrations are just manually written `*.sql` files that represent the database schemas.
|
||||||
|
Each file is executed one after the other in alphabetical order.
|
||||||
|
Each execution is stored in the `migration.history` table.
|
||||||
|
|
||||||
|
A migration contains a `*.up.sql` file and a `*.down.sql` file to rollback the migration.
|
||||||
|
The content of the `*.down.sql` file is also stored in the database.
|
||||||
|
This allows the `*.down.sql` to be executed even if the code is already rollback.
|
||||||
|
|
||||||
|
Example:
|
||||||
|
```postgresql
|
||||||
|
-- resources/sql/migrations/0000-init_schema.up.sql
|
||||||
|
create table "user"
|
||||||
|
(
|
||||||
|
id uuid default uuid_generate_v4() not null primary key,
|
||||||
|
created_at timestamptz default now() not null,
|
||||||
|
blocked_at timestamptz default null null,
|
||||||
|
username varchar(64) not null check ( username != '' and lower(username) = username) unique,
|
||||||
|
password text not null check ( password != '' ),
|
||||||
|
roles text[] default '{}' not null
|
||||||
|
);
|
||||||
|
```
|
||||||
|
|
||||||
|
```postgresql
|
||||||
|
-- resources/sql/migrations/0000-init_schema.down.sql
|
||||||
|
drop table if exists "user";
|
||||||
|
```
|
||||||
|
## Stored procedure migrations
|
||||||
|
|
||||||
|
Migrations are also stored procedures and other functions.
|
||||||
|
Each function is updated with each migration.
|
||||||
|
|
||||||
|
Example:
|
||||||
|
```postgresql
|
||||||
|
-- resources/sql/functions/insert_user.sql
|
||||||
|
create or replace function insert_user(inout resource json) language plpgsql as
|
||||||
|
$$
|
||||||
|
declare
|
||||||
|
new_id uuid;
|
||||||
|
begin
|
||||||
|
insert into "user" (id, username, password, blocked_at, roles)
|
||||||
|
select
|
||||||
|
coalesce(t.id, uuid_generate_v4()),
|
||||||
|
t.username,
|
||||||
|
crypt(resource->>'password', gen_salt('bf', 8)),
|
||||||
|
case when t.blocked_at is not null then now() else null end,
|
||||||
|
t.roles
|
||||||
|
from json_populate_record(null::"user", resource) t
|
||||||
|
returning id into new_id;
|
||||||
|
|
||||||
|
select find_user_by_id(new_id) into resource;
|
||||||
|
end;
|
||||||
|
$$;
|
||||||
|
```
|
||||||
|
|
||||||
|
```postgresql
|
||||||
|
-- resources/sql/functions/find_user_by_id.sql
|
||||||
|
create or replace function find_user_by_id(in _id uuid, out resource json) language plpgsql as
|
||||||
|
$$
|
||||||
|
begin
|
||||||
|
select to_jsonb(u) - 'password' into resource
|
||||||
|
from "user" as u
|
||||||
|
where u.id = _id;
|
||||||
|
end;
|
||||||
|
$$;
|
||||||
|
```
|
||||||
|
|
||||||
|
* [Execute migrations in application](./migrations-application.md)
|
||||||
|
* [Execute migrations with gradle](./migrations-gradle.md)
|
||||||
|
|
||||||
14
docs/usage/init-connection.md
Normal file
14
docs/usage/init-connection.md
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
# Init connection
|
||||||
|
|
||||||
|
Before execute any query you must instantiate the connection.
|
||||||
|
```kotlin
|
||||||
|
import fr.postgresjson.connexion.Connection
|
||||||
|
|
||||||
|
val connection = Connection(
|
||||||
|
host = "localhost",
|
||||||
|
port = 5432,
|
||||||
|
database = "mydb",
|
||||||
|
username = "john",
|
||||||
|
password = "azerty"
|
||||||
|
)
|
||||||
|
```
|
||||||
46
docs/usage/raw-request.md
Normal file
46
docs/usage/raw-request.md
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
# Raw request
|
||||||
|
You can execute query directly from the code like this:
|
||||||
|
(*see [Init connection](./init-connection.md) before*)
|
||||||
|
|
||||||
|
```kotlin
|
||||||
|
import fr.postgresjson.connexion.Connection
|
||||||
|
|
||||||
|
val connection: Connection = TODO()
|
||||||
|
|
||||||
|
val result: QueryResult = connection.exec(
|
||||||
|
"SELECT id FROM inventor WHERE name = :name",
|
||||||
|
mapOf("name" to "Nikola Tesla")
|
||||||
|
)
|
||||||
|
val id: String = result.rows[0].getString(0)
|
||||||
|
```
|
||||||
|
|
||||||
|
And if you must map the query result with an entity, you can do it like this:
|
||||||
|
```kotlin
|
||||||
|
import java.util.UUID
|
||||||
|
import fr.postgresjson.entity.Serializable
|
||||||
|
import fr.postgresjson.connexion.Connection
|
||||||
|
|
||||||
|
val connection: Connection = TODO()
|
||||||
|
|
||||||
|
data class Inventor(
|
||||||
|
val id: UUID = UUID.randomUUID(),
|
||||||
|
val name: String
|
||||||
|
): Serializable
|
||||||
|
|
||||||
|
val result: Inventor? = connection.selectOne(
|
||||||
|
"""
|
||||||
|
SELECT json_build_object(
|
||||||
|
'id', '9e65de49-712e-47ce-8bf2-dfffae53a82e',
|
||||||
|
'name', :name
|
||||||
|
)
|
||||||
|
""",
|
||||||
|
mapOf("name" to "Nikola Tesla")
|
||||||
|
)
|
||||||
|
|
||||||
|
val inventor = connection.selectOne<Inventor>("SELECT * FROM mytable WHERE id = :id")
|
||||||
|
|
||||||
|
val inventors: List<Inventor> = connection.select("SELECT * FROM mytable WHERE status = 'done'")
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
See [ConnectionTest.kt](/src/test/kotlin/fr/postgresjson/ConnectionTest.kt) for more examples.
|
||||||
85
docs/usage/stored-procedure.md
Normal file
85
docs/usage/stored-procedure.md
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
# Stored Procedure
|
||||||
|
*Execute stored procedure with requester*
|
||||||
|
|
||||||
|
You can execute a stored procedure (previously defined in a migration) via the Requester
|
||||||
|
|
||||||
|
To do that:
|
||||||
|
|
||||||
|
1. First, instantiate the requester
|
||||||
|
```kotlin
|
||||||
|
import fr.postgresjson.connexion.Requester
|
||||||
|
import fr.postgresjson.connexion.Connection
|
||||||
|
|
||||||
|
val connection: Connection = TODO()
|
||||||
|
|
||||||
|
val requester = Requester.RequesterFactory(
|
||||||
|
connection = connection,
|
||||||
|
functionsDirectory = this::class.java.getResource("/sql/functions")?.toURI() ?: error("No sql function found")
|
||||||
|
).createRequester()
|
||||||
|
```
|
||||||
|
|
||||||
|
2. then, define Entities
|
||||||
|
```kotlin
|
||||||
|
import java.util.UUID
|
||||||
|
import org.joda.time.DateTime
|
||||||
|
import fr.postgresjson.entity.Serializable
|
||||||
|
|
||||||
|
enum class Roles { ROLE_USER, ROLE_ADMIN }
|
||||||
|
|
||||||
|
class User(
|
||||||
|
id: UUID = UUID.randomUUID(),
|
||||||
|
override var username: String,
|
||||||
|
var blockedAt: DateTime? = null,
|
||||||
|
var roles: List<Roles> = emptyList()
|
||||||
|
): Serializable
|
||||||
|
|
||||||
|
class UserForCreate(
|
||||||
|
id: UUID = UUID.randomUUID(),
|
||||||
|
username: String,
|
||||||
|
val password: String,
|
||||||
|
blockedAt: DateTime? = null,
|
||||||
|
roles: List<Roles> = emptyList()
|
||||||
|
): Serializable
|
||||||
|
```
|
||||||
|
3. and, define Repositories
|
||||||
|
*[See SQL function](./migrations.md#Stored procedure migrations)*
|
||||||
|
|
||||||
|
```kotlin
|
||||||
|
import fr.postgresjson.connexion.Requester
|
||||||
|
import fr.postgresjson.repository.RepositoryI
|
||||||
|
import java.util.UUID
|
||||||
|
|
||||||
|
class UserRepository(override var requester: Requester): RepositoryI {
|
||||||
|
fun findById(id: UUID): User {
|
||||||
|
return requester
|
||||||
|
.getFunction("find_user_by_id") // Use the name of the function
|
||||||
|
.selectOne(
|
||||||
|
"id" to id // You can pass parameters by their names. The underscore prefix on parameters is not required to be mapped.
|
||||||
|
) ?: throw UserNotFound(id) // Throw exception if user not found
|
||||||
|
}
|
||||||
|
|
||||||
|
fun insert(user: UserForCreate): User {
|
||||||
|
return requester
|
||||||
|
.getFunction("insert_user")
|
||||||
|
.selectOne("resource" to user)
|
||||||
|
}
|
||||||
|
|
||||||
|
class UserNotFound(override val message: String?, override val cause: Throwable?): Throwable(message, cause) {
|
||||||
|
constructor(id: UUID): this("No User with ID $id", null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
4. And at last, execute queries
|
||||||
|
```kotlin
|
||||||
|
import fr.postgresjson.connexion.Requester
|
||||||
|
import java.util.UUID
|
||||||
|
|
||||||
|
val requester: Requester = TODO()
|
||||||
|
val userRepo = UserRepository(requester)
|
||||||
|
|
||||||
|
val user: User = userRepo.findById(UUID.fromString(id))
|
||||||
|
|
||||||
|
val newUser: UserForCreate = TODO()
|
||||||
|
val userInserted: User = userRepo.insert(newUser)
|
||||||
|
```
|
||||||
5
docs/usage/usage.md
Normal file
5
docs/usage/usage.md
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
## Usage
|
||||||
|
|
||||||
|
1. [Init connection](./init-connection.md)
|
||||||
|
2. [Raw request](./raw-request.md)
|
||||||
|
3. [Stored Procedure](./stored-procedure.md)
|
||||||
@@ -25,7 +25,7 @@ interface EmbedExecutable {
|
|||||||
block: SelectOneCallback<R> = {}
|
block: SelectOneCallback<R> = {}
|
||||||
): R?
|
): R?
|
||||||
|
|
||||||
/* Select Miltiples */
|
/* Select Multiples */
|
||||||
fun <R : EntityI> select(
|
fun <R : EntityI> select(
|
||||||
typeReference: TypeReference<List<R>>,
|
typeReference: TypeReference<List<R>>,
|
||||||
values: List<Any?> = emptyList(),
|
values: List<Any?> = emptyList(),
|
||||||
|
|||||||
Reference in New Issue
Block a user