- Spring Boot 2 MVC
- Works with JVM 8 and JVM 11
- Ready for JdbcTemplate-style repositories + ready PostgreSQL setup
- Database migrations with Flyway
- Spring Security email/password login + signup
- Thymeleaf templates for views + Layout dialect
- Sending emails and creating them using Thymeleaf templates
- Setup for unit testing with MockMvcTest, RepositoryTest, EmailTest, and normal JUnit4 tests for service layer
- Fast UI testing with Fluentlenium and HtmlUnit with FeatureTest
- Frontend module for your stylesheets with SCSS+Bootstrap4 included
- Example application with a few features implemented
- Run the Demo app locally
- Running the tests
- Familiarizing yourself with demo app structure
- Controller-Service-Repository pattern
- Removing the demo app code
- Removing the login-signup code if you do not need it
- Setting up the database
- Setting up the database and email for deployment
First make sure, you have the quizzy and quizzy_test databases in your local PostgreSQL installation:
# needed to run the demo app locally createdb quizzy createuser quizzy # needed to run tests createdb quizzy_test createuser quizzy_testTo run the app locally, you’ll need to activate the dev spring profile. For that provide the environment variable:
export SPRING_PROFILES_ACTIVE=dev ./gradlew bootRunOnce the app is done booting, you can visit localhost:8080 to see that it works.
Activating the dev profile gives you the following:
- When editing static files and Thymeleaf templates there is no need to restart the server, this allows for quicker development.
- When sending e-mails there is no need to provide real SMTP config, instead all e-mails will be just logged in the STDOUT of the running server (
./gradlew bootRun). - When uploading pictures, local filesystem will be used, instead of any 3rd party service.
To understand this better, take a look at application-dev.yml, and search the source code for the occurrence of @Profile("dev") and @Profile("dev", "test").
Alternatively, you can run the application from the IntelliJ IDEA. For that go to Application.kt and run the main function. This will fail because some spring beans will be missing.
You’ll need to set the dev spring profile. To do that, go to Run configurations -> Edit configurations -> Kotlin -> app.ApplicationKt, then:
- Hit the
Save Configurationbutton. - Check the
Single instance onlycheckbox. - Go to
Environment Variablesdialog. - Add
SPRING_PROFILES_ACTIVEvariable with valuedev. - Hit
OKbutton.
Now you should be able to run the application from IntelliJ IDEA.
You can run all the tests with Gradle:
./gradlew testIf your setup is correct, then all the tests should pass.
Alternatively, you can run tests in IntelliJ IDEA by selecting the directory src -> test -> kotlin and choosing Run 'Tests' in 'kotlin' from the context menu or by pressing the hot key to run the current selection (for Mac: CMD+SHIFT+R, for Linux/Win: CTRL+SHIFT+R).
If you want to run specific package or class, you can do that as well in IntelliJ.
Let’s begin from the top level:
PROJECT/ frontend/ this is where SCSS+Bootstrap4 stylesheets live src/ this is where your back-end application lives build.gradle this is where you define your dependencies with Gradle Now, let’s dive into the structure of the production code for the back-end application:
src/ main/ kotlin/ app/ Application.kt this is our Application class—entrypoint to the app config/ config package contains general configuration of the web app email/ email package contains code helping you send emails util/ various helper functions and classes needed throughout the codebase auth/ auth package contains security, login, signup, and logout concerns quiz/ example demo application code [can be safely removed before you start] resources/ application.yml your main application configuration file [edit to your liking] application-dev.yml your local development configuration file application-cloud.yml.example copy this file to application-cloud.yml and fill in the blanks [for deployment] db/ migration/ this package contains Flyway migration files static/ (soft-link) this soft-link allows back-end to “see” the files generated by frontend module translations/ messages*.properties these files contain translations for different languages templates/ layouts/ this package contains Thymeleaf layouts (using layout dialect) emails/ this package contains Thymeleaf email templates, render them with EmailTemplate helper auth/ this package contains login, signup and logout related templates quizzes/ this package contains demo app’s templates The unit test side mirrors this structure exactly; more interesting are the feature tests:
src/ test/ kotlin/ app/ auth/ email/ quiz/ featuretests/ this is where all UI/feature tests live auth/ feature tests for login, signup and logout quiz/ feature tests for demo application helpers/ FeatureTest class that provides default feature test configuration EmailTest class that provides default email test configuration MockMvcTest class that provides a standalone mock mvc controller test configuration RepositoryTest class that provides default JdbcTemplate repository test configuration templates/ emails/ this package contains unit tests for email templates using EmailTemplate helper Finally, let’s take a look at the front-end structure:
frontend/ package.json this is where you define all your dependencies node_modules/ this is where your front-end dependencies live, get these with `npm install` scss/ src/ this package is where your SCSS code lives index.scss your “root” file for stylesheets [run 'npm start' to compile & watch] static/ this is where compiled stylesheets end up If you take a look at the auth or quiz packages you’ll see that there is a repeating pattern:
app/ quiz/ QuizController [Controller] QuizService [Service] QuizRepository [Repository] .. plus some data classes .. auth/ signup/ SignupController [Controller] ConfirmController [Controller] ConfirmationLinkService [Service] ForceLoginService [Service] .. plus some data classes .. AuthService [Service] user/ UserRepository [Repository] - Controllers depend (via dependency injection) on Services, and call them.
- Controller is never calling the repository.
- Services depend (via dependency injection) on other services or repositories, and call them.
- Repositories depend only on JdbcTemplate. (You can also have JPA repositories here if you wanted, but I’ve found that they don’t scale very well, and create more trouble for you than saving in the long run, especially if you are unit-testing them).
I have found this pattern very useful on countless projects, and it is an architectural sweet spot for most of the business domains. Moreover, when these three concepts are not enough, you can always have services calling other services, thus the pattern can scale to any level of domain complexity.
To remove the demo app code, you can run a single shell-script:
./remove-demo.shIf you don’t need the classic login/signup code, you can remove it with a single shell-script:
./remove-auth.shAfter you have chosen the name for your development database (let’s pretend its name is mydbname), you’ll need to create the database and the user to access it on your local postgres installation:
createdb mydbname createuser mydbname # and you’ll need a "_test" version of the db to use in the test suite: createdb mydbname_test createuser mydbname_testNow, you’ll need to set this database name and user name in the src/main/resources/application.yml:
spring: datasource: url: jdbc:postgresql://localhost/mydbname username: mydbname password: mydbname driver-class-name: org.postgresql.DriverFinally, you’ll need to set similar values for the test environment in the src/test/resources/application-test.yml:
spring: datasource: url: jdbc:postgresql://localhost/mydbname_test username: mydbname_test password: mydbname_test driver-class-name: org.postgresql.DriverNow, once you’ve decided how you will deploy your application, you could either provide an application-cloud.yml configuration file (see example in application-cloud.yml.example), or you could supply all the required variables through the environment variables, for example:
export SPRING_DATASOURCE_URL=<your db url> export SPRING_DATASOURCE_USERNAME=<your db username> export SPRING_DATASOURCE_PASSWORD=<your db password> export SPRING_MAIL_HOST=<your smtp host> export SPRING_MAIL_PORT=<your smtp port> export SPRING_MAIL_USERNAME=<your smtp username> export SPRING_MAIL_PASSWORD=<your smtp password> export APP_AUTH_CONFIRMATION_EMAILS_FROM="Your Name <your-email@example.org>"Thank you for reading this and giving it a try.
To make me super happy you can star this repo and tweet about it!