Spring Boot Integration-testing mit Keycloak

Raphael_

Aktives Mitglied
Hi,
ich habe derzeit seit einigen Tagen ein Problem, an dem ich arbeite. Konkret versuche ich, einen Endpunkt zu testen, der über Keycloak authentifiziert wird. Vor jedem meiner Tests wird ein Keycloak Docker-Container gestartet und gemäß meiner realm-export.json-Datei konfiguriert. Das funktioniert einwandfrei. Dadurch erhalte ich ein Zugriffstoken, das von diesem Container als gültig erachtet wird.

Das Problem liegt darin, dass, wenn ich meinen "echten" Endpunkt aufrufe, das Zugriffstoken aufgrund meiner SecurityConfig-Datei an Keycloak weitergeleitet wird. Um auf mein speziell für Integrationstests konfiguriertes Keycloak zuzugreifen, habe ich eine application-test.yml erstellt, auf die die SecurityConfig während der Tests zugreift. Mein Problem ist, dass der Port, auf dem der Container gestartet wird, zufällig festgelegt wird, und ich es nicht schaffe, diesen Port in der application-test.yml anzugeben. Die Verwendung einer Variable scheint nicht zu funktionieren, da die YAML-Datei vor meinem Test ausgeführt wird, und daher erhalte ich einen Fehler, dass die Variable nicht gefunden werden kann. Es scheint auch nicht möglich zu sein, den Port für diesen Container festzulegen.

Meine Frage lautet also: Wie kann ich sicherstellen, dass meine application-test.yml den richtigen Port kennt, um das Token zu validieren?
 

Oneixee5

Top Contributor
Den Port eines Containers kannst du in einem Startscript finden und als Umgebungsvariable oder Property setzen:
Bash:
docker inspect --format='{{(index (index .NetworkSettings.Ports "${interner Port}/tcp") 0).HostPort}}' ${container id}
- interner Port könnte bei Keykloak z.B.: 8080 oder 8443 sein
-ContainerId kannst du wie üblich erkennen oder setzen

Ich mache das mit z.B. mit Postgres
Bash:
docker ps -a
CONTAINER ID   IMAGE             COMMAND                  CREATED        STATUS         PORTS                     NAMES
84ef1073eefc   postgres:latest   "docker-entrypoint.s…"   7 months ago   Up 8 minutes   0.0.0.0:55000->5432/tcp   postgres-N4ES
Das Ergebnis:
Bash:
docker inspect --format='{{(index (index .NetworkSettings.Ports "5432/tcp") 0).HostPort}}' 84ef1073eefc
Code:
55000
 

Raphael_

Aktives Mitglied
Den Port eines Containers kannst du in einem Startscript finden und als Umgebungsvariable oder Property setzen:
Bash:
docker inspect --format='{{(index (index .NetworkSettings.Ports "${interner Port}/tcp") 0).HostPort}}' ${container id}
- interner Port könnte bei Keykloak z.B.: 8080 oder 8443 sein
-ContainerId kannst du wie üblich erkennen oder setzen

Ich mache das mit z.B. mit Postgres
Bash:
docker ps -a
CONTAINER ID   IMAGE             COMMAND                  CREATED        STATUS         PORTS                     NAMES
84ef1073eefc   postgres:latest   "docker-entrypoint.s…"   7 months ago   Up 8 minutes   0.0.0.0:55000->5432/tcp   postgres-N4ES
Das Ergebnis:
Bash:
docker inspect --format='{{(index (index .NetworkSettings.Ports "5432/tcp") 0).HostPort}}' 84ef1073eefc
Code:
55000
Aber das soll ja ein Integrationtest werden, also soll das ja autmatisch passieren. Ich könnte natürlich den docker container manuell hochfahren, mir den port anschauen und ihn die yml reinschreiben, aber das ist ja nicht der Sinn des Integrationtests. Ich muss das also irgendwie automatisch vor dem Integrationtest umsetzen, oder bin ich gerade falsch?
 

Raphael_

Aktives Mitglied
Ich denke wir reden gerade etwas aneinander vorbei. Also ich setze so den Container auf:

Java:
@Container
    KeycloakContainer KEYCLOAK = new KeycloakContainer()
            .withRealmImportFile("/realm-export.json");

Dabei wird ein random Port gewählt. Das ist halt vom Framework einfach so, wahrscheinlich, dass kein bereits belegter Port verwendet wird.

Dann habe ich hier meinen Test:

Code:
@Test
    void testProtectedEndpoint() throws Exception {
        Assertions.assertTrue(KEYCLOAK.isRunning());

        String authServerUrl = KEYCLOAK.getAuthServerUrl();

        String accessToken = given()
                .contentType("application/x-www-form-urlencoded")
                .formParams(Map.of(
                        "username", "username",
                        "password", "password",
                        "grant_type", "password",
                        "client_id","test-rest-api",
                        "client_secret", "secret"
                ))
                .post(authServerUrl + "/realms/realm/protocol/openid-connect/token")
                .then().assertThat().statusCode(200)
                .extract().path("access_token");
        System.out.println(accessToken);




        String requestBody = "{\"folderName\": \"User2\", \"rootFolder\": null}";

        mockMvc.perform(MockMvcRequestBuilders
                        .post("http://localhost:444/api/v1/folder")
                        .content(requestBody)
                        .contentType(MediaType.APPLICATION_JSON)
                        .header(HttpHeaders.AUTHORIZATION, "Bearer " + accessToken))

                .andExpect(MockMvcResultMatchers.status().isOk());
    }

Dabei hole ich mir mit dem oberen Block den token, den ich brauche um auf den unteren Endpoint zuzugreifen.
Und genau da liegt das Problem, denn die Anfrage an den Endpoint durchläuft diese SecurityConfig file:

Code:
@Bean
    public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception {
        http
                .csrf()
                .disable()
                .authorizeHttpRequests(auth -> auth
                        .anyRequest()
                        .authenticated()
                )
                .exceptionHandling()
                .authenticationEntryPoint((request, response, authException) -> response.sendError(HttpStatus.UNAUTHORIZED.value(), "Connection failed"));

        http
                .oauth2ResourceServer(oauth2 -> oauth2.jwt(jwtConfigurer -> jwtConfigurer.jwtAuthenticationConverter(jwtAuthConverter)));
        http
                .sessionManagement()
                .sessionCreationPolicy(STATELESS);

        return http.build();
    }


Wie ihr sehen könnt wird die Anfrage authentifiziert und um die url meines keycloaks (ich meine die, die vom Container, den ich oben erstellt habe) anzugeben habe ich eine application-test.yml. Diese sieht folgendermaßen aus:
Code:
  security:
    oauth2:
      resourceserver:
        jwt:
          issuer-uri: http://localhost:RandomPortDenIchNichtKenne/realms/realm
          jwk-set-uri: ${spring.security.oauth2.resourceserver.jwt.issuer-uri}/protocol/openid-connect/certs


der Port den ich bei issuer-uri angegeben habe kenne ich nicht, weshalb ich ihn RandomPortDenIchNichtKenne genannt habe. Hier müsste eigentlich der Port rein, unter dem der Container für den Integrationtest läuft. Da dieser aber random ist kann ich hier keinen Wert reinschreiben und mithilfe einer Variable kann ich das auch nicht lösen, da die yml vor meinem test "ausgeführt" wird und somit bekomme ich dann natürlich einen Fehler, dass die Variable nicht gefunden werden kann.
Ich hoffe das hat das Ganze etwas klarer gemacht.
 

Marinek

Bekanntes Mitglied
1. Möglichkeit: Spring Boot / Keycloak mitteilen, welcher Port verwendet werden soll. (bevorzugte Variante)

Wenn man statt zu Trollen versuchen würde dem Thread zu folgen, so haben wir gerade und nachweislich diese Option aus sehr guten Gründen ausgeschlossen.

2. Möglichkeit: network: host für den Service in Docker verwenden, und den Service, wie von @Oneixee5 beschrieben, "anpingen".

Nicht notwendig. Der Container sagt einem die korrekte url und den Port.

Eine andere möglichkeit gibt es nicht. In Docker gibt es keine Möglichkeit, um alle Serviceports automatisch zu binden

Was auch immer diese Aussage bedeutet. Aber mehr als unqualifiziert ist es nicht.
 

Marinek

Bekanntes Mitglied
Ich finde, besonders Anfängern, sollten keine falschen Antworten gegeben werden.

Witzig, dass du dich aktuell damit beschäftigst Buchstaben zu zählen in einem Wort, und das nach 10 Jahren Informatik Foren trollen. Und dann denkst der TO hier sei Anfänger. Oder irgendjemand, der hier mithilft das Problem zu lösen.

Das Level wirfst du in 100 Jahren nicht erreichen.

Du musst dir echt dieses sinnbild mit dem Geisterfahrer ergoogeln.

Immerhin sind die anderen schlauer und ignorieren deine Postings. Das muss dich wahnsinnig machen.
 

Raphael_

Aktives Mitglied
Siehe hier: https://github.com/dasniko/testcontainers-keycloak#usage-in-your-application-framework-tests

Ich habe sowas noch nicht benutzt bisher, daher lese ich das gerade, was das für dein Problem bedeutet.

Aber Tenor ist: Du musst keine Portangabe tätigen.

Da ist auch ein ausführliches --Buch-- (sorry 2023) Youtube Video verlinkt: Und hier spricht er glaube ich dein Problem an:

Hi, also das Video, war eigentlich genau das, was ich gebraucht habe, also danke dafür. Dennoch habe ich noch Probleme, da er das dort mit Quarkus umsetzt und ich Spring Boot verwende.
Ich habe meinen Test jetzt so aufgebaut:

Java:
@ContextConfiguration(classes = KeycloakResource.class)
@SpringBootTest
@ActiveProfiles("test")
class FolderControllerTest{


    @Value("${spring.oidc.auth-server-url}")
    String keycloakAuthServerUrl;


    @Test
    void testProtectedEndpoint() throws Exception {
        System.out.println(keycloakAuthServerUrl + "/protocol/openid-connect/token");

        String accessToken = given()
                .contentType("application/x-www-form-urlencoded")
                .formParams(Map.of(
                        "username", "username",
                        "password", "password",
                        "grant_type", "password",
                        "client_id","test-rest-api",
                        "client_secret", "secret"
                ))
                .post( keycloakAuthServerUrl + "/realms/realm/protocol/openid-connect/token")
                .then().assertThat().statusCode(200)
                .extract().path("access_token");
        System.out.println(accessToken);

Ich versuche derzeit nur den access Token korrekt zu erhalten (Den eigentlichen Endpoint habe ich hier jetzt mal weggelassen)

Dann habe ich wie im Video gezeigt wurde so eine KeycloakResource file erstellt, die vor dem Test ausgeführt wird und die url in der application-test.properties überschreiben soll (Ja ich habe für Testzwecke jetzt einfach mal auf eine .properties Datei gesetzt). Dort vermute ich liegt der Fehler, da es mir nicht gelungen ist den Code des Videos auf Spring Boot abzuändern.
Das Ganze sieht so aus:
Code:
@ActiveProfiles("test")
@ContextConfiguration()
public class KeycloakResource {

    @Container
    public static GenericContainer<?> keycloakContainer = new GenericContainer<>("jboss/keycloak");

    public static class Initializer implements ApplicationContextInitializer<ConfigurableApplicationContext> {
        @Override
        public void initialize(ConfigurableApplicationContext applicationContext) {
            TestPropertyValues.of(
                    "spring.oidc.auth-server-url" + keycloakContainer.getHost() + ":" + keycloakContainer.getFirstMappedPort() + "/auth/realms/realm"
            ).applyTo(applicationContext.getEnvironment());
        }
    }
}


Wie vorhin erwähnt habe ich jetzt einfach aus Testzwecken eine application-test.properties Datei gewählt, welche so aussieht:
Code:
spring.oidc.auth-server-url=http://localhost:8080/auth/realms/realm


Wenn ich den Test jetzt ausführe, und mir die url printen lasse, dann steht dort genau die url drinnen, die als Standartwert gesetzt wurde, also:
http://localhost:8080/auth/realms/realm. Es ist also ziemlich klar, dass der Code in der KeycloakResources File die url nicht überschreibt.
Wie gesagt konnte ich für Spring Boot jetzt nicht wirklich etwas finden und der Code sieht mir ehrlich gesagt eh recht suspekt aus. Also vielleicht hat ja jemand von euch noch eine Idee.
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
R Spring Boot Integration-testing mit Keycloak Frameworks - Spring, Play, Blade, Vaadin & Co 1
R Spring Boot Test Assertions mit Objekten Frameworks - Spring, Play, Blade, Vaadin & Co 6
8u3631984 Pfad zu Test Datei in application.yml in Spring Boot Test Frameworks - Spring, Play, Blade, Vaadin & Co 7
R Spring Boot sql Beziehungen Frameworks - Spring, Play, Blade, Vaadin & Co 12
M Spring Boot 3 Datenbanken zur Laufzeit Verbinden Frameworks - Spring, Play, Blade, Vaadin & Co 5
ExceptionOfExpectation In Meiner Spring-Boot Applikation verlangt die Datenbank Wert für eine ID Frameworks - Spring, Play, Blade, Vaadin & Co 5
H Spring Boot Applikation und JHM Benchmark sowie ContextConfiguration in H2 Tests ich bekomme es nicht hin Frameworks - Spring, Play, Blade, Vaadin & Co 2
ExceptionOfExpectation Tests in Spring-Boot Frameworks - Spring, Play, Blade, Vaadin & Co 4
Avalon Get Request doppelt abfeuern ohne Post Redirect Get Pattern. Spring Boot Thymeleaf MVC Frameworks - Spring, Play, Blade, Vaadin & Co 12
thor_norsk Konfigurationsprobleme mit Spring Boot Frameworks - Spring, Play, Blade, Vaadin & Co 9
thor_norsk Spring Boot Fehler Frameworks - Spring, Play, Blade, Vaadin & Co 1
thor_norsk Spring Boot und Docker Frameworks - Spring, Play, Blade, Vaadin & Co 5
D Spring Boot Test ob Validation geprüft wurde Frameworks - Spring, Play, Blade, Vaadin & Co 8
K Spring Boot OneToMany Frameworks - Spring, Play, Blade, Vaadin & Co 6
8u3631984 Spring Boot Docker Image erstellen und mit docker-compose konfigurieren Frameworks - Spring, Play, Blade, Vaadin & Co 1
gradlew.bat spring-boot:run funktioniert nicht Frameworks - Spring, Play, Blade, Vaadin & Co 4
Zrebna Spring Boot/Thymeleaf: Bestätigungsemail senden. Frameworks - Spring, Play, Blade, Vaadin & Co 2
B Spring Boot und JPA Error creating bean Frameworks - Spring, Play, Blade, Vaadin & Co 24
R Spring Boot: Warum soll PasswordEncoder in einer neuen Methode definiert sein? Frameworks - Spring, Play, Blade, Vaadin & Co 1
D Spring Boot und Microservices Frameworks - Spring, Play, Blade, Vaadin & Co 1
M Spring Boot additional Datasource for a single entity Frameworks - Spring, Play, Blade, Vaadin & Co 0
Avalon Wie sieht bei Euch das Deployment einer Spring Boot Anwendung aus? Frameworks - Spring, Play, Blade, Vaadin & Co 4
M Threads in Spring Boot Frameworks - Spring, Play, Blade, Vaadin & Co 7
W DI-Problem in Spring Boot Frameworks - Spring, Play, Blade, Vaadin & Co 4
T Spring Boot: Was bewirkt parent in maven genau? Frameworks - Spring, Play, Blade, Vaadin & Co 4
OnDemand Vaadin+Spring Boot erster Seitenload nach Neustart endlos Frameworks - Spring, Play, Blade, Vaadin & Co 0
doncarlito87 Wie erhalte ich ein JSON aus eine NativeQuery (Spring Boot)? Frameworks - Spring, Play, Blade, Vaadin & Co 8
Avalon @Query Select Abfrage liefert falsche Werte (Spring Boot, JPA, Hibernate) Frameworks - Spring, Play, Blade, Vaadin & Co 3
Avalon Erstellung Dockerimage mit spring-boot:build-image in Spring Boot mit Umgebungsvariablen Frameworks - Spring, Play, Blade, Vaadin & Co 0
D Spring Boot Field Injection in MapStruct Frameworks - Spring, Play, Blade, Vaadin & Co 5
OnDemand Spring Boot seltsame Logeinträge: Manipulationsversuche? Frameworks - Spring, Play, Blade, Vaadin & Co 2
D Spring Boot Mile Stone und Snapshot Versionen Frameworks - Spring, Play, Blade, Vaadin & Co 2
OnDemand Spring Boot Exception Body Frameworks - Spring, Play, Blade, Vaadin & Co 2
N Spring Boot - Overkill für private Projekte? Frameworks - Spring, Play, Blade, Vaadin & Co 3
K Migration eines internen Frameworks zu Spring:Boot Frameworks - Spring, Play, Blade, Vaadin & Co 0
Z Hibernate & Postgres in Spring Boot (Syntaxprobleme) Frameworks - Spring, Play, Blade, Vaadin & Co 2
Z Spring Boot mit JPA;, Hibernate, Rest & Lombok Frameworks - Spring, Play, Blade, Vaadin & Co 8
OnDemand Spring Security/Boot/Vaadin Cookie Problem bei iFrame Frameworks - Spring, Play, Blade, Vaadin & Co 4
OnDemand Spring Boot: Feld ignorieren Frameworks - Spring, Play, Blade, Vaadin & Co 3
N Buch zum Spring Framework bzw. Spring Boot Frameworks - Spring, Play, Blade, Vaadin & Co 3
OnDemand Jasypt Spring Boot HIbernate wie komme ich an den Key? Frameworks - Spring, Play, Blade, Vaadin & Co 4
8u3631984 Spring Boot im Docker Container - Logback wird nicht verwendet Frameworks - Spring, Play, Blade, Vaadin & Co 13
sascha-sphw Spring Boot Resource Server (OAuth2) @MockMvc Frameworks - Spring, Play, Blade, Vaadin & Co 5
OnDemand Spring Boot CDN Frameworks - Spring, Play, Blade, Vaadin & Co 1
OnDemand Spring Boot Deserialiserung mit JSON Frameworks - Spring, Play, Blade, Vaadin & Co 9
OnDemand Spring Boot POST zu PHP API Frameworks - Spring, Play, Blade, Vaadin & Co 12
OnDemand Spring Boot WebClient ErrorHandling Frameworks - Spring, Play, Blade, Vaadin & Co 5
8u3631984 Spring boot : Unique Constraint mit Many to Many Frameworks - Spring, Play, Blade, Vaadin & Co 2
OnDemand Spring Boot Repository null Frameworks - Spring, Play, Blade, Vaadin & Co 1
OnDemand Spring Boot Limits API calls Frameworks - Spring, Play, Blade, Vaadin & Co 20
J Spring boot mit Keyclaok rollen basiert Frameworks - Spring, Play, Blade, Vaadin & Co 2
OnDemand Spring Boot Scheduler flexibel machen Frameworks - Spring, Play, Blade, Vaadin & Co 4
OnDemand Spring Boot + Vaadin API Security Frameworks - Spring, Play, Blade, Vaadin & Co 1
S Spring Boot und seine Module verstehen Frameworks - Spring, Play, Blade, Vaadin & Co 3
R Architektur für Spring Boot Projekt Frameworks - Spring, Play, Blade, Vaadin & Co 5
M Spring Boot, Versionskontrolle und Entwicklungsumgebung und Deployment Frameworks - Spring, Play, Blade, Vaadin & Co 2
Avalon Image aus JSON Objekt im Browser anzeigen (BLOB, extrahieren, konvertieren) Spring Boot Frameworks - Spring, Play, Blade, Vaadin & Co 4
Avalon CSS Datei wird in Spring Boot mit Thymeleaf im Standardordner nicht gefunden Frameworks - Spring, Play, Blade, Vaadin & Co 3
OnDemand Spring Boot Mapping null bei GET Aufruf Frameworks - Spring, Play, Blade, Vaadin & Co 8
OnDemand Spring Boot Test mit Junit Frameworks - Spring, Play, Blade, Vaadin & Co 8
J Spring Boot - Reactor Frameworks - Spring, Play, Blade, Vaadin & Co 1
L Spring Boot, H2 und Restschnittstelle synchonisation mit Client Frameworks - Spring, Play, Blade, Vaadin & Co 10
TonioTec Spring Boot Datenbankabfrage Frameworks - Spring, Play, Blade, Vaadin & Co 7
OnDemand Spring Boot encoding Slash in URL Parameter Frameworks - Spring, Play, Blade, Vaadin & Co 7
S Spring Boot startet nicht Frameworks - Spring, Play, Blade, Vaadin & Co 3
S Spring Boot -> War File | 404 in Tomcat Frameworks - Spring, Play, Blade, Vaadin & Co 20
LimDul Spring Boot Anwendung "automatisch" neustarten Frameworks - Spring, Play, Blade, Vaadin & Co 3
A Spring boot Frameworks - Spring, Play, Blade, Vaadin & Co 18
bueseb84 Unit Test mit Spring Boot - Service Autowired Frameworks - Spring, Play, Blade, Vaadin & Co 5
OnDemand DTO <> Entity Hibernate Spring Boot Frameworks - Spring, Play, Blade, Vaadin & Co 28
OnDemand Verständnisfrage DTO Spring Boot Frameworks - Spring, Play, Blade, Vaadin & Co 19
OnDemand Spring Boot Load Balancing Frameworks - Spring, Play, Blade, Vaadin & Co 5
S Spring Boot Frameworks - Spring, Play, Blade, Vaadin & Co 1
OnDemand Spring Boot Threads Frameworks - Spring, Play, Blade, Vaadin & Co 6
S Spring Boot: Wert aus Config Datei ermitteln Frameworks - Spring, Play, Blade, Vaadin & Co 3
S Spring Boot: Ein geschütztes Verzeichnis Frameworks - Spring, Play, Blade, Vaadin & Co 1
bueseb84 Probleme mit Spring Boot Docker und Bootstrap Frameworks - Spring, Play, Blade, Vaadin & Co 9
OnDemand Spring Boot Eureka/Zuul Frameworks - Spring, Play, Blade, Vaadin & Co 0
F Redirect in einer Spring Boot WebApplication Frameworks - Spring, Play, Blade, Vaadin & Co 7
B Erstes Spring Boot Projekt: Gameserver-Manager Frameworks - Spring, Play, Blade, Vaadin & Co 5
V Spring Boot, thymeleaf, Eingaben cashen oder per session speichern? Frameworks - Spring, Play, Blade, Vaadin & Co 2
S Spring Boot Security Frameworks - Spring, Play, Blade, Vaadin & Co 3
S Spring Boot i18n Frameworks - Spring, Play, Blade, Vaadin & Co 28
bueseb84 Spring Boot : Update H2 Repository Frameworks - Spring, Play, Blade, Vaadin & Co 14
A Spring Boot/ OneToMany Relation Frameworks - Spring, Play, Blade, Vaadin & Co 29
S Spring Boot testen mit Service Klassen Frameworks - Spring, Play, Blade, Vaadin & Co 4
L Controller Spring Boot mit Java Frameworks - Spring, Play, Blade, Vaadin & Co 20
J Spring Boot H2 Datbase Frameworks - Spring, Play, Blade, Vaadin & Co 2
J Spring Boot Thymleaf mit Java.Optional Frameworks - Spring, Play, Blade, Vaadin & Co 0
L JavaFX Anwendung mit Spring Boot und mehrere FXML Forms Frameworks - Spring, Play, Blade, Vaadin & Co 5
P Mit Maven (und Spring-boot) ein FatJAR und ein EAR bauen Frameworks - Spring, Play, Blade, Vaadin & Co 3
OnDemand Spring Boot Basic Authentication langsam Frameworks - Spring, Play, Blade, Vaadin & Co 2
OnDemand Start Stop Spring Boot Frameworks - Spring, Play, Blade, Vaadin & Co 4
H Spring Boot - Dependency Injection Frameworks - Spring, Play, Blade, Vaadin & Co 26
B Java Spring Boot - POM-Problem Frameworks - Spring, Play, Blade, Vaadin & Co 8
OnDemand Spring Boot parallele Requests Frameworks - Spring, Play, Blade, Vaadin & Co 1
OnDemand Spring Boot Resttemplate exchange XML Frameworks - Spring, Play, Blade, Vaadin & Co 6
S Root-Pfad zu meiner Spring Boot Application Frameworks - Spring, Play, Blade, Vaadin & Co 3
OnDemand Spring Boot Error Logging Frameworks - Spring, Play, Blade, Vaadin & Co 5
OnDemand Spring Boot Transactions Frameworks - Spring, Play, Blade, Vaadin & Co 10

Ähnliche Java Themen

Neue Themen


Oben