Boots your Vaadin app in embedded Jetty from your main()
method quickly and easily, without Spring.
Tired of complexity of running your app as WAR from your IDE? Tired of using Spring Boot just to start your app in a simple way? Tired of constant debugging issues? Then this project is for you. From now on, you can use the free Intellij Community to develop your projects.
Who is Vaadin Boot for? Vaadin Boot is for developers that prefer to create their own solution, DIY exactly to fit their needs. You don't like to start by including a pre-fabricated application framework, complex and abstract enough to handle hundreds of use-cases. You don't have hundreds of use-cases: you only have one. You only use what you need. There lies maximum simplicity which you own, understand and can rely on.
First Principles: you only need a servlet container to run Vaadin apps.
Vaadin Boot is published in Maven Central; simply add a dependency on it:
repositories {
mavenCentral()
}
dependencies {
implementation("com.github.mvysny.vaadin-boot:vaadin-boot:13.0")
}
Or Maven:
<project>
<dependencies>
<dependency>
<groupId>com.github.mvysny.vaadin-boot</groupId>
<artifactId>vaadin-boot</artifactId>
<version>13.0</version>
</dependency>
</dependencies>
</project>
Compatibility chart:
Vaadin-Boot version | Min Java | Servlet Spec | Supported Vaadin | Jetty | Tomcat |
---|---|---|---|---|---|
10.7+ | Java 11+ | javax.servlet | Vaadin 8-23 | 10.x | N/A |
11.x (deprecated) | Java 17+ | jakarta.servlet | Vaadin 24+ | 11.x | N/A |
12.x (deprecated) | Java 17+ | jakarta.servlet | Vaadin 24+ | 12.x | N/A |
13.x | Java 17+ | jakarta.servlet | Vaadin 24+ | 12.x | 10.1.x |
See the Vaadin-Boot Git Tags for the list of released Vaadin-Boot versions.
Then, add the following Main
class to your project:
public class Main {
public static void main(String[] args) throws Exception {
new VaadinBoot().run();
}
}
Then add an empty file named src/main/resources/webapp/ROOT
to your project -
Vaadin Boot will then serve static files from this folder.
By default, VaadinBoot listens on all interfaces; call localhostOnly()
to
only listen on localhost.
To run your ap with Tomcat, make sure to use Vaadin-Boot 13.0 or higher; then
depend on vaadin-boot-tomcat
instead on vaadin-boot
.
Important: you need to define your own servlet when running Tomcat, otherwise you'll get HTTP 404. Simply define this class in your project:
@WebServlet(urlPatterns = "/*")
class MyServlet extends VaadinServlet {}
Both are excellent choices, battle-tested in production. If you have no opinion on this, just go with Jetty.
Since we're not packaging to WAR, the src/main/webapp
folder is ignored. We need to package the webapp
folder
in a way so that it's served from the jar itself. Indeed, the directory is located at src/main/resources/webapp
.
Make sure to add an empty file named src/main/resources/webapp/ROOT
- that will allow Vaadin Boot
to quickly figure out the precise location of your webapp directory in the class loader.
Packaging webapp
folder to jar file indeed slows down the retrieval of static resources a bit.
However, this is not really a big performance problem since the browsers will cache the static resources;
this also simplifies and hardens Vaadin Boot itself since we don't have to figure out
the webapp folder location based on the app zip file structure or your dev env file system structure.
webapp
location also doesn't depend on a CWD (current working directory) which is important for Intellij IDEA: in multi-module projects,
IDEA is dumb to insist to setting CWD to the root of the project, not to the module.
At the moment it's "Do it yourself". Good start is to use Apache commons-cli or picocli.
However, certain settings can be configured via env variables / jvm args, please see the Configuration section below.
Very basic example apps using Vaadin Boot:
- Vaadin 24, Gradle: vaadin-boot-example-gradle
- Vaadin 24, Maven: vaadin-boot-example-maven
- Vaadin 24, Maven, Tomcat: vaadin-boot-example-maven-tomcat
- Vaadin 23, Gradle: vaadin-boot-example-gradle, the v23 branch.
- Vaadin 23, Maven: vaadin-boot-example-maven, the v23 branch.
- Vaadin 14, Gradle: vaadin14-boot-example-gradle
- Vaadin 14, Maven: vaadin14-boot-example-maven
- Vaadin 8, Maven: vaadin8-sampler
More advanced examples, demoing both security and SQL access:
- Vaadin 24, Gradle: vaadin-simple-security-example
- All Karibu-DSL example apps run on Vaadin-Boot, too.
Please install Java JDK 11 or higher (JDK 17+ if you're using Vaadin 24+).
Vaadin build requires node.js and npm to build the 'frontend bundle'.
However, that will happen automatically so there's nothing you need to do: Vaadin plugin will automatically download
node.js and npm for you (node.js will be downloaded and run from $HOME/.vaadin
).
Grab the sources of your app from the git repository. To run your app quickly from command-line, without having to run your IDE:
- Run
./gradlew run
(or./mvnw -C exec:java
for Maven)- Note that your app needs to use the Gradle Application plugin, or the
exec-maven-plugin
. - All example apps use those plugins automatically, you don't need to do anything.
- Note that your app needs to use the Gradle Application plugin, or the
- Your app will be running on http://localhost:8080.
To run the app from your IDE (we recommend using Intellij IDEA, the Community edition):
- Import the project into your IDE
- Run
./gradle vaadinPrepareFrontend
in the project once (or./mvnw -C vaadin:prepare-frontend
for Maven), to configure Vaadin paths. - Run/Debug the
Main
class as an application (run themain()
method). The app will use npm to download all javascript libraries (may take a long time) and will start in development mode. - Your app will be running on http://localhost:8080.
Gradle tip: Intellij will by default use Gradle to start up your app, which is slower and takes more memory. To optimize startup time, head to Intellij Settings / Build, Execution, Deployment / Build Tools / Gradle and change the "Build and run using" from Gradle to Intellij IDEA.
When deploying your app to production: see the "Production" chapter below. In short:
- Build your app in production mode, via
./gradlew clean build -Pvaadin.productionMode
or./mvnw -C clean package -Pproduction
. - The build of your app should produce a zip file; unzip the file and launch the run script.
We recommend to develop Vaadin Boot apps using an IDE instead of just a plain text editor. The IDE has huge advantages of providing auto-completion, documentation, access to library sources, debugging and hot-redeployment etc. We recommend Intellij IDEA Community Edition: download, install, then import this project into the IDEA.
Open the app in your IDE, and debug the Main
class as an application (run the main()
method in debugging mode).
Then, open your browser and hit http://localhost:8080.
This will activate two things:
- Contrary to what Vaadin says ("Java live reload unavailable"), Vaadin will automatically detect changes in your CSS/JavaScript files, will rebuild the JavaScript bundle and will reload the page to apply the new values.
- When you do changes in your java files and recompile (Ctrl+F9 in Intellij), Java will update classes in your running app. Just press F5 in your browser to reload the page and to see your changes. See below on tips to vastly improve the basic hot-redeployment support.
There are lots of pre-existing Vaadin components; you can check out the Beverage Buddy example app for more examples of component usage. You should also read the full Vaadin documentation.
The browser is a very powerful IDE which can help you debug CSS- and layout-related issue. Take your time and read slowly through the following tutorials, to get acquinted with the browser developer tools:
The default Java hot-redeployment is limited to Java method in-body code changes only. The easiest way to improve is to use JetBrainsRuntime (or JBR) which is a Java version modified for better hot-redeployment (to be precise, contains DCEVM patches). The easiest way to obtain and use JBR is:
- Open your Project Settings in IDEA, then locate the SDK "Edit" button and press it
- Click the upper
+
button, then "Download JDK". - Select JDK version 17, then the "JetBrains Runtime". Both the basic version and the JCEF version work; we recommend the basic version since JCEF version is bigger and therefore takes longer to download, and JCEF is not used by Vaadin apps.
- When running your app from Intellij, make sure to:
- Run via Intellij instead of via Gradle, see the "Build and run using" tip above
- Add the following JVM flags:
-XX:+AllowEnhancedClassRedefinition -dcevm
This will give you pretty awesome hot-redeployment capabilities, but Vaadin will still complain that
"Java live reload unavailable", and won't refresh the browser automatically. To achieve that, the
recommended way is to
follow the Live Reload documentation and add
the hotswap-agent.jar
as directed. Alternatively:
- Install a JVM which supports DCEVM+HotswapAgent (e.g. trava-jdk);
you may then need to run the app with the following VM options:
-dcevm -XX:HotswapAgent=fatjar
. - Use JRebel
Simply add the following WebListener to your project:
@WebListener
public class Bootstrap implements ServletContextListener {
private static final Logger log = LoggerFactory.getLogger(Bootstrap.class);
@Override
public void contextInitialized(ServletContextEvent sce) {
// will be called exactly once, before any request is served. Initialize your JVM singletons here.
log.info("Testapp Initialized");
}
@Override
public void contextDestroyed(ServletContextEvent sce) {
// will be called exactly once, after your server stops serving http requests, but before the JVM terminates.
log.info("Testapp shut down");
}
}
Vaadin-Boot is using the slf4j logging framework by default, and your
apps should use it too. We initially recommend you to use the SLF4J SimpleLogger
logger (use it by adding this dependency to your project: implementation("org.slf4j:slf4j-simple:2.0.6")
.
You can configure SimpleLogger with the following file (placed into src/main/resources/simplelogger.properties
):
org.slf4j.simpleLogger.defaultLogLevel = info
org.slf4j.simpleLogger.showDateTime = true
org.slf4j.simpleLogger.dateTimeFormat = yyyy-MM-dd HH:mm:ss.SSS
org.slf4j.simpleLogger.log.org.atmosphere = warn
org.slf4j.simpleLogger.log.org.eclipse.jetty = warn
org.slf4j.simpleLogger.log.org.eclipse.jetty.annotations.AnnotationParser = error
This will suppress cluttering of stdout/logs with verbose messages from Atmosphere and Jetty.
We recommend using Javalin for simplicity reasons.
- For Vaadin 24+ and jakarta.servlet: use Javalin 5.x
- For Vaadin 23- and javax.servlet: use Javalin 4.x
Add Javalin to your build script:
implementation("io.javalin:javalin:4.6.7") { // or 5.4.2 if you're on Vaadin 24
exclude(group = "org.eclipse.jetty")
exclude(group = "org.eclipse.jetty.websocket")
exclude(group = "com.fasterxml.jackson.core")
}
Then add the following class to your project:
@WebServlet(name = "MyJavalinServlet", urlPatterns = {"/rest/*"})
public class MyJavalinServlet extends HttpServlet {
private final JavalinServlet javalin = Javalin.createStandalone()
.get("/rest", ctx -> ctx.result("Hello!"))
.javalinServlet();
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
javalin.service(req, resp);
}
}
Vaadin-Boot will automatically discover the servlet and initialize it properly. To test, you can run
$ curl -v localhost:8080/rest
Testing: follow this example, to only initialize the REST servlet (no need to initialize the Vaadin servlet as well):
class MyJavalinServletTest {
private var server: Server? = null
@BeforeEach
fun startJetty() {
val ctx = WebAppContext()
ctx.baseResource = EmptyResource.INSTANCE
ctx.addServlet(MyJavalinServlet::class.java, "/rest/*")
server = Server(30123)
server!!.handler = ctx
server!!.start()
}
@AfterEach
fun stopJetty() {
server?.stop()
}
@Test
fun testRest() {
assertEquals("Hello!", URL("http://localhost:30123/rest").readText())
}
}
The simplest way is to add the @WebServlet
annotation to your servlet - it will be auto-discovered
by Jetty. Please see the Javalin example above for more details.
Another way is to add the servlets manually to the WebAppContext
. The following example registers the RESTEasy application as a servlet:
public class Main {
public static void main(String[] args) throws Exception {
new VaadinBoot() {
@Override
protected @NotNull WebAppContext createWebAppContext() throws IOException {
final WebAppContext context = super.createWebAppContext();
ServletHolder holder = new ServletHolder(new HttpServletDispatcher());
holder.setInitParameter("javax.ws.rs.Application", AnalyticsRSApplication.class.getName());
holder.setInitParameter("resteasy.scan", "true");
holder.setInitParameter("resteasy.servlet.mapping.prefix", "/report/download");
context.addServlet(holder, "/report/download/*");
return context;
}
}.withArgs(args).run();
}
}
This way is not recommended since any annotations on the servlet will be ignored.
Simply introduce a class into your project which extends VaadinServlet
, then add any necessary annotations,
for example:
@WebServlet(name = "myservlet", urlPatterns = {"/*"}, initParams = @WebInitParam(name = "foo", value = "bar"))
class MyServlet extends VaadinServlet {}
By default, Vaadin's ServletDeployer
will auto-register VaadinServlet
but it will skip
this kind of registration if there's already another servlet inheriting from VaadinServlet
.
This part documents hints for buildscripts (pom.xml
/build.gradle
) of your app. When in doubt, take a look
at the example apps mentioned above.
Vaadin-Boot runs embedded Jetty itself. Therefore, Vaadin-Boot-based apps do not use the Gretty Gradle plugin and do not package themselves as WAR files - instead the apps are packaged as a Java application: a zip file with all jar dependencies and a run script.
Simply use the Gradle Application Plugin to build your app:
plugins {
id 'java'
id 'application'
id 'com.vaadin' version '23.3.6' // or 24.0.0
}
application {
mainClassName = "com.yourapp.Main"
}
This will cause Gradle to build your app as a zip file with all dependencies and a run script. Please see the example apps for more details.
The Vaadin Gradle Plugin is used to package all JavaScript stuff into a JavaScript bundle. See the Plugin home page for more details.
Info: Eclipse+BuildShip may need a workaround in order for this project to work, please see this vaadin thread for more info. This applies to Visual Studio Code as well since it also uses Eclipse bits and BuildShip underneath - see Bug #4 for more details.
We'll use two Maven plugins: the appassembler-maven-plugin to prepare run scripts and the app; and the assembly plugin to create a zip file out of the app.
To configure the assembly plugin, create the src/main/assembly/zip.xml
file with the following contents:
<assembly xmlns="http://maven.apache.org/ASSEMBLY/2.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/ASSEMBLY/2.0.0 http://maven.apache.org/xsd/assembly-2.0.0.xsd">
<id>zip</id>
<formats>
<format>zip</format>
</formats>
<includeBaseDirectory>false</includeBaseDirectory>
<fileSets>
<fileSet>
<directory>target/appassembler</directory>
<outputDirectory>/</outputDirectory>
<fileMode>0755</fileMode>
</fileSet>
</fileSets>
</assembly>
Then configure the plugins in your pom.xml
:
<plugins>
<!-- creates an executable app -->
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>appassembler-maven-plugin</artifactId>
<version>2.1.0</version>
<configuration>
<programs>
<program>
<mainClass>com.vaadin.starter.skeleton.Main</mainClass>
<name>app</name>
</program>
</programs>
</configuration>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>assemble</goal>
</goals>
</execution>
</executions>
</plugin>
<!-- creates the "zip" distribution -->
<plugin>
<artifactId>maven-assembly-plugin</artifactId>
<version>3.2.0</version>
<configuration>
<descriptors>
<descriptor>src/main/assembly/zip.xml</descriptor>
</descriptors>
</configuration>
<executions>
<execution>
<id>make-assembly</id> <!-- this is used for inheritance merges -->
<phase>package</phase> <!-- append to the packaging phase. -->
<goals>
<goal>single</goal> <!-- goals == mojos -->
</goals>
</execution>
</executions>
</plugin>
</plugins>
You can also use the exec-maven-plugin
to run your app easily from Maven:
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>exec-maven-plugin</artifactId>
<version>1.6.0</version>
<configuration>
<mainClass>com.vaadin.starter.skeleton.Main</mainClass>
</configuration>
</plugin>
Afterwards you can run your app via mvn -C exec:java
.
By default, Vaadin is running in development mode. Vaadin will start a DevServer which detects changes done in your CSS and JavaScript files and rebuilds the JavaScript bundle automatically. This is great for development but not fit for production. Please read Deploying to Production for more details.
All Vaadin apps follow the following convention when building for production:
- Gradle: build your app with
./gradlew clean build -Pvaadin.productionMode
. Vaadin Gradle Plugin will automatically build your app in production mode and will includeflow-server-production-mode.jar
. - Maven: build your app with
mvn -C clean package -Pproduction
. You need to add theproduction
profile which handles everything correctly when activated. Please see example apps for details.
In both cases, the JavaScript bundle is built at build time as opposed at runtime with devmode. You can easily verify that your app has been built in production mode:
- When you run the app, Vaadin will log to stdout that it's running in production mode
- The
flow-server-production-mode.jar
jar file is packaged in the zip file of your app. - The
yourapp.jar/META-INF/VAADIN/config/flow-build.info.json
will say"productionMode":true
- There are JavaScript files in
yourapp.jar/META-INF/VAADIN/webapp/VAADIN/build/
(this applies to Vaadin 23+; for Vaadin 14 the file structure is a bit different)- Read more at Vaadin: The missing guide, the "production" mode.
Jetty is perfectly capable of running in production as documented on the Jetty web page.
If the dev mode isn't working in your dev env, you can enable the 'always-on' production mode:
public class Main {
public static void main(String[] args) throws Exception {
System.setProperty("vaadin.productionMode", "true");
new VaadinBoot().run();
}
}
Make sure to build your app in production mode first, before starting it.
All configuration options are exposed via a Java API on the VaadinBoot
class, e.g.
new VaadinBoot().localhostOnly().setPort(8081).run();
On top of that, the following Vaadin Boot properties are configurable via environment variables and also Java system properties:
Vaadin Boot config property | Env variable | Java system property | Example Value |
---|---|---|---|
port | SERVER_PORT | server.port | 18080 |
listen interface | SERVER_ADDRESS | server.address | localhost |
context root | SERVER_SERVLET_CONTEXT_PATH | server.servlet.context-path | /admin |
Note: Vaadin Boot 13.1 and older honored
SERVER_SERVLET_CONTEXT-PATH
instead ofSERVER_SERVLET_CONTEXT_PATH
.
You can not pass the Java system properties to your app run scripts directly, since they will be treated as
program parameters. Instead, pass them via the JAVA_OPTS
env variable (only works with script created by Gradle):
- Linux:
JAVA_OPTS=-Dserver.port=8082 ./my-app
Packaging your apps as docker images is incredibly easy. We use Docker Multi-stage builds:
- We initialize the build environment and build the app in one docker image;
- We copy the result app to a new image and throw away the build environment completely, to not clutter our production image.
Example Dockerfile
for a Gradle-based app:
# The "Build" stage. Copies the entire project into the container, into the /app/ folder, and builds it.
FROM openjdk:11 AS BUILD
COPY . /app/
WORKDIR /app/
RUN ./gradlew clean build -Pvaadin.productionMode --no-daemon --info --stacktrace
WORKDIR /app/build/distributions/
RUN ls -la
RUN tar xvf app.tar
# At this point, we have the app (executable bash scrip plus a bunch of jars) in the
# /app/build/distributions/app/ folder.
# The "Run" stage. Start with a clean image, and copy over just the app itself, omitting gradle, npm and any intermediate build files.
FROM openjdk:11
COPY --from=BUILD /app/build/distributions/app /app/
WORKDIR /app/bin
EXPOSE 8080
ENTRYPOINT ./app
You then run the following commands from terminal: first one will build the docker image, the second one will run your app in Docker:
$ docker build --no-cache -t test/yourapp:latest .
$ docker run --rm -ti -p8080:8080 test/yourapp
Please find the Dockerfile
in each of the example apps above.
To use paid Vaadin Pro components you'll need an offline server key (Vaadin License key).
You can then pass the key to the docker build
via the means of Docker build-args:
- Add the following to your
Dockerfile
:ARG offlinekey
;ENV VAADIN_OFFLINE_KEY=$offlinekey
- Build the app with
$ docker build --no-cache -t test/yourapp:latest --build-arg offlinekey='eyJra.....the_very_long_1600_character_offline_key_text_blob' .
Vaadin Boot doesn't support https at the moment. The usual setup is to have Nginx unwrap ssl and pass it through to a Boot app listening for http on localhost. The reason is that you can safely restart Nginx when there's a need to apply new certificate. There are manuals on the interwebs on:
- how to have Nginx automatically poll in newest Let's Encrypt certificates and apply them automatically;
- how to unwrap https and pass it over to a http port
In short, here are brief steps to setup Nginx+Let's Encrypt on Ubuntu machine:
- First, make sure your Vaadin Boot project is listening on localhost only. This will ensure that all requests will go through Nginx.
- Then, remove the default site:
sudo rm /etc/nginx/sites-enabled/default
- Then write your own
/etc/nginx/sites-enabled/yourapp
, something like this:
server {
location / {
proxy_pass http://localhost:8080/;
# proxy_cookie_path / /foo; # use this if you mount your app to `location /foo/`
proxy_cookie_domain localhost $host;
}
}
Reload nginx configuration (sudo systemctl reload nginx.service
) and verify that you can access your app
via http://yourserver
.
Then follow Let's Encrypt's getting started; the command-line certbot
instructions for Ubuntu worked the best for me.
sudo certbot --nginx
will download the certificates and will modify your nginx config file to use the certificates
and to automatically redirect from http to https. It will also install itself to cron, to auto-refresh the certificate.
It is very easy to test Vaadin-based apps. We will test using Karibu-Testing. The browserless testing technique has numerous advantages over testing in a browser: you don't have to start the browser, and you don't have to start Jetty to test your app:
public class MainViewTest {
@NotNull
private static final Routes routes = new Routes().autoDiscoverViews("com.example");
@BeforeAll
public static void setupApp() {
// initializes your services
new Bootstrap().contextInitialized(null);
}
@AfterAll
public static void tearDownApp() {
// stops your services
new Bootstrap().contextDestroyed(null);
}
@BeforeEach
public void setupVaadin() {
// Fakes Vaadin so that you can navigate in your app straight from your test code
MockVaadin.setup(routes);
}
@AfterEach
public void tearDownVaadin() {
// Removes all Vaadin fake objects
MockVaadin.tearDown();
}
@Test
public void smoke() {
UI.getCurrent().navigate(MainView.class);
_assertOne(MainView.class);
assertTrue(Bootstrap.initialized);
_click(_get(Button.class, spec -> spec.withCaption("Click Me")));
}
}
Please see Karibu-Testing documentation for further details.
The "Creating Vaadin App from scratch" video series:
The ideas here are taken from My Favorite Vaadin Architecture article.
The easiest way to access a SQL database is to use jdbi-orm which provides a full-blown CRUD (Create/Update/Delete) bean editing and to-database mapping. For a working example please take a look at jdbi-orm-vaadin-crud-demo.
To initialize the library and use an in-memory database for quick prototyping, create a Bootstrap
class
as described above, then initialize JDBI there:
@WebListener
public class Bootstrap implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent servletContextEvent) {
final HikariConfig hikariConfig = new HikariConfig();
hikariConfig.setJdbcUrl("jdbc:h2:mem:test;DB_CLOSE_DELAY=-1");
hikariConfig.setMinimumIdle(0);
JdbiOrm.setDataSource(new HikariDataSource(hikariConfig));
jdbi().useHandle(handle -> handle.createUpdate("create table if not exists Person (\n" +
" id bigint primary key auto_increment,\n" +
" name varchar not null,\n" +
" age integer not null,\n" +
" dateOfBirth date,\n" +
" created timestamp,\n" +
" modified timestamp,\n" +
" alive boolean,\n" +
" maritalStatus varchar" +
")").execute());
System.out.println(Person.dao.findAll());
}
@Override
public void contextDestroyed(ServletContextEvent servletContextEvent) {
JdbiOrm.destroy();
}
}
To create more tables and entities, please see jdbi-orm documentation.
When the database grows bigger, it's much better to use flyway database
migration tool to update your database schema automatically on every app startup. Please
see jdbi-orm-vaadin-crud-demo for an example.
Generally you should run Flyway on app startup from your Bootstrap
class, to automatically upgrade the database.
Then, place the migration scripts to the src/main/resources/db/migration
folder.
Note that the tests will also bootstrap the database. This way you don't have to waste your time mocking/faking data loading code - instead you can test the application as-is since you'll have everything up-and-running, including services and the database.
The Entity DAO should only contain code which makes database queries easier - it should not contain any business code. For that it's best to create a service layer in the SOA (Service Oriented Architecture) fashion, thus creating a three-tiered application:
- First tier, the web browser
- Second tier, the server-side business logic
- Third tier, the database, entities, DAOs
The simplest way is to create a Services
class with a bunch of static getters, each returning
the service instance. The advantages are that the IDE autocompletion works perfectly, the lookup is simple and fast,
and are in full control how the services are instantiated - e.g. you can add a special flag for testing and
then create a different set of services for testing if need be. It's really simple to also create stateful services:
public class MyService {
public String sayHello() {
return "Hello, " + Person.dao.findFirst();
}
}
public class MyStatefulService implements Serializable {
private Person loggedInUser;
public void login(String username, String password) {
final Person person = Person.dao.findByName(username);
person.verifyPassword(password);
loggedInUser = person;
}
}
public class Services {
public static MyService getMyService() {
return new MyService();
}
public static MyStatefulService getMyStatefulService() {
MyStatefulService service = VaadinSession.getCurrent().getAttribute(MyStatefulService.class);
if (service == null) {
service = new MyStatefulService();
VaadinSession.getCurrent().setAttribute(MyStatefulService.class, service);
}
return service;
}
}
See Vaadin Simple Security for guides and example projects on how to add security simply to your Vaadin-Boot-based app.
Simply use a Java ResourceBundle-based localization and the static tr()
function
to localize your apps. Please find more at Vaadin Localization.
To configure your app, simply read a config file from the filesystem, e.g. /etc/your-app/config.toml
.
We recommend to use TOML file via the tomlj parser,
but any of the following will do as well:
You read the config file directly to a Java bean. If the file doesn't exist, you can notify the user and use a default config file, or you may throw an exception if the configuration file is required.
You can also use a standard Java validation (or your own validate()
methods) to validate the values in
the beans. You probably already have Hibernate Validator on your classpath since you're probably using
BeanValidationBinder
(or validation in jdbi-orm): you can reuse Hibernate Validator to validate your config classes as well:
Validation.buildDefaultValidatorFactory().getValidator().validate(yourConfigBean);
You can load the configuration either in your Bootstrap @WebListener
, or in your main()
function, before vaadin-boot runs your app. If you use the latter way, you can configure Vaadin Boot
itself - the port it's running, the context root, etc. Vaadin Boot will never introduce
config loading itself - every app has different needs, and one unifying solution would lead to terrible complexity.
The Kotlin Programming Language is quickly gaining popularity, and fixes many of Java's shortcomings. You can definitely use Kotlin with your Vaadin-Boot-based apps! Please feel free to add the following libraries to your app:
- vok-orm builds on
jdbi-orm
and adds first-class support for Kotlin - karibu-dsl offers structured way of building your Vaadin components and routes.
- kotlinx.serialization to load the configuration JSON file
Many more example projects:
- Simple one-page-one-button example app: karibu-helloworld-application
- A simple database-backed one-page task list app: vaadin-kotlin-pwa
- Two-page app demoing grids and database: Beverage Buddy VoK
If you like Kotlin and you like the simplicity of the ideas above, please use the Vaadin-on-Kotlin framework which is based on the ideas above.
Running your Vaadin Boot app as a service in Linux under systemd is very easy. Make sure to build the app in production mode first.
Then, create a new user (for example myappuser
) which will run the app, and unpack the app into that user home folder.
Then, create a file named /etc/systemd/system/myapp.service
(replace myapp
with your app name) with the following contents:
[Unit]
Description=MyApp
After=network.target
StartLimitIntervalSec=0
[Service]
Type=simple
Restart=always
RestartSec=1
User=myappuser
ExecStart=/home/myappuser/app/bin/app
# Environment=JAVA_HOME=/home/myappuser/jdks/temurin-20
[Install]
WantedBy=multi-user.target
ExecStart
points to the shell script running your app. For example, when buildingvaadin-boot-example-gradle
, the shell script is revealed when you unzipbuild/distributions/vaadin-boot-example-gradle.zip
.- Optionally uncomment the
Environment
line and specify a different JVM for the app.
Done. Run:
sudo systemctl start myapp
to start the appsudo systemctl status myapp
to see the app's status and logsudo systemctl enable myapp
to make the app survive machine restartsudo journalctl -u myapp
to see the app's logsudo systemctl stop myapp
to start the app
Jetty can optionally start faster if we don't classpath-scan for resources, and instead pass in a QuickStart XML file with all resources listed. This is mandatory for native mode, since classpath scanning doesn't work in native mode.
See Jetty QuickStart Documentation for more details; see Jetty Maven plugin documentation as well on how to generate the QuickStart configuration file. Also see Issue #11.
To enable QuickStart mode, add a dependency on Jetty QuickStart: org.eclipse.jetty:jetty-quickstart:11.0.14
.
The quickstart configuration lists e.g. a list of Vaadin Routes, and therefore it's good to generate it during compile time. Unfortunately, at the moment, Maven Jetty plugin can't do that, see&vote for Jetty #9497.
Once 9497 is fixed, you will be able to add the Jetty plugin to your pom.xml
:
<plugin>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-maven-plugin</artifactId>
<version>11.0.15</version>
<configuration>
<supportedPackagings><packaging>jar</packaging></supportedPackagings>
</configuration>
</plugin>
Then run mvn jetty:effective-web-xml
and find the generated file at target/effective-web.xml
.
Workaround is to generate the config file manually. You need to start your app in order for Jetty to
perform the classpath scanning. Don't forget to run the app in production mode, otherwise
the quickstart config file will contain Vaadin dev mode stuff like DevModeStartupListener
.
public class Main {
public static void main(String[] args) throws Exception {
final boolean generateQuickstartWeb = true;
final VaadinBoot boot = new VaadinBoot() {
protected WebAppContext createWebAppContext() throws IOException {
WebAppContext ctx = super.createWebAppContext();
if (generateQuickstartWeb) {
ctx.setAttribute(QuickStartConfiguration.MODE, QuickStartConfiguration.Mode.GENERATE);
final File quickstartWeb = new File("quickstart-web.xml").getAbsoluteFile();
System.out.println("Quickstart will be generated to " + quickstartWeb);
ctx.setAttribute(QuickStartConfiguration.QUICKSTART_WEB_XML, new PathResource(quickstartWeb));
} else {
ctx.setAttribute(QuickStartConfiguration.MODE, QuickStartConfiguration.Mode.QUICKSTART);
// this way works also in native mode
final Resource quickstartXml = Objects.requireNonNull(Resource.newClassPathResource("/webapp/WEB-INF/quickstart-web.xml"));
ctx.setAttribute(QuickStartConfiguration.QUICKSTART_WEB_XML, quickstartXml);
}
return ctx;
}
};
if (!generateQuickstartWeb) {
boot.disableClasspathScanning();
}
boot.run();
}
}
Place the file here: src/main/resources/webapp/WEB-INF/quickstart-web.xml
, and commit it to git. From now on,
Jetty should read the QuickStart config and skip the classpath scanning automatically,
you just need to set the generateQuickstartWeb
flag to false.
Note: in
quickstart-web.xml
, theorg.eclipse.jetty.resources
context parameter contains full paths to jars, which makes the app not portable. You will have to delete that part of thequickstart-web.xml
file manually, otherwise Jetty will fail to start and will throw an exception.
Warning: alpha quality. Please see Issue #10 for more details.
It is possible to package a Vaadin-Boot app as a native binary, using GraalVM. In Native mode, Jetty can not perform classpath scanning, and therefore you must configure Jetty QuickStart as a prerequisite.
First, follow the Getting Started with GraalVM
and install the GraalVM on your machine. Then, add the Gradle org.graalvm.buildtools.native
plugin
to your app.
GraalVM needs a bunch of configuration files, in order to know which reflective/native calls to preserve. Run the following command as described at the Agent Documentation:
./gradlew clean build run -Pvaadin.productionMode -Pagent
The GraalVM agent will collect the resources necessary for your app to start, into
the build/native/agent-output/run/session-*/
folder. Copy all files into the src/main/resources/META-INF/native-image/
folder of your app, and commit them to git.
Note: make sure the
resource-config.json
file contains"pattern":"\\Qwebapp/WEB-INF/quickstart-web.xml\\E"
Now you're ready to run
./gradlew clean build nativeCompile -Pvaadin.productionMode
Find the native binary in the /build/native/nativeCompile/
folder.
At the moment the binary will fail to start, with the IllegalStateException("Bad Quickstart location")
exception -
please see Jetty #9514 for more details.
Current workaround is to build Jetty from sources and patch the QuickStartConfiguration
class and remove the
check at line 99.
The vaadin-boot-example-gradle example project contains preliminary support for native; see the readme file for more details.
See CONTRIBUTING