All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.github.tomakehurst.wiremock.junit5.WireMockExtension Maven / Gradle / Ivy

/*
 * Copyright (C) 2021-2024 Thomas Akehurst
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.github.tomakehurst.wiremock.junit5;

import com.github.tomakehurst.wiremock.WireMockServer;
import com.github.tomakehurst.wiremock.client.WireMock;
import com.github.tomakehurst.wiremock.core.Options;
import com.github.tomakehurst.wiremock.core.WireMockConfiguration;
import com.github.tomakehurst.wiremock.http.JvmProxyConfigurer;
import com.github.tomakehurst.wiremock.junit.DslWrapper;
import java.util.Optional;
import org.junit.jupiter.api.extension.*;
import org.junit.platform.commons.support.AnnotationSupport;

/**
 * JUnit Jupiter extension that manages a WireMock server instance's lifecycle and configuration.
 *
 * 

See http://wiremock.org/docs/junit-jupiter/ for * full documentation. */ public class WireMockExtension extends DslWrapper implements ParameterResolver, BeforeEachCallback, BeforeAllCallback, AfterEachCallback, AfterAllCallback { private final boolean configureStaticDsl; private final boolean failOnUnmatchedRequests; private final boolean isDeclarative; private Options options; private WireMockServer wireMockServer; private WireMockRuntimeInfo runtimeInfo; private boolean isNonStatic = false; private Boolean proxyMode; WireMockExtension() { configureStaticDsl = true; failOnUnmatchedRequests = false; isDeclarative = true; } /** * Constructor intended for subclasses. * *

The parameter is a builder so that we can avoid a constructor explosion or * backwards-incompatible changes when new options are added. * * @param builder a {@link com.github.tomakehurst.wiremock.junit5.WireMockExtension.Builder} * instance holding the initialisation parameters for the extension. */ protected WireMockExtension(Builder builder) { this.options = builder.options; this.configureStaticDsl = builder.configureStaticDsl; this.failOnUnmatchedRequests = builder.failOnUnmatchedRequests; this.proxyMode = builder.proxyMode; this.isDeclarative = false; } private WireMockExtension( Options options, boolean configureStaticDsl, boolean failOnUnmatchedRequests, boolean proxyMode) { this.options = options; this.configureStaticDsl = configureStaticDsl; this.failOnUnmatchedRequests = failOnUnmatchedRequests; this.proxyMode = proxyMode; this.isDeclarative = false; } /** * Alias for {@link #newInstance()} for use with custom subclasses, with a more relevant name for * that use. * * @return a new {@link com.github.tomakehurst.wiremock.junit5.WireMockExtension.Builder} * instance. */ public static Builder extensionOptions() { return newInstance(); } /** * Create a new builder for the extension. * * @return a new {@link com.github.tomakehurst.wiremock.junit5.WireMockExtension.Builder} * instance. */ public static Builder newInstance() { return new Builder(); } /** * To be overridden in subclasses in order to run code immediately after per-class WireMock setup. * * @param wireMockRuntimeInfo port numbers, base URLs and HTTPS info for the running WireMock * instance/ */ protected void onBeforeAll(WireMockRuntimeInfo wireMockRuntimeInfo) {} /** * To be overridden in subclasses in order to run code immediately after per-class WireMock setup. * * @param extensionContext the current extension context; never {@code null} * @param wireMockRuntimeInfo port numbers, base URLs and HTTPS info for the running WireMock * instance/ */ protected void onBeforeAll( ExtensionContext extensionContext, WireMockRuntimeInfo wireMockRuntimeInfo) { this.onBeforeAll(wireMockRuntimeInfo); } /** * To be overridden in subclasses in order to run code immediately after per-test WireMock setup. * * @param wireMockRuntimeInfo port numbers, base URLs and HTTPS info for the running WireMock * instance/ */ protected void onBeforeEach(WireMockRuntimeInfo wireMockRuntimeInfo) {} /** * To be overridden in subclasses in order to run code immediately after per-test cleanup of * WireMock and its associated resources. * * @param extensionContext the current extension context; never {@code null} * @param wireMockRuntimeInfo port numbers, base URLs and HTTPS info for the running WireMock * instance/ */ protected void onBeforeEach( ExtensionContext extensionContext, WireMockRuntimeInfo wireMockRuntimeInfo) { this.onBeforeEach(wireMockRuntimeInfo); } /** * To be overridden in subclasses in order to run code immediately after per-test cleanup of * WireMock and its associated resources. * * @param wireMockRuntimeInfo port numbers, base URLs and HTTPS info for the running WireMock * instance/ */ protected void onAfterEach(WireMockRuntimeInfo wireMockRuntimeInfo) {} /** * To be overridden in subclasses in order to run code immediately after per-class cleanup of * WireMock. * * @param extensionContext the current extension context; never {@code null} * @param wireMockRuntimeInfo port numbers, base URLs and HTTPS info for the running WireMock * instance/ */ protected void onAfterEach( ExtensionContext extensionContext, WireMockRuntimeInfo wireMockRuntimeInfo) { this.onAfterEach(wireMockRuntimeInfo); } /** * To be overridden in subclasses in order to run code immediately after per-class cleanup of * WireMock. * * @param wireMockRuntimeInfo port numbers, base URLs and HTTPS info for the running WireMock * instance/ */ protected void onAfterAll(WireMockRuntimeInfo wireMockRuntimeInfo) {} /** * To be overridden in subclasses in order to run code immediately after per-class cleanup of * WireMock. * * @param extensionContext the current extension context; never {@code null} * @param wireMockRuntimeInfo port numbers, base URLs and HTTPS info for the running WireMock * instance/ */ protected void onAfterAll( ExtensionContext extensionContext, WireMockRuntimeInfo wireMockRuntimeInfo) { this.onAfterAll(wireMockRuntimeInfo); } @Override public boolean supportsParameter( final ParameterContext parameterContext, final ExtensionContext extensionContext) throws ParameterResolutionException { return parameterIsWireMockRuntimeInfo(parameterContext); } @Override public Object resolveParameter( final ParameterContext parameterContext, final ExtensionContext extensionContext) throws ParameterResolutionException { if (parameterIsWireMockRuntimeInfo(parameterContext)) { return runtimeInfo; } return null; } private void startServerIfRequired(ExtensionContext extensionContext) { if (wireMockServer == null || !wireMockServer.isRunning()) { wireMockServer = new WireMockServer(resolveOptions(extensionContext)); wireMockServer.start(); runtimeInfo = new WireMockRuntimeInfo(wireMockServer); this.admin = wireMockServer; this.stubbing = wireMockServer; if (configureStaticDsl) { WireMock.configureFor(new WireMock(this)); } } } private void setAdditionalOptions(ExtensionContext extensionContext) { if (proxyMode == null) { proxyMode = extensionContext .getElement() .flatMap( annotatedElement -> AnnotationSupport.findAnnotation(annotatedElement, WireMockTest.class)) .map(WireMockTest::proxyMode) .orElse(false); } } private Options resolveOptions(ExtensionContext extensionContext) { final Options defaultOptions = WireMockConfiguration.options().dynamicPort(); return extensionContext .getElement() .flatMap( annotatedElement -> this.isDeclarative ? AnnotationSupport.findAnnotation(annotatedElement, WireMockTest.class) : Optional.empty()) .map(this::buildOptionsFromWireMockTestAnnotation) .orElseGet(() -> Optional.ofNullable(this.options).orElse(defaultOptions)); } private Options buildOptionsFromWireMockTestAnnotation(WireMockTest annotation) { WireMockConfiguration options = WireMockConfiguration.options() .port(annotation.httpPort()) .extensionScanningEnabled(annotation.extensionScanningEnabled()) .enableBrowserProxying(annotation.proxyMode()); if (annotation.httpsEnabled()) { options.httpsPort(annotation.httpsPort()); } return options; } private void stopServerIfRunning() { if (wireMockServer.isRunning()) { wireMockServer.stop(); } } private boolean parameterIsWireMockRuntimeInfo(ParameterContext parameterContext) { return parameterContext.getParameter().getType().equals(WireMockRuntimeInfo.class) && this.isDeclarative; } @Override public final void beforeAll(ExtensionContext context) throws Exception { startServerIfRequired(context); setAdditionalOptions(context); onBeforeAll(context, runtimeInfo); } @Override public final void beforeEach(ExtensionContext context) throws Exception { if (wireMockServer == null) { isNonStatic = true; startServerIfRequired(context); } else { resetToDefaultMappings(); } setAdditionalOptions(context); if (proxyMode) { JvmProxyConfigurer.configureFor(wireMockServer); } onBeforeEach(context, runtimeInfo); } @Override public final void afterAll(ExtensionContext context) throws Exception { stopServerIfRunning(); onAfterAll(context, runtimeInfo); } @Override public final void afterEach(ExtensionContext context) throws Exception { if (failOnUnmatchedRequests) { wireMockServer.checkForUnmatchedRequests(); } if (isNonStatic) { stopServerIfRunning(); } if (proxyMode) { JvmProxyConfigurer.restorePrevious(); } onAfterEach(context, runtimeInfo); } public WireMockRuntimeInfo getRuntimeInfo() { return new WireMockRuntimeInfo(wireMockServer); } public String baseUrl() { return wireMockServer.baseUrl(); } public String url(String path) { return wireMockServer.url(path); } public int getHttpsPort() { return wireMockServer.httpsPort(); } public int getPort() { return wireMockServer.port(); } public static class Builder { private Options options = WireMockConfiguration.wireMockConfig().dynamicPort(); private boolean configureStaticDsl = false; private boolean failOnUnmatchedRequests = false; private boolean proxyMode = false; public Builder options(Options options) { this.options = options; return this; } public Builder configureStaticDsl(boolean configureStaticDsl) { this.configureStaticDsl = configureStaticDsl; return this; } public Builder failOnUnmatchedRequests(boolean failOnUnmatched) { this.failOnUnmatchedRequests = failOnUnmatched; return this; } public Builder proxyMode(boolean proxyMode) { this.proxyMode = proxyMode; return this; } public WireMockExtension build() { if (proxyMode && !options.browserProxySettings().enabled() && (options instanceof WireMockConfiguration)) { ((WireMockConfiguration) options).enableBrowserProxying(true); } return new WireMockExtension(options, configureStaticDsl, failOnUnmatchedRequests, proxyMode); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy