Then we move to CRUD
- Create/Update/Delete Operation, these operations is common when you process business.
Demo projects:
- Standalone - 6093 :
up-thea
package up.god.micro.tabular;
import io.vertx.core.json.JsonObject;
import io.vertx.up.annotations.Address;
import io.vertx.up.annotations.Codex;
import io.vertx.up.annotations.EndPoint;
import javax.ws.rs.*;
@EndPoint
@Path("/api")
public interface TabularApi {
@Path("tabular")
@POST
@Address("ZERO://QUEUE/TABULAR/CREATE")
JsonObject create(@BodyParam @Codex JsonObject data);
@Path("tabular/{id}")
@PUT
@Address("ZERO://QUEUE/TABULAR/UPDATE")
JsonObject update(@PathParam("id") Long id, @BodyParam @Codex JsonObject data);
@Path("tabular/{id}")
@DELETE
@Address("ZERO://QUEUE/TABULAR/DELETE")
JsonObject delete(@PathParam("id") Long id);
}
Above three apis are mapping to insert, update, delete
operations.
Here we add methods to original consumer code as following:
package up.god.micro.tabular;
import io.vertx.core.Future;
import io.vertx.core.json.JsonObject;
import io.vertx.up.unity.Ux;
import io.vertx.up.annotations.Address;
import io.vertx.up.annotations.Queue;
import io.vertx.up.commune.Envelop;
import jakarta.inject.Inject;
@Queue
public class TabularWorker {
@Inject
private transient TabularStub stub;
@Address("ZERO://QUEUE/TABULAR/ID")
Future<JsonObject> get(final Envelop envelop) {
final Long id = Ux.getLong(envelop);
// First version
return this.stub.fetchOne(id);
}
// New added three methods for writing operations.
@Address("ZERO://QUEUE/TABULAR/CREATE")
Future<JsonObject> create(final Envelop envelop) {
final JsonObject data = Ux.getJson(envelop);
return this.stub.create(data);
}
@Address("ZERO://QUEUE/TABULAR/UPDATE")
Future<JsonObject> update(final Envelop envelop) {
final Long id = Ux.getLong(envelop);
final JsonObject data = Ux.getJson1(envelop);
return this.stub.update(id, data);
}
@Address("ZERO://QUEUE/TABULAR/DELETE")
Future<JsonObject> delete(final Envelop envelop) {
final Long id = Ux.getLong(envelop);
return this.stub.delete(id);
}
}
The definition of service interface as following:
package up.god.micro.tabular;
import io.vertx.core.Future;
import io.vertx.core.json.JsonObject;
public interface TabularStub {
Future<JsonObject> fetchOne(Long id);
// New added three apis for writing operations.
Future<JsonObject> create(JsonObject data);
Future<JsonObject> update(Long id, JsonObject data);
Future<JsonObject> delete(Long id);
}
Here we need to mention is that we'll split different code logical for insert, update, delete
to explain the usage
of Ux.Jooq
class.
@Override
public Future<JsonObject> create(final JsonObject data) {
final SysTabular tabular = Ux.fromJson(data, SysTabular.class, "tabular");
tabular.setZCreateTime(LocalDateTime.now());
tabular.setZUpdateTime(LocalDateTime.now());
return Ux.Jooq.on(SysTabularDao.class)
.insertReturningPrimaryAsync(tabular, tabular::setPkId)
.compose(item -> Ux.thenJsonOne(item, "tabular"));
}
Here the first part is the same that we called Ux.Jooq.on
api to mount our client to SysTabularDao
dao class, then
we called insertReturningPrimaryAsync
method for object tabular, this method will help you to set auto generated
primary key, here we need to mention:
- All the generated key is
Long
type instead ofInteger
. - The generated key will be passed into
tabular::setPkId
method to consume instead you use it.
There is another method insertAsync
that you could use is that you do not care the returned generated key, it just
like following:
return Ux.Jooq.on(SysTabularDao.class).insertAsync(tabular)
.compose(item -> Ux.thenJsonOne(item, "tabular"));
In zero system here provide three apis for Jooq creating:
insertAsync(T)
insertAsync(List<T>)
insertReturningPrimaryAsync(T, Consumer<Long>)
The second method that we'll introduce is update operation
@Override
public Future<JsonObject> update(final Long id, final JsonObject data) {
return Ux.Jooq.on(SysTabularDao.class).findByIdAsync(id)
.compose(item -> Ux.thenJsonOne(item, "tabular"))
.compose(item -> Future.succeededFuture(item.mergeIn(data)))
.compose(item -> Future.succeededFuture(Ux.fromJson(item, SysTabular.class, "tabular")))
.compose(item -> Ux.Jooq.on(SysTabularDao.class).updateAsync(item))
.compose(item -> Ux.thenJsonOne(item, "tabular"));
}
Above code described how to update the data in database, please be careful about the code logical:
- Get the entity
SysTabular
from database by id first; - Convert the data object into JsonObject;
- Merged JsonObject with updated json data;
- Convert the updated JsonObject data back to
SysTabular
; - Update the
SysTabular
data object and returned the updated data object.
It's a little complex and not very well, in this situation we recommend to use another method of Ux.Jooq
as following:
@Override
public Future<JsonObject> update(final Long id, final JsonObject data) {
final SysTabular updated = Ux.fromJson(data, SysTabular.class, "tabular");
return Ux.Jooq.on(SysTabularDao.class).saveAsync(id, updated)
.compose(item -> Ux.thenJsonOne(item, "tabular"));
}
Here we called saveAsync
with id
and updated
data object, then you can update the data here. When you send
request, you can see following results:
URL : http://localhost:6093/api/tabular/172
Method : PUT
Request :
{
"key": 172,
"name": "Lang3",
"active": true,
"sigma": "mock",
"language": "cn",
"code": "LANG.CODEEX",
"type": "testx.type",
"order": 3
}
Response :
{
"data": {
"key": 172,
"active": true,
"createTime": "2018-02-10T13:50:44",
"zupdateTime": "2018-02-10T13:50:44",
"type": "testx.type",
"code": "LANG.CODEX",
"name": "Lang3",
"order": 3,
"sigma": "mock",
"language": "cn"
}
}
Although we provide update apis here, but the save
apis are more useful, here are all the apis of Ux.Jooq
:
<T> Future<T> updateAsync(T entity)
<T> Future<List<T>> updateAsync(List<T> entities)
<T> Future<T> saveAsync(Object id, T updated)
<T> Future<T> saveAsync(Object id, Function<T,T> copyFun)
The last write operation api of Jooq is delete
, you can do as following:
@Override
public Future<JsonObject> delete(final Long id) {
return Ux.Jooq.on(SysTabularDao.class).deleteByIdAsync(id)
.compose(result -> (result) ?
Future.succeededFuture(new JsonObject().put("result", Boolean.TRUE)) :
Future.succeededFuture(new JsonObject().put("result", Boolean.FALSE))
);
}
Here you can delete record from database.
URL : http://localhost:6093/api/tabular/172
Method : DELETE
Response :
{
"data": {
"result": true
}
}
Here we listed all the delete api that Ux.Jooq
provided:
<T> Future<T> deleteAsync(T entity)
<T> Future<Boolean> deleteByIdAsync(Object id)
<T> Future<Boolean> deleteByIdAsync(Collection<Object> ids)
<T> Future<Boolean> deleteByIdAsync(Object... ids)
Until now you have known how to insert, update, delete
data by Jooq, here we have finished all the CRUD operation
tutorials, from next chapter we'll discuss some advanced Jooq features for real business project usage such as
pagination, sorting, searching etc.