Skip to content

MBurchard/spring-boot-kotlin-vue-starter

Repository files navigation

Spring Boot Kotlin Starter

The Beginning

The Tag v1 is more or less what https://start.spring.io/ generates for you.

I have chosen some dependencies that are quite useful for each project.

  • Spring Boot DevTools: Provides fast application restarts, LiveReload, and configurations for enhanced development experience.
  • Spring Configuration Processor: Generate metadata for developers to offer contextual help and "code completion" when working with custom configuration keys (ex.application.properties/.yml files).
  • Spring Web: Build web, including RESTful, applications using Spring MVC. Uses Apache Tomcat as the default embedded container.
  • Spring Security: Highly customizable authentication and access-control framework for Spring applications.
  • Spring Boot Actuator: Supports built in (or custom) endpoints that let you monitor and manage your application - such as application health, metrics, sessions, etc.

Due to the use of org.springframework.boot:spring-boot-starter-security the application will show a login page when opening it in the browser.
After inserting user as username and the generated security password from the console the application show the Whitelabel Error Page because no controller has been created so fare.

Prepare the Project

The Tag v2 shows some files and settings that I use basically for every project.

Editorconfig

This file is recognized by lots of IDEs and editors and leads to unified layout for every team member.

application.yml

That is a question of taste. I don't like the properties files and prefer to use YAML.

It contains default settings for Spring Actuator. Now you can try the following URLs with and without authorisation:

The application.yml also contains typical settings for the web server, like compression, HTTP2 and port.

At least I prefer to have test as the default profile when running everything.

gradle.properties

This file will contain some configuration later. For now, it just enables parallel builds.

Dependencies, or how to stay up to date

Projects live long, often much longer than you think.

The v3 changes are required in order to easily detect whether dependencies have new versions during this time and then easily adapt them if necessary.

Gradle Versions Plugin

This plugin from Ben Manes is very useful and in my projects for a long time now.

The lines 45 - 65 in the build.gradle.kts show my default configuration for the plugin.

The usage of the Gradle task dependencyUpdates is straight forward.

build.gradle.kts

Other changes in that file are also simple. The javaVersion variable is included to avoid having two positions where it needs to be changed in case.

The Gradle wrapper task is helpful to easily change the used gradle wrapper. This can also be done via command line, but I prefer it the clicky way.

settings.gradle.kts

The additions in this file are responsible for the plugin versions used in the whole project. This is not only helpful when using the defined version from the gradle.properties but also in a multi module project.

v4: Who is at the door?

It's often interesting to know who's knocking at the door and where you've just arrived.
For this purpose, I use the method whoIsKnocking in the DefaultController and the Mustache template knock.mustache in my applications.
But first things first...

Logging

Most developers want to know what their application is doing. Kotlin logging is recommended for that.

So you need the dependency in the build.gradle.kts line 13 and 41 and the version in gradle.properties line 9.

The logger will be imported above the class definition as on can see in AcmeApp.kt line 9.

Logging itself will be done using a lambda expression.

log.info { "Version: ${buildProperties.version}" }

The good thing is, you don't need to worry about the computing costs of that logging. Because of its lambda nature it is not calculated if the loglevel isn't info. So you don't need to deal with placeholders like known from SLF4J, simply use the variables where you want them.

For debugging you need to set the loglevel in the application.yml.

logging:
  level:
    de.mbur.acme: debug

Spring Boot Build Info

One can provide information from the build at runtime.
Add the following lines to your build.gradle.kts:

springBoot {
  buildInfo()
}

And use this like every other Spring bean like shown in the AcmeApp.kt.

A bit of security configuration

Since it is needed anyway, we introduce the WebSecurityConfiguration here, which inherits from WebSecurityConfigurerAdapter.
The annotations used are typical. The Spring documentation reveals more about this.

The function configure(http: HttpSecurity) has to be overridden.
To be able to use the handy HttpSecurityDsl you need this import:

import org.springframework.security.config.web.servlet.invoke
  override fun configure(http: HttpSecurity) {
    http {
      authorizeRequests {
        authorize("/knock", permitAll)
        authorize(anyRequest, authenticated)
      }
      exceptionHandling { authenticationEntryPoint = HttpStatusEntryPoint(HttpStatus.UNAUTHORIZED) }
    }
  }

For simplicity, we allow access to the /knock path. Everything else only gets authenticated access.

Spring responds with status 403 if one is not authenticated, so we add the line with exceptionHandling, now Spring responds with status 401 which has quite a different meaning.

You Shall Not Pass!

The version 5 brings in a helpful and friendly gatekeeper.

As I watch my projects out there in the evil world, I see the same rather stupid attempts to exploit security vulnerabilities over and over again.

These are usually intercepted by Spring somewhere in the filter chain anyway or lead to errors that hopefully cannot be exploited, but that is actually too late for me.
I want to block malicious attempts as early as possible to avoid wasting computing time unnecessarily.

For this I have hired the Gatekeeper.

It shows different approaches that are more or less effektiv.

For one, you can simply check a list of URLs that the application offers and reject everything else.
Line 167 to 170 show how simple this is:

if (!allowedUrls.matches(req)) {
  buildResponse(resp, responseNotFound, HttpServletResponse.SC_NOT_FOUND)
  return
}

Because it is difficult to tell at this point whether these were malicious requests, there is no appropriate reaction in the example here.

The lines 171 and following, on the other hand, show how to protect the path against all that stupid PHP and ASP stuff and all the other nasty things that may be requested.

The Gatekeeper shows also how to collect information and show it regularly. Caches are used for this purpose.

Some utility functions make the whole thing complete.

Last but not least, the gatekeeper also has error handling for typical errors that can occur in a Spring application. These are not necessarily programming errors, but for example wrong HTTP methods, wrong parameters, like requests using a string instead of a long parameter.

A MockMVC test shows how to test the Gatekeeper.

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published