we need to establish a JDBC database connection, and also make sure that the database schema is in place, and
we need to start a HTTP server for the web application.
Each phase can fail (e.g., the HTTP server TCP port is already being used), and they should not run in parallel as the web application code first needs the database access to work.
To make our code cleaner we will define 1 method per phase, and adopt a pattern of returning a future object to notify when each of the phases completes, and whether it did so successfully or not:
By having each method returning a future object, the implementation of the method becomes a composition:
@Override
public void start(Promise<Void> promise) throws Exception {
Future<Void> steps = prepareDatabase().compose(v -> startHttpServer());
steps.setHandler(promise);
}
When the future of prepareDatabase
completes successfully, then startHttpServer
is called and the steps
future completes depending of the outcome of the future returned by startHttpServer
. startHttpServer
is never called if prepareDatabase
encounters an error, in which case the steps
future is in a failed state and becomes completed with the exception describing the error.
Eventually steps
completes: setHandler
defines a handler to be called upon completion. In our case we simply want to complete startFuture
with steps
and use the completer
method to obtain a handler. This is equivalent to:
Future<Void> steps = prepareDatabase().compose(v -> startHttpServer());
steps.setHandler(ar -> { (1)
if (ar.succeeded()) {
promise.complete();
} else {
promise.fail(ar.cause());
}
});
ar
is of typeAsyncResult<Void>
.AsyncResult<T>
is used to pass the result of an asynchronous processing and may either yield a value of typeT
on success or a failure exception if the processing failed.
Database initialization
The wiki database schema consists of a single table Pages
with the following columns:
The database operations will be typical create, read, update, delete operations. To get us started, we simply store the corresponding SQL queries as static fields of the MainVerticle
class. Note that they are written in a SQL dialect that HSQLDB understands, but that other relational databases may not necessarily support:
- The
?
in the queries are placeholders to pass data when executing queries, and the Vert.x JDBC client prevents from SQL injections.
The application verticle needs to keep a reference to a JDBCClient
object (from the io.vertx.ext.jdbc
package) that serves as the connection to the database. We do so using a field in MainVerticle
, and we also create a general-purpose logger from the org.slf4j
package:
private JDBCClient dbClient;
private static final Logger LOGGER = LoggerFactory.getLogger(MainVerticle.class);
Last but not least, here is the complete implementation of the prepareDatabase
method. It attempts to obtain a JDBC client connection, then performs a SQL query to create the Pages
table unless it already exists:
private Future<Void> prepareDatabase() {
Promise<Void> promise = Promise.promise();
dbClient = JDBCClient.createShared(vertx, new JsonObject() (1)
.put("url", "jdbc:hsqldb:file:db/wiki") (2)
.put("max_pool_size", 30)); (4)
dbClient.getConnection(ar -> { (5)
if (ar.failed()) {
LOGGER.error("Could not open a database connection", ar.cause());
promise.fail(ar.cause()); (6)
} else {
SQLConnection connection = ar.result(); (7)
connection.close(); (8)
if (create.failed()) {
LOGGER.error("Database preparation error", create.cause());
promise.fail(create.cause());
} else {
promise.complete(); (9)
}
});
}
});
return promise.future();
}
createShared
creates a shared connection to be shared among verticles known to thevertx
instance, which in general is a good thing.The JDBC client connection is made by passing a Vert.x JSON object. Here
url
is the JDBC URL.Just like
url
,driver_class
is specific to the JDBC driver being used and points to the driver class.max_pool_size
is the number of concurrent connections. We chose 30 here, but it is just an arbitrary number.Getting a connection is an asynchronous operation that gives us an
AsyncResult<SQLConnection>
. It must then be tested to see if the connection could be established or not (AsyncResult
is actually a super-interface ofFuture
).If the SQL connection could not be obtained, then the method future is completed to fail with the
AsyncResult
-provided exception via thecause
method.The
SQLConnection
is the result of the successfulAsyncResult
. We can use it to perform a SQL query.We complete the method future object with a success.
Tip | The SQL database modules supported by the Vert.x project do not currently offer anything beyond passing SQL queries (e.g., an object-relational mapper) as they focus on providing asynchronous access to databases. However, nothing forbids using more advanced modules from the community, and we especially recommend checking out projects like or this POJO mapper. |
Notes about logging
The previous subsection introduced a logger, and we opted for the SLF4J library. Vert.x is also unopinionated on logging: you can choose any popular Java logging library. We recommend SLF4J since it is a popular logging abstraction and unification library in the Java ecosystem.
We also recommend using as a logger implementation. Integrating both SLF4J and Logback can be done by adding two dependencies, or just logback-classic
that points to both libraries (incidentally they are from the same author):
By default SLF4J outputs many log events to the console from Vert.x, Netty, C3PO and the wiki application. We can reduce the verbosity by adding the a src/main/resources/logback.xml
configuration file (see https://logback.qos.ch/ for more details):
<configuration>
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
</encoder>
</appender>
<logger name="com.mchange.v2" level="warn"/>
<logger name="io.netty" level="warn"/>
<logger name="io.vertx" level="info"/>
<root level="debug">
<appender-ref ref="STDOUT"/>
</configuration>
Last but not least HSQLDB does not integrate well with loggers when embedded. By default it tries to reconfigure the logging system in place, so we need to disable it by passing a -Dhsqldb.reconfig_logging=false
property to the Java Virtual Machine when executing applications.
HTTP server initialization
The HTTP server makes use of the vertx-web
project to easily define dispatching routes for incoming HTTP requests. Indeed, the Vert.x core API allows to start HTTP servers and listen for incoming connections, but it does not provide any facility to, say, have different handlers depending on the requested URL or processing request bodies. This is the role of a router as it dispatches requests to different processing handlers depending on the URL, the HTTP method, etc.
The initialization consists in setting up a request router, then starting the HTTP server:
private FreeMarkerTemplateEngine templateEngine;
private Future<Void> startHttpServer() {
Promise<Void> promise = Promise.promise();
HttpServer server = vertx.createHttpServer(); (1)
Router router = Router.router(vertx); (2)
router.get("/").handler(this::indexHandler);
router.get("/wiki/:page").handler(this::pageRenderingHandler); (3)
router.post().handler(BodyHandler.create()); (4)
router.post("/save").handler(this::pageUpdateHandler);
router.post("/create").handler(this::pageCreateHandler);
router.post("/delete").handler(this::pageDeletionHandler);
templateEngine = FreeMarkerTemplateEngine.create(vertx);
server
.requestHandler(router) (5)
.listen(8080, ar -> { (6)
if (ar.succeeded()) {
LOGGER.info("HTTP server running on port 8080");
promise.complete();
} else {
LOGGER.error("Could not start a HTTP server", ar.cause());
promise.fail(ar.cause());
}
});
return promise.future();
}
The
vertx
context object provides methods to create HTTP servers, clients, TCP/UDP servers and clients, etc.The
Router
class comes fromvertx-web
:io.vertx.ext.web.Router
.Routes have their own handlers, and they can be defined by URL and/or by HTTP method. For short handlers a Java lambda is an option, but for more elaborate handlers it is a good idea to reference private methods instead. Note that URLs can be parametric:
/wiki/:page
will match a request like/wiki/Hello
, in which case apage
parameter will be available with valueHello
.This makes all HTTP POST requests go through a first handler, here
io.vertx.ext.web.handler.BodyHandler
. This handler automatically decodes the body from the HTTP requests (e.g., form submissions), which can then be manipulated as Vert.x buffer objects.The router object can be used as a HTTP server handler, which then dispatches to other handlers as defined above.