use @Slf4j (+logback) for logging instead of System.out/err.println (#165)
Co-authored-by: Michael Hoennig <michael@hoennig.de> Reviewed-on: https://dev.hostsharing.net/hostsharing/hs.hsadmin.ng/pulls/165 Reviewed-by: Marc Sandlus <marc.sandlus@hostsharing.net>
This commit is contained in:
		| @@ -66,6 +66,7 @@ dependencies { | |||||||
|     implementation 'org.springframework.boot:spring-boot-starter-validation' |     implementation 'org.springframework.boot:spring-boot-starter-validation' | ||||||
|     implementation 'org.springframework.boot:spring-boot-starter-actuator' |     implementation 'org.springframework.boot:spring-boot-starter-actuator' | ||||||
|     implementation 'org.springframework.boot:spring-boot-starter-security' |     implementation 'org.springframework.boot:spring-boot-starter-security' | ||||||
|  |     implementation 'org.springdoc:springdoc-openapi-starter-webmvc-ui:2.8.5' | ||||||
|     implementation 'com.github.gavlyukovskiy:datasource-proxy-spring-boot-starter:1.10.0' |     implementation 'com.github.gavlyukovskiy:datasource-proxy-spring-boot-starter:1.10.0' | ||||||
|     implementation 'org.postgresql:postgresql' |     implementation 'org.postgresql:postgresql' | ||||||
|     implementation 'org.liquibase:liquibase-core' |     implementation 'org.liquibase:liquibase-core' | ||||||
| @@ -76,7 +77,6 @@ dependencies { | |||||||
|     implementation 'net.java.dev.jna:jna:5.16.0' |     implementation 'net.java.dev.jna:jna:5.16.0' | ||||||
|     implementation 'org.modelmapper:modelmapper:3.2.2' |     implementation 'org.modelmapper:modelmapper:3.2.2' | ||||||
|     implementation 'org.iban4j:iban4j:3.2.10-RELEASE' |     implementation 'org.iban4j:iban4j:3.2.10-RELEASE' | ||||||
|     implementation 'org.springdoc:springdoc-openapi-starter-webmvc-ui:2.8.5' |  | ||||||
|     implementation 'org.reflections:reflections:0.10.2' |     implementation 'org.reflections:reflections:0.10.2' | ||||||
|  |  | ||||||
|     compileOnly 'org.projectlombok:lombok' |     compileOnly 'org.projectlombok:lombok' | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ package net.hostsharing.hsadminng.config; | |||||||
|  |  | ||||||
| import io.micrometer.core.annotation.Timed; | import io.micrometer.core.annotation.Timed; | ||||||
| import lombok.SneakyThrows; | import lombok.SneakyThrows; | ||||||
|  | import lombok.extern.slf4j.Slf4j; | ||||||
| import org.springframework.beans.factory.annotation.Value; | import org.springframework.beans.factory.annotation.Value; | ||||||
| import org.springframework.security.authentication.BadCredentialsException; | import org.springframework.security.authentication.BadCredentialsException; | ||||||
| import org.springframework.util.LinkedMultiValueMap; | import org.springframework.util.LinkedMultiValueMap; | ||||||
| @@ -14,6 +15,8 @@ import javax.xml.parsers.DocumentBuilderFactory; | |||||||
| import javax.xml.parsers.ParserConfigurationException; | import javax.xml.parsers.ParserConfigurationException; | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
|  |  | ||||||
|  | // HOWTO add logger | ||||||
|  | @Slf4j | ||||||
| public class RealCasAuthenticator implements CasAuthenticator { | public class RealCasAuthenticator implements CasAuthenticator { | ||||||
|  |  | ||||||
|     @Value("${hsadminng.cas.server}") |     @Value("${hsadminng.cas.server}") | ||||||
| @@ -32,7 +35,8 @@ public class RealCasAuthenticator implements CasAuthenticator { | |||||||
|                 ? fetchServiceTicket(ticket) |                 ? fetchServiceTicket(ticket) | ||||||
|                 : ticket; |                 : ticket; | ||||||
|         final var userName = extractUserName(verifyServiceTicket(serviceTicket)); |         final var userName = extractUserName(verifyServiceTicket(serviceTicket)); | ||||||
|         System.err.println("CAS-user: " + userName); |         // HOWTO log some message for a certain log level (trace, debug, info, warn, error) | ||||||
|  |         log.debug("CAS-user: {}", userName); | ||||||
|         return userName; |         return userName; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -65,9 +69,7 @@ public class RealCasAuthenticator implements CasAuthenticator { | |||||||
|     private String extractUserName(final Document verification) { |     private String extractUserName(final Document verification) { | ||||||
|  |  | ||||||
|         if (verification.getElementsByTagName("cas:authenticationSuccess").getLength() == 0) { |         if (verification.getElementsByTagName("cas:authenticationSuccess").getLength() == 0) { | ||||||
|             System.err.println("CAS service ticket could not be validated"); |             throwBadCredentialsException("CAS service ticket could not be verified"); | ||||||
|             System.err.println(verification); |  | ||||||
|             throwBadCredentialsException("CAS service ticket could not be validated"); |  | ||||||
|         } |         } | ||||||
|         return verification.getElementsByTagName("cas:user").item(0).getTextContent(); |         return verification.getElementsByTagName("cas:user").item(0).getTextContent(); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -28,6 +28,7 @@ import java.util.regex.Pattern; | |||||||
| import static net.hostsharing.hsadminng.errors.CustomErrorResponse.*; | import static net.hostsharing.hsadminng.errors.CustomErrorResponse.*; | ||||||
|  |  | ||||||
| @ControllerAdvice | @ControllerAdvice | ||||||
|  | // HOWTO handle exceptions to produce specific http error codes and sensible error messages | ||||||
| public class RestResponseEntityExceptionHandler | public class RestResponseEntityExceptionHandler | ||||||
|         extends ResponseEntityExceptionHandler { |         extends ResponseEntityExceptionHandler { | ||||||
|  |  | ||||||
|   | |||||||
| @@ -121,10 +121,4 @@ public final class HashGenerator { | |||||||
|         } |         } | ||||||
|         return withSalt(stringBuilder.toString()); |         return withSalt(stringBuilder.toString()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static void main(String[] args) { |  | ||||||
|         System.out.println( |  | ||||||
|             HashGenerator.using(Algorithm.LINUX_YESCRYPT).withRandomSalt().hash("my plaintext domain transfer passphrase") |  | ||||||
|         ); |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -125,10 +125,4 @@ public class Dns { | |||||||
|             return Result.fromException(exception); |             return Result.fromException(exception); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static void main(String[] args) { |  | ||||||
|         final var result = new Dns("example.org").fetchRecordsOfType("TXT"); |  | ||||||
|         System.out.println(result); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1261,16 +1261,12 @@ public class RbacSpec { | |||||||
|                         m -> isStatic(m.getModifiers()) && m.getName().equals("main") |                         m -> isStatic(m.getModifiers()) && m.getName().equals("main") | ||||||
|                 ) |                 ) | ||||||
|                 .findFirst() |                 .findFirst() | ||||||
|                 .orElse(null); |                 .orElseThrow(() -> new RuntimeException("no main method in: " + c.getName() + " => cannot generate RBAC rules")); | ||||||
|         if (mainMethod != null) { |  | ||||||
|         try { |         try { | ||||||
|             mainMethod.invoke(null, new Object[] { null }); |             mainMethod.invoke(null, new Object[] { null }); | ||||||
|         } catch (IllegalAccessException | InvocationTargetException e) { |         } catch (IllegalAccessException | InvocationTargetException e) { | ||||||
|             throw new RuntimeException(e); |             throw new RuntimeException(e); | ||||||
|         } |         } | ||||||
|         } else { |  | ||||||
|             System.err.println("WARNING: no main method in: " + c.getName() + " => no RBAC rules generated"); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static Set<Class<? extends BaseEntity>> findRbacEntityClasses(String packageName) { |     public static Set<Class<? extends BaseEntity>> findRbacEntityClasses(String packageName) { | ||||||
|   | |||||||
| @@ -202,6 +202,5 @@ public class RbacViewMermaidFlowchartGenerator { | |||||||
|                     .replace("%{flowchart}", flowchart.toString()) |                     .replace("%{flowchart}", flowchart.toString()) | ||||||
|                     .replace("%{case}", forCase == null ? "" : " " + forCase), |                     .replace("%{case}", forCase == null ? "" : " " + forCase), | ||||||
|                 StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING); |                 StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING); | ||||||
|         System.out.println("Markdown-File: " + path.toAbsolutePath()); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -49,6 +49,5 @@ public class RbacViewPostgresGenerator { | |||||||
|                 toString(), |                 toString(), | ||||||
|                 StandardOpenOption.CREATE, |                 StandardOpenOption.CREATE, | ||||||
|                 StandardOpenOption.TRUNCATE_EXISTING); |                 StandardOpenOption.TRUNCATE_EXISTING); | ||||||
|         System.out.println(outputPath.toAbsolutePath()); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -358,9 +358,6 @@ class RolesGrantsAndPermissionsGenerator { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private String roleRef(final PostgresTriggerReference rootRefVar, final RbacSpec.RbacRoleDefinition roleDef) { |     private String roleRef(final PostgresTriggerReference rootRefVar, final RbacSpec.RbacRoleDefinition roleDef) { | ||||||
|         if (roleDef == null) { |  | ||||||
|             System.out.println("null"); |  | ||||||
|         } |  | ||||||
|         if (roleDef.getEntityAlias().isGlobal()) { |         if (roleDef.getEntityAlias().isGlobal()) { | ||||||
|             return "rbac.global_ADMIN()"; |             return "rbac.global_ADMIN()"; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -68,8 +68,10 @@ metrics: | |||||||
|                 server: |                 server: | ||||||
|                     requests: true |                     requests: true | ||||||
|  |  | ||||||
|  | # HOWTO set logging-levels for certain Java packages (trace, debug, info, warn, error) | ||||||
| logging: | logging: | ||||||
|     level: |     level: | ||||||
|         org: |         org.springframework.security: info | ||||||
|             springframework: | # HOWTO configure logging, e.g. logging to a separate file, see: | ||||||
|                 security: TRACE | #   https://docs.spring.io/spring-boot/reference/features/logging.html | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,11 +1,16 @@ | |||||||
| package net.hostsharing.hsadminng.config; | package net.hostsharing.hsadminng.config; | ||||||
|  |  | ||||||
| import com.github.tomakehurst.wiremock.WireMockServer; | import com.github.tomakehurst.wiremock.WireMockServer; | ||||||
|  | import net.hostsharing.hsadminng.test.LogbackLogPattern; | ||||||
| import org.junit.jupiter.api.Tag; | import org.junit.jupiter.api.Tag; | ||||||
| import org.junit.jupiter.api.Test; | import org.junit.jupiter.api.Test; | ||||||
|  | import org.junit.jupiter.api.extension.ExtendWith; | ||||||
| import org.springframework.beans.factory.annotation.Autowired; | import org.springframework.beans.factory.annotation.Autowired; | ||||||
| import org.springframework.beans.factory.annotation.Value; | import org.springframework.beans.factory.annotation.Value; | ||||||
|  | import org.springframework.boot.logging.LogLevel; | ||||||
| import org.springframework.boot.test.context.SpringBootTest; | import org.springframework.boot.test.context.SpringBootTest; | ||||||
|  | import org.springframework.boot.test.system.CapturedOutput; | ||||||
|  | import org.springframework.boot.test.system.OutputCaptureExtension; | ||||||
| import org.springframework.boot.test.web.client.TestRestTemplate; | import org.springframework.boot.test.web.client.TestRestTemplate; | ||||||
| import org.springframework.http.HttpEntity; | import org.springframework.http.HttpEntity; | ||||||
| import org.springframework.http.HttpMethod; | import org.springframework.http.HttpMethod; | ||||||
| @@ -13,15 +18,23 @@ import org.springframework.http.HttpStatus; | |||||||
| import org.springframework.test.context.ActiveProfiles; | import org.springframework.test.context.ActiveProfiles; | ||||||
| import org.springframework.test.context.TestPropertySource; | import org.springframework.test.context.TestPropertySource; | ||||||
|  |  | ||||||
|  | import static com.github.tomakehurst.wiremock.client.WireMock.aResponse; | ||||||
|  | import static com.github.tomakehurst.wiremock.client.WireMock.get; | ||||||
|  | import static com.github.tomakehurst.wiremock.client.WireMock.urlEqualTo; | ||||||
| import static net.hostsharing.hsadminng.config.HttpHeadersBuilder.headers; | import static net.hostsharing.hsadminng.config.HttpHeadersBuilder.headers; | ||||||
| import static org.apache.commons.lang3.RandomStringUtils.randomAlphanumeric; | import static org.apache.commons.lang3.RandomStringUtils.randomAlphanumeric; | ||||||
| import static org.assertj.core.api.Assertions.assertThat; | import static org.assertj.core.api.Assertions.assertThat; | ||||||
| import static com.github.tomakehurst.wiremock.client.WireMock.*; |  | ||||||
|  |  | ||||||
| @SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT) | @SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT) | ||||||
| @TestPropertySource(properties = {"server.port=0", "hsadminng.cas.server=http://localhost:8088"}) | @TestPropertySource(properties = { | ||||||
|  |         "server.port=0", | ||||||
|  |         "hsadminng.cas.server=http://localhost:8088", | ||||||
|  |  | ||||||
|  |         "logging.level.root=DEBUG" | ||||||
|  | }) | ||||||
| @ActiveProfiles({"wiremock", "realCasAuthenticator"}) // IMPORTANT: To test prod config, do NOT use test profile! | @ActiveProfiles({"wiremock", "realCasAuthenticator"}) // IMPORTANT: To test prod config, do NOT use test profile! | ||||||
| @Tag("generalIntegrationTest") | @Tag("generalIntegrationTest") | ||||||
|  | @ExtendWith(OutputCaptureExtension.class) | ||||||
| class CasAuthenticationFilterIntegrationTest { | class CasAuthenticationFilterIntegrationTest { | ||||||
|  |  | ||||||
|     @Value("${local.server.port}") |     @Value("${local.server.port}") | ||||||
| @@ -37,7 +50,7 @@ class CasAuthenticationFilterIntegrationTest { | |||||||
|     private WireMockServer wireMockServer; |     private WireMockServer wireMockServer; | ||||||
|  |  | ||||||
|     @Test |     @Test | ||||||
|     public void shouldAcceptRequestWithValidCasTicket() { |     public void shouldAcceptRequestWithValidCasTicket(final CapturedOutput capturedOutput) { | ||||||
|         // given |         // given | ||||||
|         final var username = "test-user-" + randomAlphanumeric(4); |         final var username = "test-user-" + randomAlphanumeric(4); | ||||||
|         wireMockServer.stubFor(get(urlEqualTo("/cas/p3/serviceValidate?service=" + serviceUrl + "&ticket=ST-valid")) |         wireMockServer.stubFor(get(urlEqualTo("/cas/p3/serviceValidate?service=" + serviceUrl + "&ticket=ST-valid")) | ||||||
| @@ -63,6 +76,9 @@ class CasAuthenticationFilterIntegrationTest { | |||||||
|         // then |         // then | ||||||
|         assertThat(result.getStatusCode()).isEqualTo(HttpStatus.OK); |         assertThat(result.getStatusCode()).isEqualTo(HttpStatus.OK); | ||||||
|         assertThat(result.getBody()).isEqualTo("pong " + username + "\n"); |         assertThat(result.getBody()).isEqualTo("pong " + username + "\n"); | ||||||
|  |         // HOWTO assert log messages | ||||||
|  |         assertThat(capturedOutput.getOut()).containsPattern( | ||||||
|  |                 LogbackLogPattern.of(LogLevel.DEBUG, RealCasAuthenticator.class, "CAS-user: " + username)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Test |     @Test | ||||||
|   | |||||||
| @@ -39,8 +39,8 @@ class RestResponseEntityExceptionHandlerUnitTest { | |||||||
|         final var errorResponse = exceptionHandler.handleConflict(givenException, givenWebRequest); |         final var errorResponse = exceptionHandler.handleConflict(givenException, givenWebRequest); | ||||||
|  |  | ||||||
|         // then |         // then | ||||||
|         assertThat(errorResponse.getBody().getStatusCode()).isEqualTo(409); |         assertThat(errorResponse.getBody()).extracting(CustomErrorResponse::getStatusCode).isEqualTo(409); | ||||||
|         assertThat(errorResponse.getBody().getMessage()).isEqualTo("ERROR: [409] First Line"); |         assertThat(errorResponse.getBody()).extracting(CustomErrorResponse::getMessage).isEqualTo("ERROR: [409] First Line"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Test |     @Test | ||||||
|   | |||||||
| @@ -0,0 +1,21 @@ | |||||||
|  | package net.hostsharing.hsadminng.test; | ||||||
|  |  | ||||||
|  | import ch.qos.logback.classic.pattern.TargetLengthBasedClassNameAbbreviator; | ||||||
|  | import org.springframework.boot.logging.LogLevel; | ||||||
|  |  | ||||||
|  | public class LogbackLogPattern { | ||||||
|  |  | ||||||
|  |     private static final int LOGBACK_MAX_CLASSNAME_LENGTH = 36; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @return a regular expression for a log message | ||||||
|  |      */ | ||||||
|  |     public static CharSequence of( | ||||||
|  |             final LogLevel logLevel, | ||||||
|  |             final Class<?> loggingClass, | ||||||
|  |             final String message) { | ||||||
|  |         final var abbreviator = new TargetLengthBasedClassNameAbbreviator(LOGBACK_MAX_CLASSNAME_LENGTH); | ||||||
|  |         final var shortenedClassName = abbreviator.abbreviate(loggingClass.getName()); | ||||||
|  |         return logLevel + " [0-9]+ .* " + shortenedClassName +  " *: " + message; | ||||||
|  |     } | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user